Rename conflicting wires on flatten/techmap, add "hierconn" attribute, fixes #1220

Signed-off-by: Clifford Wolf <clifford@clifford.at>
This commit is contained in:
Clifford Wolf 2019-09-05 13:51:53 +02:00
parent 694a8f75cf
commit 30f1ac7ce9
2 changed files with 27 additions and 8 deletions

View File

@ -332,6 +332,9 @@ Verilog Attributes and non-standard features
that represent module parameters or localparams (when the HDL front-end that represent module parameters or localparams (when the HDL front-end
is run in ``-pwires`` mode). is run in ``-pwires`` mode).
- Wires marked with the ``hierconn`` attribute are connected to wires with the
same name when they are imported from sub-modules by ``flatten``.
- The ``clkbuf_driver`` attribute can be set on an output port of a blackbox - The ``clkbuf_driver`` attribute can be set on an output port of a blackbox
module to mark it as a clock buffer output, and thus prevent ``clkbufmap`` module to mark it as a clock buffer output, and thus prevent ``clkbufmap``
from inserting another clock buffer on a net driven by such output. from inserting another clock buffer on a net driven by such output.

View File

@ -205,6 +205,7 @@ struct TechmapWorker
} }
std::map<RTLIL::IdString, RTLIL::IdString> positional_ports; std::map<RTLIL::IdString, RTLIL::IdString> positional_ports;
dict<Wire*, IdString> temp_renamed_wires;
for (auto &it : tpl->wires_) { for (auto &it : tpl->wires_) {
if (it.second->port_id > 0) if (it.second->port_id > 0)
@ -213,15 +214,20 @@ struct TechmapWorker
apply_prefix(cell->name, w_name); apply_prefix(cell->name, w_name);
RTLIL::Wire *w = module->wire(w_name); RTLIL::Wire *w = module->wire(w_name);
if (w != nullptr) { if (w != nullptr) {
if (!flatten_mode) if (!flatten_mode || !w->get_bool_attribute(ID(hierconn))) {
log_error("Signal %s.%s conflicts with %s.%s (via %s.%s).\n", log_id(module), log_id(w), temp_renamed_wires[w] = w->name;
log_id(tpl), log_id(it.second), log_id(module), log_id(cell)); module->rename(w, NEW_ID);
if (GetSize(w) < GetSize(it.second)) { w = nullptr;
log_warning("Widening signal %s.%s to match size of %s.%s (via %s.%s).\n", log_id(module), log_id(w), } else {
log_id(tpl), log_id(it.second), log_id(module), log_id(cell)); w->attributes.erase(ID(hierconn));
w->width = GetSize(it.second); if (GetSize(w) < GetSize(it.second)) {
log_warning("Widening signal %s.%s to match size of %s.%s (via %s.%s).\n", log_id(module), log_id(w),
log_id(tpl), log_id(it.second), log_id(module), log_id(cell));
w->width = GetSize(it.second);
}
} }
} else { }
if (w == nullptr) {
w = module->addWire(w_name, it.second); w = module->addWire(w_name, it.second);
w->port_input = false; w->port_input = false;
w->port_output = false; w->port_output = false;
@ -392,6 +398,16 @@ struct TechmapWorker
} }
module->remove(cell); module->remove(cell);
for (auto &it : temp_renamed_wires)
{
Wire *w = it.first;
IdString name = it.second;
IdString altname = module->uniquify(name);
Wire *other_w = module->wire(name);
module->rename(other_w, altname);
module->rename(w, name);
}
} }
bool techmap_module(RTLIL::Design *design, RTLIL::Module *module, RTLIL::Design *map, std::set<RTLIL::Cell*> &handled_cells, bool techmap_module(RTLIL::Design *design, RTLIL::Module *module, RTLIL::Design *map, std::set<RTLIL::Cell*> &handled_cells,