opt_share: Fix handling of fine cells.
authorMarcin Kościelnicki <mwk@0x04.net>
Tue, 26 Nov 2019 23:46:21 +0000 (00:46 +0100)
committerMarcin Kościelnicki <mwk@0x04.net>
Wed, 27 Nov 2019 07:01:07 +0000 (08:01 +0100)
Fixes #1525.

passes/opt/opt_share.cc
tests/opt/bug1525.ys [new file with mode: 0644]

index 2c456705cf878767de131fa97ad52039306905e4..f59f978a6d9672ac691fa5f50d897cfc0cf83377 100644 (file)
@@ -83,7 +83,9 @@ struct ExtSigSpec {
        bool operator==(const ExtSigSpec &other) const { return is_signed == other.is_signed && sign == other.sign && sig == other.sig && semantics == other.semantics; }
 };
 
-#define BITWISE_OPS ID($_AND_), ID($_NAND_), ID($_OR_), ID($_NOR_), ID($_XOR_), ID($_XNOR_), ID($_ANDNOT_), ID($_ORNOT_), ID($and), ID($or), ID($xor), ID($xnor)
+#define FINE_BITWISE_OPS ID($_AND_), ID($_NAND_), ID($_OR_), ID($_NOR_), ID($_XOR_), ID($_XNOR_), ID($_ANDNOT_), ID($_ORNOT_)
+
+#define BITWISE_OPS FINE_BITWISE_OPS, ID($and), ID($or), ID($xor), ID($xnor)
 
 #define REDUCTION_OPS ID($reduce_and), ID($reduce_or), ID($reduce_xor), ID($reduce_xnor), ID($reduce_bool), ID($reduce_nand)
 
@@ -250,14 +252,19 @@ void merge_operators(RTLIL::Module *module, RTLIL::Cell *mux, const std::vector<
                shared_op->setPort(ID(CO), alu_co.extract(0, conn_width));
        }
 
-       shared_op->setParam(ID(Y_WIDTH), conn_width);
+       bool is_fine = shared_op->type.in(FINE_BITWISE_OPS);
+
+       if (!is_fine)
+               shared_op->setParam(ID(Y_WIDTH), conn_width);
 
        if (decode_port(shared_op, ID::A, &assign_map) == operand) {
                shared_op->setPort(ID::B, mux_to_oper);
-               shared_op->setParam(ID(B_WIDTH), max_width);
+               if (!is_fine)
+                       shared_op->setParam(ID(B_WIDTH), max_width);
        } else {
                shared_op->setPort(ID::A, mux_to_oper);
-               shared_op->setParam(ID(A_WIDTH), max_width);
+               if (!is_fine)
+                       shared_op->setParam(ID(A_WIDTH), max_width);
        }
 }
 
diff --git a/tests/opt/bug1525.ys b/tests/opt/bug1525.ys
new file mode 100644 (file)
index 0000000..972bc0a
--- /dev/null
@@ -0,0 +1,13 @@
+read_verilog << EOF
+module top(...);
+input A1, A2, B, S;
+output O;
+
+assign O = S ? (A1 & B) : (A2 & B);
+
+endmodule
+EOF
+
+simplemap
+opt_share
+dump