3
0
Fork 0
mirror of https://github.com/YosysHQ/yosys synced 2025-04-06 01:24:10 +00:00

clockgate: EN can be a bit on a multi-bit wire

This commit is contained in:
Emil J. Tywoniak 2024-09-11 19:18:25 +02:00
parent 8b464341c2
commit 1e999a3cb7
2 changed files with 38 additions and 13 deletions

View file

@ -73,20 +73,20 @@ struct ClockgatePass : public Pass {
// if the number of FFs associated with it is sufficent // if the number of FFs associated with it is sufficent
struct ClkNetInfo { struct ClkNetInfo {
// Original, ungated clock into enabled FF // Original, ungated clock into enabled FF
Wire* clk_net; SigBit clk_bit;
// Original clock enable into enabled FF // Original clock enable into enabled FF
Wire* ce_net; SigBit ce_bit;
bool pol_clk; bool pol_clk;
bool pol_ce; bool pol_ce;
unsigned int hash() const { unsigned int hash() const {
auto t = std::make_tuple(clk_net, ce_net, pol_clk, pol_ce); auto t = std::make_tuple(clk_bit, ce_bit, pol_clk, pol_ce);
unsigned int h = mkhash_init; unsigned int h = mkhash_init;
h = mkhash(h, hash_ops<decltype(t)>::hash(t)); h = mkhash(h, hash_ops<decltype(t)>::hash(t));
return h; return h;
} }
bool operator==(const ClkNetInfo& other) const { bool operator==(const ClkNetInfo& other) const {
return (clk_net == other.clk_net) && return (clk_bit == other.clk_bit) &&
(ce_net == other.ce_net) && (ce_bit == other.ce_bit) &&
(pol_clk == other.pol_clk) && (pol_clk == other.pol_clk) &&
(pol_ce == other.pol_ce); (pol_ce == other.pol_ce);
} }
@ -100,8 +100,8 @@ struct ClockgatePass : public Pass {
}; };
ClkNetInfo clk_info_from_ff(FfData& ff) { ClkNetInfo clk_info_from_ff(FfData& ff) {
Wire* clk = ff.sig_clk.as_wire(); SigBit clk = ff.sig_clk[0];
Wire* ce = ff.sig_ce.as_wire(); SigBit ce = ff.sig_ce[0];
ClkNetInfo info{clk, ce, ff.pol_clk, ff.pol_ce}; ClkNetInfo info{clk, ce, ff.pol_clk, ff.pol_ce};
return info; return info;
} }
@ -147,7 +147,12 @@ struct ClockgatePass : public Pass {
FfData ff(nullptr, cell); FfData ff(nullptr, cell);
// It would be odd to get constants, but we better handle it // It would be odd to get constants, but we better handle it
if (ff.has_ce && ff.sig_clk.is_wire() && ff.sig_ce.is_wire()) { if (ff.has_ce) {
if (!ff.sig_clk.is_bit() || !ff.sig_ce.is_bit())
continue;
if (!ff.sig_clk[0].is_wire() || !ff.sig_ce[0].is_wire())
continue;
ce_ffs.insert(cell); ce_ffs.insert(cell);
ClkNetInfo info = clk_info_from_ff(ff); ClkNetInfo info = clk_info_from_ff(ff);
@ -159,7 +164,6 @@ struct ClockgatePass : public Pass {
} }
for (auto& clk_net : clk_nets) { for (auto& clk_net : clk_nets) {
log_debug("checking clk net %s\n", clk_net.first.clk_net->name.c_str());
auto& clk = clk_net.first; auto& clk = clk_net.first;
auto& gclk = clk_net.second; auto& gclk = clk_net.second;
@ -176,10 +180,9 @@ struct ClockgatePass : public Pass {
if (!matching_icg_desc) if (!matching_icg_desc)
continue; continue;
log_debug("building ICG for clk net %s\n", clk_net.first.clk_net->name.c_str());
Cell* icg = module->addCell(NEW_ID, matching_icg_desc->name); Cell* icg = module->addCell(NEW_ID, matching_icg_desc->name);
icg->setPort(matching_icg_desc->ce_pin, clk.ce_net); icg->setPort(matching_icg_desc->ce_pin, clk.ce_bit);
icg->setPort(matching_icg_desc->clk_in_pin, clk.clk_net); icg->setPort(matching_icg_desc->clk_in_pin, clk.clk_bit);
gclk.new_net = module->addWire(NEW_ID); gclk.new_net = module->addWire(NEW_ID);
icg->setPort(matching_icg_desc->clk_out_pin, gclk.new_net); icg->setPort(matching_icg_desc->clk_out_pin, gclk.new_net);
// Tie low DFT ports like scan chain enable // Tie low DFT ports like scan chain enable
@ -187,7 +190,7 @@ struct ClockgatePass : public Pass {
icg->setPort(port, Const(0)); icg->setPort(port, Const(0));
// Fix CE polarity if needed // Fix CE polarity if needed
if (!clk.pol_ce) { if (!clk.pol_ce) {
SigBit ce_fixed_pol = module->NotGate(NEW_ID, clk.ce_net); SigBit ce_fixed_pol = module->NotGate(NEW_ID, clk.ce_bit);
icg->setPort(matching_icg_desc->ce_pin, ce_fixed_pol); icg->setPort(matching_icg_desc->ce_pin, ce_fixed_pol);
} }
} }

View file

@ -171,4 +171,26 @@ select -module bad2 -assert-count 0 t:\\pdk_icg
#------------------------------------------------------------------------------ #------------------------------------------------------------------------------
# Regression test: EN is a bit from a multi-bit wire
design -reset
read_verilog << EOT
module dffe_wide_11( input clk, input [1:0] en,
input [3:0] d1, output reg [3:0] q1,
);
always @( posedge clk ) begin
if ( en[0] )
q1 <= d1;
end
endmodule
EOT
proc
opt
clockgate -pos pdk_icg ce:clkin:clkout -tie_lo scanen
select -assert-count 1 t:\\pdk_icg
#------------------------------------------------------------------------------
# TODO test -tie_lo # TODO test -tie_lo