pattern dffmux state cemuxAB rstmuxBA state sigD match dff select dff->type == $dff select GetSize(port(dff, \D)) > 1 endmatch match rstmux select rstmux->type == $mux select GetSize(port(rstmux, \Y)) > 1 index port(rstmux, \Y) === port(dff, \D) choice BA {\B, \A} select port(rstmux, BA).is_fully_const() set rstmuxBA BA optional endmatch code sigD if (rstmux) sigD = port(rstmux, rstmuxBA == \B ? \A : \B); else sigD = port(dff, \D); endcode match cemux select cemux->type == $mux select GetSize(port(cemux, \Y)) > 1 index port(cemux, \Y) === sigD choice AB {\A, \B} index port(cemux, AB) === port(dff, \Q) set cemuxAB AB endmatch code SigSpec D = port(cemux, cemuxAB == \A ? \B : \A); SigSpec Q = port(dff, \Q); Const rst; if (rstmux) rst = port(rstmux, rstmuxBA).as_const(); int width = GetSize(D); if (D[width-1] == D[width-2]) { did_something = true; SigBit sign = D[width-1]; bool is_signed = sign.wire; int i; for (i = width-1; i >= 2; i--) { if (!is_signed) { module->connect(Q[i], sign); if (D[i-1] != sign || (rst.size() && rst[i-1] != rst[width-1])) break; } else { module->connect(Q[i], Q[i-1]); if (D[i-2] != sign || (rst.size() && rst[i-1] != rst[width-1])) break; } } cemux->connections_.at(\A).remove(i, width-i); cemux->connections_.at(\B).remove(i, width-i); cemux->connections_.at(\Y).remove(i, width-i); cemux->fixup_parameters(); dff->connections_.at(\D).remove(i, width-i); dff->connections_.at(\Q).remove(i, width-i); dff->fixup_parameters(); log("dffcemux pattern in %s: dff=%s, cemux=%s; removed top %d bits.\n", log_id(module), log_id(dff), log_id(cemux), width-i); accept; } else { int count = 0; for (int i = width-1; i >= 0; i--) { if (D[i].wire) continue; Wire *w = Q[i].wire; auto it = w->attributes.find(\init); State init; if (it != w->attributes.end()) init = it->second[Q[i].offset]; else init = State::Sx; if (init == State::Sx || init == D[i].data) { count++; module->connect(Q[i], D[i]); cemux->connections_.at(\A).remove(i); cemux->connections_.at(\B).remove(i); cemux->connections_.at(\Y).remove(i); dff->connections_.at(\D).remove(i); dff->connections_.at(\Q).remove(i); } } if (count > 0) { did_something = true; cemux->fixup_parameters(); dff->fixup_parameters(); log("dffcemux pattern in %s: dff=%s, cemux=%s; removed %d constant bits.\n", log_id(module), log_id(dff), log_id(cemux), count); } accept; } endcode