From ec065186d309e1f5dba05eb652d75a7a0a84778c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Martin=20Povi=C5=A1er?= Date: Tue, 9 Jan 2024 11:52:41 +0100 Subject: [PATCH 1/2] opt_clean: Add commentary around wire cleaning, NFC --- passes/opt/opt_clean.cc | 46 +++++++++++++++++++++++++++++------------ 1 file changed, 33 insertions(+), 13 deletions(-) diff --git a/passes/opt/opt_clean.cc b/passes/opt/opt_clean.cc index a219e4708..c9d85028e 100644 --- a/passes/opt/opt_clean.cc +++ b/passes/opt/opt_clean.cc @@ -240,6 +240,7 @@ int count_nontrivial_wire_attrs(RTLIL::Wire *w) return count; } +// Should we pick `s2` over `s1` to represent a signal? bool compare_signals(RTLIL::SigBit &s1, RTLIL::SigBit &s2, SigPool ®s, SigPool &conns, pool &direct_wires) { RTLIL::Wire *w1 = s1.wire; @@ -292,9 +293,10 @@ bool check_public_name(RTLIL::IdString id) bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbose) { + // `register_signals` and `connected_signals` will help us decide later on + // on picking representatives out of groups of connected signals SigPool register_signals; SigPool connected_signals; - if (!purge_mode) for (auto &it : module->cells_) { RTLIL::Cell *cell = it.second; @@ -309,20 +311,27 @@ bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbos } SigMap assign_map(module); - pool direct_sigs; + + // construct a pool of wires which are directly driven by a known celltype, + // this will influence our choice of representatives pool direct_wires; - for (auto &it : module->cells_) { - RTLIL::Cell *cell = it.second; - if (ct_all.cell_known(cell->type)) - for (auto &it2 : cell->connections()) - if (ct_all.cell_output(cell->type, it2.first)) - direct_sigs.insert(assign_map(it2.second)); - } - for (auto &it : module->wires_) { - if (direct_sigs.count(assign_map(it.second)) || it.second->port_input) - direct_wires.insert(it.second); + { + pool direct_sigs; + for (auto &it : module->cells_) { + RTLIL::Cell *cell = it.second; + if (ct_all.cell_known(cell->type)) + for (auto &it2 : cell->connections()) + if (ct_all.cell_output(cell->type, it2.first)) + direct_sigs.insert(assign_map(it2.second)); + } + for (auto &it : module->wires_) { + if (direct_sigs.count(assign_map(it.second)) || it.second->port_input) + direct_wires.insert(it.second); + } } + // weight all options for representatives with `compare_signals`, + // the one that wins will be what `assign_map` maps to for (auto &it : module->wires_) { RTLIL::Wire *wire = it.second; for (int i = 0; i < wire->width; i++) { @@ -332,21 +341,30 @@ bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbos } } + // we are removing all connections module->connections_.clear(); + // used signals sigmapped SigPool used_signals; + // used signals pre-sigmapped SigPool raw_used_signals; + // used signals sigmapped, ignoring drivers (we keep track of this to set `unused_bits`) SigPool used_signals_nodrivers; + + // gather the usage information for cells for (auto &it : module->cells_) { RTLIL::Cell *cell = it.second; for (auto &it2 : cell->connections_) { - assign_map.apply(it2.second); + assign_map.apply(it2.second); // modify the cell connection in place raw_used_signals.add(it2.second); used_signals.add(it2.second); if (!ct_all.cell_output(cell->type, it2.first)) used_signals_nodrivers.add(it2.second); } } + + // gather the usage information for ports, wires with `keep`, + // also gather init bits dict init_bits; for (auto &it : module->wires_) { RTLIL::Wire *wire = it.second; @@ -374,6 +392,7 @@ bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbos } } + // set init attributes on all wires of a connected group for (auto wire : module->wires()) { bool found = false; Const val(State::Sx, wire->width); @@ -388,6 +407,7 @@ bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbos wire->attributes[ID::init] = val; } + // now decide for each wire if we should be deleting it pool del_wires_queue; for (auto wire : module->wires()) { From 7afc0696e2ffefde3ea3c92a5b22c6de1309b68e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Martin=20Povi=C5=A1er?= Date: Tue, 9 Jan 2024 11:54:38 +0100 Subject: [PATCH 2/2] opt_clean: Assert an impossible path isn't taken --- passes/opt/opt_clean.cc | 3 +++ 1 file changed, 3 insertions(+) diff --git a/passes/opt/opt_clean.cc b/passes/opt/opt_clean.cc index c9d85028e..8b56895fe 100644 --- a/passes/opt/opt_clean.cc +++ b/passes/opt/opt_clean.cc @@ -438,6 +438,9 @@ bool rmunused_module_signals(RTLIL::Module *module, bool purge_mode, bool verbos goto delete_this_wire; } else if (!used_signals.check_any(s2)) { + // this path shouldn't be possible: this wire is used directly (otherwise it would get cleaned up above), and indirectly + // used wires are a superset of those used directly + log_assert(false); // delete wires that aren't used by anything indirectly, even though other wires may alias it goto delete_this_wire; }