/* * yosys -- Yosys Open SYnthesis Suite * * Copyright (C) 2012 Clifford Wolf * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * */ #include "kernel/register.h" #include "kernel/rtlil.h" #include "kernel/log.h" struct DeleteWireWorker { RTLIL::Module *module; std::set *delete_wires_p; void operator()(RTLIL::SigSpec &sig) { sig.optimize(); for (auto &c : sig.chunks_rw()) if (c.wire != NULL && delete_wires_p->count(c.wire->name)) { c.wire = module->addWire(NEW_ID, c.width); c.offset = 0; } } }; struct DeletePass : public Pass { DeletePass() : Pass("delete", "delete objects in the design") { } virtual void help() { // |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---| log("\n"); log(" delete [selection]\n"); log("\n"); log("Deletes the selected objects. This will also remove entire modules, if the\n"); log("whole module is selected.\n"); log("\n"); log("\n"); log(" delete {-input|-output|-port} [selection]\n"); log("\n"); log("Does not delete any object but removes the input and/or output flag on the\n"); log("selected wires, thus 'deleting' module ports.\n"); log("\n"); } virtual void execute(std::vector args, RTLIL::Design *design) { bool flag_input = false; bool flag_output = false; size_t argidx; for (argidx = 1; argidx < args.size(); argidx++) { if (args[argidx] == "-input") { flag_input = true; continue; } if (args[argidx] == "-output") { flag_output = true; continue; } if (args[argidx] == "-port") { flag_input = true; flag_output = true; continue; } break; } extra_args(args, argidx, design); std::vector delete_mods; for (auto &mod_it : design->modules) { if (design->selected_whole_module(mod_it.first) && !flag_input && !flag_output) { delete_mods.push_back(mod_it.first); continue; } if (!design->selected_module(mod_it.first)) continue; RTLIL::Module *module = mod_it.second; if (flag_input || flag_output) { for (auto &it : module->wires) if (design->selected(module, it.second)) { if (flag_input) it.second->port_input = false; if (flag_output) it.second->port_output = false; } module->fixup_ports(); continue; } std::set delete_wires; std::set delete_cells; std::set delete_procs; std::set delete_mems; for (auto &it : module->wires) if (design->selected(module, it.second)) delete_wires.insert(it.first); for (auto &it : module->memories) if (design->selected(module, it.second)) delete_mems.insert(it.first); for (auto &it : module->cells) { if (design->selected(module, it.second)) delete_cells.insert(it.first); if ((it.second->type == "$memrd" || it.second->type == "$memwr") && delete_mems.count(it.second->parameters.at("\\MEMID").decode_string()) != 0) delete_cells.insert(it.first); } for (auto &it : module->processes) if (design->selected(module, it.second)) delete_procs.insert(it.first); DeleteWireWorker delete_wire_worker; delete_wire_worker.module = module; delete_wire_worker.delete_wires_p = &delete_wires; module->rewrite_sigspecs(delete_wire_worker); for (auto &it : delete_wires) { delete module->wires.at(it); module->wires.erase(it); } for (auto &it : delete_mems) { delete module->memories.at(it); module->memories.erase(it); } for (auto &it : delete_cells) { delete module->cells.at(it); module->cells.erase(it); } for (auto &it : delete_procs) { delete module->processes.at(it); module->processes.erase(it); } module->fixup_ports(); } for (auto &it : delete_mods) { delete design->modules.at(it); design->modules.erase(it); } } } DeletePass;