mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-26 09:24:37 +00:00 
			
		
		
		
	Revert "Refactor full_selection"
This commit is contained in:
		
							parent
							
								
									98d4355b82
								
							
						
					
					
						commit
						d49364d96f
					
				
					 38 changed files with 270 additions and 707 deletions
				
			
		|  | @ -1215,6 +1215,9 @@ struct FirrtlBackend : public Backend { | ||||||
| 		} | 		} | ||||||
| 		extra_args(f, filename, args, argidx); | 		extra_args(f, filename, args, argidx); | ||||||
| 
 | 
 | ||||||
|  | 		if (!design->full_selection()) | ||||||
|  | 			log_cmd_error("This command only operates on fully selected designs!\n"); | ||||||
|  | 
 | ||||||
| 		log_header(design, "Executing FIRRTL backend.\n"); | 		log_header(design, "Executing FIRRTL backend.\n"); | ||||||
| 		log_push(); | 		log_push(); | ||||||
| 
 | 
 | ||||||
|  | @ -1227,7 +1230,7 @@ struct FirrtlBackend : public Backend { | ||||||
| 		autoid_counter = 0; | 		autoid_counter = 0; | ||||||
| 
 | 
 | ||||||
| 		// Get the top module, or a reasonable facsimile - we need something for the circuit name.
 | 		// Get the top module, or a reasonable facsimile - we need something for the circuit name.
 | ||||||
| 		Module *top = nullptr; | 		Module *top = design->top_module(); | ||||||
| 		Module *last = nullptr; | 		Module *last = nullptr; | ||||||
| 		// Generate module and wire names.
 | 		// Generate module and wire names.
 | ||||||
| 		for (auto module : design->modules()) { | 		for (auto module : design->modules()) { | ||||||
|  |  | ||||||
|  | @ -304,8 +304,8 @@ void RTLIL_BACKEND::dump_conn(std::ostream &f, std::string indent, const RTLIL:: | ||||||
| 
 | 
 | ||||||
| void RTLIL_BACKEND::dump_module(std::ostream &f, std::string indent, RTLIL::Module *module, RTLIL::Design *design, bool only_selected, bool flag_m, bool flag_n) | void RTLIL_BACKEND::dump_module(std::ostream &f, std::string indent, RTLIL::Module *module, RTLIL::Design *design, bool only_selected, bool flag_m, bool flag_n) | ||||||
| { | { | ||||||
| 	bool print_header = flag_m || module->is_selected_whole(); | 	bool print_header = flag_m || design->selected_whole_module(module->name); | ||||||
| 	bool print_body = !flag_n || !module->is_selected_whole(); | 	bool print_body = !flag_n || !design->selected_whole_module(module->name); | ||||||
| 
 | 
 | ||||||
| 	if (print_header) | 	if (print_header) | ||||||
| 	{ | 	{ | ||||||
|  |  | ||||||
|  | @ -51,10 +51,10 @@ struct Test2Pass : public Pass { | ||||||
|     Test2Pass() : Pass("test2", "demonstrating sigmap on test module") { } |     Test2Pass() : Pass("test2", "demonstrating sigmap on test module") { } | ||||||
|     void execute(std::vector<std::string>, RTLIL::Design *design) override |     void execute(std::vector<std::string>, RTLIL::Design *design) override | ||||||
|     { |     { | ||||||
|         if (design->selection().empty()) |         if (design->selection_stack.back().empty()) | ||||||
|             log_cmd_error("This command can't operator on an empty selection!\n"); |             log_cmd_error("This command can't operator on an empty selection!\n"); | ||||||
| 
 | 
 | ||||||
|         RTLIL::Module *module = design->module("\\test"); |         RTLIL::Module *module = design->modules_.at("\\test"); | ||||||
| 
 | 
 | ||||||
|         RTLIL::SigSpec a(module->wire("\\a")), x(module->wire("\\x")), y(module->wire("\\y")); |         RTLIL::SigSpec a(module->wire("\\a")), x(module->wire("\\x")), y(module->wire("\\y")); | ||||||
|         log("%d %d %d\n", a == x, x == y, y == a); // will print "0 0 0"
 |         log("%d %d %d\n", a == x, x == y, y == a); // will print "0 0 0"
 | ||||||
|  |  | ||||||
|  | @ -237,7 +237,7 @@ Use ``log_cmd_error()`` to report a recoverable error: | ||||||
| 
 | 
 | ||||||
| .. code:: C++ | .. code:: C++ | ||||||
| 
 | 
 | ||||||
|     if (design->selection().empty()) |     if (design->selection_stack.back().empty()) | ||||||
|         log_cmd_error("This command can't operator on an empty selection!\n"); |         log_cmd_error("This command can't operator on an empty selection!\n"); | ||||||
| 
 | 
 | ||||||
| Use ``log_assert()`` and ``log_abort()`` instead of ``assert()`` and ``abort()``. | Use ``log_assert()`` and ``log_abort()`` instead of ``assert()`` and ``abort()``. | ||||||
|  |  | ||||||
|  | @ -111,7 +111,7 @@ void run(const char *command) | ||||||
| 		log_last_error = ""; | 		log_last_error = ""; | ||||||
| 	} catch (...) { | 	} catch (...) { | ||||||
| 		while (GetSize(yosys_get_design()->selection_stack) > selSize) | 		while (GetSize(yosys_get_design()->selection_stack) > selSize) | ||||||
| 			yosys_get_design()->pop_selection(); | 			yosys_get_design()->selection_stack.pop_back(); | ||||||
| 		throw; | 		throw; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -260,18 +260,18 @@ void Pass::call(RTLIL::Design *design, std::vector<std::string> args) | ||||||
| 	pass_register[args[0]]->execute(args, design); | 	pass_register[args[0]]->execute(args, design); | ||||||
| 	pass_register[args[0]]->post_execute(state); | 	pass_register[args[0]]->post_execute(state); | ||||||
| 	while (design->selection_stack.size() > orig_sel_stack_pos) | 	while (design->selection_stack.size() > orig_sel_stack_pos) | ||||||
| 		design->pop_selection(); | 		design->selection_stack.pop_back(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Pass::call_on_selection(RTLIL::Design *design, const RTLIL::Selection &selection, std::string command) | void Pass::call_on_selection(RTLIL::Design *design, const RTLIL::Selection &selection, std::string command) | ||||||
| { | { | ||||||
| 	std::string backup_selected_active_module = design->selected_active_module; | 	std::string backup_selected_active_module = design->selected_active_module; | ||||||
| 	design->selected_active_module.clear(); | 	design->selected_active_module.clear(); | ||||||
| 	design->push_selection(selection); | 	design->selection_stack.push_back(selection); | ||||||
| 
 | 
 | ||||||
| 	Pass::call(design, command); | 	Pass::call(design, command); | ||||||
| 
 | 
 | ||||||
| 	design->pop_selection(); | 	design->selection_stack.pop_back(); | ||||||
| 	design->selected_active_module = backup_selected_active_module; | 	design->selected_active_module = backup_selected_active_module; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -279,11 +279,11 @@ void Pass::call_on_selection(RTLIL::Design *design, const RTLIL::Selection &sele | ||||||
| { | { | ||||||
| 	std::string backup_selected_active_module = design->selected_active_module; | 	std::string backup_selected_active_module = design->selected_active_module; | ||||||
| 	design->selected_active_module.clear(); | 	design->selected_active_module.clear(); | ||||||
| 	design->push_selection(selection); | 	design->selection_stack.push_back(selection); | ||||||
| 
 | 
 | ||||||
| 	Pass::call(design, args); | 	Pass::call(design, args); | ||||||
| 
 | 
 | ||||||
| 	design->pop_selection(); | 	design->selection_stack.pop_back(); | ||||||
| 	design->selected_active_module = backup_selected_active_module; | 	design->selected_active_module = backup_selected_active_module; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -291,12 +291,12 @@ void Pass::call_on_module(RTLIL::Design *design, RTLIL::Module *module, std::str | ||||||
| { | { | ||||||
| 	std::string backup_selected_active_module = design->selected_active_module; | 	std::string backup_selected_active_module = design->selected_active_module; | ||||||
| 	design->selected_active_module = module->name.str(); | 	design->selected_active_module = module->name.str(); | ||||||
| 	design->push_empty_selection(); | 	design->selection_stack.push_back(RTLIL::Selection(false)); | ||||||
| 	design->select(module); | 	design->selection_stack.back().select(module); | ||||||
| 
 | 
 | ||||||
| 	Pass::call(design, command); | 	Pass::call(design, command); | ||||||
| 
 | 
 | ||||||
| 	design->pop_selection(); | 	design->selection_stack.pop_back(); | ||||||
| 	design->selected_active_module = backup_selected_active_module; | 	design->selected_active_module = backup_selected_active_module; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -304,12 +304,12 @@ void Pass::call_on_module(RTLIL::Design *design, RTLIL::Module *module, std::vec | ||||||
| { | { | ||||||
| 	std::string backup_selected_active_module = design->selected_active_module; | 	std::string backup_selected_active_module = design->selected_active_module; | ||||||
| 	design->selected_active_module = module->name.str(); | 	design->selected_active_module = module->name.str(); | ||||||
| 	design->push_empty_selection(); | 	design->selection_stack.push_back(RTLIL::Selection(false)); | ||||||
| 	design->select(module); | 	design->selection_stack.back().select(module); | ||||||
| 
 | 
 | ||||||
| 	Pass::call(design, args); | 	Pass::call(design, args); | ||||||
| 
 | 
 | ||||||
| 	design->pop_selection(); | 	design->selection_stack.pop_back(); | ||||||
| 	design->selected_active_module = backup_selected_active_module; | 	design->selected_active_module = backup_selected_active_module; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -651,7 +651,7 @@ void Backend::backend_call(RTLIL::Design *design, std::ostream *f, std::string f | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	while (design->selection_stack.size() > orig_sel_stack_pos) | 	while (design->selection_stack.size() > orig_sel_stack_pos) | ||||||
| 		design->pop_selection(); | 		design->selection_stack.pop_back(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct SimHelper { | struct SimHelper { | ||||||
|  |  | ||||||
							
								
								
									
										231
									
								
								kernel/rtlil.cc
									
										
									
									
									
								
							
							
						
						
									
										231
									
								
								kernel/rtlil.cc
									
										
									
									
									
								
							|  | @ -766,23 +766,8 @@ vector<int> RTLIL::AttrObject::get_intvec_attribute(const RTLIL::IdString &id) c | ||||||
| 	return data; | 	return data; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Selection::boxed_module(const RTLIL::IdString &mod_name) const |  | ||||||
| { |  | ||||||
| 	if (current_design != nullptr) { |  | ||||||
| 		auto module = current_design->module(mod_name); |  | ||||||
| 		return module && module->get_blackbox_attribute(); |  | ||||||
| 	} else { |  | ||||||
| 		log_warning("Unable to check if module is boxed for null design.\n"); |  | ||||||
| 		return false; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool RTLIL::Selection::selected_module(const RTLIL::IdString &mod_name) const | bool RTLIL::Selection::selected_module(const RTLIL::IdString &mod_name) const | ||||||
| { | { | ||||||
| 	if (complete_selection) |  | ||||||
| 		return true; |  | ||||||
| 	if (!selects_boxes && boxed_module(mod_name)) |  | ||||||
| 		return false; |  | ||||||
| 	if (full_selection) | 	if (full_selection) | ||||||
| 		return true; | 		return true; | ||||||
| 	if (selected_modules.count(mod_name) > 0) | 	if (selected_modules.count(mod_name) > 0) | ||||||
|  | @ -794,10 +779,6 @@ bool RTLIL::Selection::selected_module(const RTLIL::IdString &mod_name) const | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Selection::selected_whole_module(const RTLIL::IdString &mod_name) const | bool RTLIL::Selection::selected_whole_module(const RTLIL::IdString &mod_name) const | ||||||
| { | { | ||||||
| 	if (complete_selection) |  | ||||||
| 		return true; |  | ||||||
| 	if (!selects_boxes && boxed_module(mod_name)) |  | ||||||
| 		return false; |  | ||||||
| 	if (full_selection) | 	if (full_selection) | ||||||
| 		return true; | 		return true; | ||||||
| 	if (selected_modules.count(mod_name) > 0) | 	if (selected_modules.count(mod_name) > 0) | ||||||
|  | @ -807,10 +788,6 @@ bool RTLIL::Selection::selected_whole_module(const RTLIL::IdString &mod_name) co | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Selection::selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const | bool RTLIL::Selection::selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const | ||||||
| { | { | ||||||
| 	if (complete_selection) |  | ||||||
| 		return true; |  | ||||||
| 	if (!selects_boxes && boxed_module(mod_name)) |  | ||||||
| 		return false; |  | ||||||
| 	if (full_selection) | 	if (full_selection) | ||||||
| 		return true; | 		return true; | ||||||
| 	if (selected_modules.count(mod_name) > 0) | 	if (selected_modules.count(mod_name) > 0) | ||||||
|  | @ -823,17 +800,7 @@ bool RTLIL::Selection::selected_member(const RTLIL::IdString &mod_name, const RT | ||||||
| 
 | 
 | ||||||
| void RTLIL::Selection::optimize(RTLIL::Design *design) | void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| { | { | ||||||
| 	if (design != current_design) { | 	if (full_selection) { | ||||||
| 		current_design = design; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (selects_boxes && full_selection) |  | ||||||
| 		complete_selection = true; |  | ||||||
| 	if (complete_selection) { |  | ||||||
| 		full_selection = false; |  | ||||||
| 		selects_boxes = true; |  | ||||||
| 	} |  | ||||||
| 	if (selects_all()) { |  | ||||||
| 		selected_modules.clear(); | 		selected_modules.clear(); | ||||||
| 		selected_members.clear(); | 		selected_members.clear(); | ||||||
| 		return; | 		return; | ||||||
|  | @ -843,7 +810,7 @@ void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| 
 | 
 | ||||||
| 	del_list.clear(); | 	del_list.clear(); | ||||||
| 	for (auto mod_name : selected_modules) { | 	for (auto mod_name : selected_modules) { | ||||||
| 		if (current_design->modules_.count(mod_name) == 0 || (!selects_boxes && boxed_module(mod_name))) | 		if (design->modules_.count(mod_name) == 0) | ||||||
| 			del_list.push_back(mod_name); | 			del_list.push_back(mod_name); | ||||||
| 		selected_members.erase(mod_name); | 		selected_members.erase(mod_name); | ||||||
| 	} | 	} | ||||||
|  | @ -852,7 +819,7 @@ void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| 
 | 
 | ||||||
| 	del_list.clear(); | 	del_list.clear(); | ||||||
| 	for (auto &it : selected_members) | 	for (auto &it : selected_members) | ||||||
| 		if (current_design->modules_.count(it.first) == 0 || (!selects_boxes && boxed_module(it.first))) | 		if (design->modules_.count(it.first) == 0) | ||||||
| 			del_list.push_back(it.first); | 			del_list.push_back(it.first); | ||||||
| 	for (auto mod_name : del_list) | 	for (auto mod_name : del_list) | ||||||
| 		selected_members.erase(mod_name); | 		selected_members.erase(mod_name); | ||||||
|  | @ -860,7 +827,7 @@ void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| 	for (auto &it : selected_members) { | 	for (auto &it : selected_members) { | ||||||
| 		del_list.clear(); | 		del_list.clear(); | ||||||
| 		for (auto memb_name : it.second) | 		for (auto memb_name : it.second) | ||||||
| 			if (current_design->modules_[it.first]->count_id(memb_name) == 0) | 			if (design->modules_[it.first]->count_id(memb_name) == 0) | ||||||
| 				del_list.push_back(memb_name); | 				del_list.push_back(memb_name); | ||||||
| 		for (auto memb_name : del_list) | 		for (auto memb_name : del_list) | ||||||
| 			it.second.erase(memb_name); | 			it.second.erase(memb_name); | ||||||
|  | @ -871,8 +838,8 @@ void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| 	for (auto &it : selected_members) | 	for (auto &it : selected_members) | ||||||
| 		if (it.second.size() == 0) | 		if (it.second.size() == 0) | ||||||
| 			del_list.push_back(it.first); | 			del_list.push_back(it.first); | ||||||
| 		else if (it.second.size() == current_design->modules_[it.first]->wires_.size() + current_design->modules_[it.first]->memories.size() + | 		else if (it.second.size() == design->modules_[it.first]->wires_.size() + design->modules_[it.first]->memories.size() + | ||||||
| 				current_design->modules_[it.first]->cells_.size() + current_design->modules_[it.first]->processes.size()) | 				design->modules_[it.first]->cells_.size() + design->modules_[it.first]->processes.size()) | ||||||
| 			add_list.push_back(it.first); | 			add_list.push_back(it.first); | ||||||
| 	for (auto mod_name : del_list) | 	for (auto mod_name : del_list) | ||||||
| 		selected_members.erase(mod_name); | 		selected_members.erase(mod_name); | ||||||
|  | @ -881,24 +848,13 @@ void RTLIL::Selection::optimize(RTLIL::Design *design) | ||||||
| 		selected_modules.insert(mod_name); | 		selected_modules.insert(mod_name); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (selected_modules.size() == current_design->modules_.size()) { | 	if (selected_modules.size() == design->modules_.size()) { | ||||||
|  | 		full_selection = true; | ||||||
| 		selected_modules.clear(); | 		selected_modules.clear(); | ||||||
| 		selected_members.clear(); | 		selected_members.clear(); | ||||||
| 		if (selects_boxes) |  | ||||||
| 			complete_selection = true; |  | ||||||
| 		else |  | ||||||
| 			full_selection = true; |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void RTLIL::Selection::clear() |  | ||||||
| { |  | ||||||
| 	full_selection = false; |  | ||||||
| 	complete_selection = false; |  | ||||||
| 	selected_modules.clear(); |  | ||||||
| 	selected_members.clear(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| RTLIL::Design::Design() | RTLIL::Design::Design() | ||||||
|   : verilog_defines (new define_map_t) |   : verilog_defines (new define_map_t) | ||||||
| { | { | ||||||
|  | @ -907,7 +863,7 @@ RTLIL::Design::Design() | ||||||
| 	hashidx_ = hashidx_count; | 	hashidx_ = hashidx_count; | ||||||
| 
 | 
 | ||||||
| 	refcount_modules_ = 0; | 	refcount_modules_ = 0; | ||||||
| 	push_full_selection(); | 	selection_stack.push_back(RTLIL::Selection()); | ||||||
| 
 | 
 | ||||||
| #ifdef WITH_PYTHON | #ifdef WITH_PYTHON | ||||||
| 	RTLIL::Design::get_all_designs()->insert(std::pair<unsigned int, RTLIL::Design*>(hashidx_, this)); | 	RTLIL::Design::get_all_designs()->insert(std::pair<unsigned int, RTLIL::Design*>(hashidx_, this)); | ||||||
|  | @ -952,7 +908,7 @@ const RTLIL::Module *RTLIL::Design::module(const RTLIL::IdString& name) const | ||||||
| 	return modules_.count(name) ? modules_.at(name) : NULL; | 	return modules_.count(name) ? modules_.at(name) : NULL; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| RTLIL::Module *RTLIL::Design::top_module() const | RTLIL::Module *RTLIL::Design::top_module() | ||||||
| { | { | ||||||
| 	RTLIL::Module *module = nullptr; | 	RTLIL::Module *module = nullptr; | ||||||
| 	int module_count = 0; | 	int module_count = 0; | ||||||
|  | @ -1110,7 +1066,6 @@ void RTLIL::Design::sort() | ||||||
| void RTLIL::Design::check() | void RTLIL::Design::check() | ||||||
| { | { | ||||||
| #ifndef NDEBUG | #ifndef NDEBUG | ||||||
| 	log_assert(!selection_stack.empty()); |  | ||||||
| 	for (auto &it : modules_) { | 	for (auto &it : modules_) { | ||||||
| 		log_assert(this == it.second->design); | 		log_assert(this == it.second->design); | ||||||
| 		log_assert(it.first == it.second->name); | 		log_assert(it.first == it.second->name); | ||||||
|  | @ -1134,21 +1089,27 @@ bool RTLIL::Design::selected_module(const RTLIL::IdString& mod_name) const | ||||||
| { | { | ||||||
| 	if (!selected_active_module.empty() && mod_name != selected_active_module) | 	if (!selected_active_module.empty() && mod_name != selected_active_module) | ||||||
| 		return false; | 		return false; | ||||||
| 	return selection().selected_module(mod_name); | 	if (selection_stack.size() == 0) | ||||||
|  | 		return true; | ||||||
|  | 	return selection_stack.back().selected_module(mod_name); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Design::selected_whole_module(const RTLIL::IdString& mod_name) const | bool RTLIL::Design::selected_whole_module(const RTLIL::IdString& mod_name) const | ||||||
| { | { | ||||||
| 	if (!selected_active_module.empty() && mod_name != selected_active_module) | 	if (!selected_active_module.empty() && mod_name != selected_active_module) | ||||||
| 		return false; | 		return false; | ||||||
| 	return selection().selected_whole_module(mod_name); | 	if (selection_stack.size() == 0) | ||||||
|  | 		return true; | ||||||
|  | 	return selection_stack.back().selected_whole_module(mod_name); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Design::selected_member(const RTLIL::IdString& mod_name, const RTLIL::IdString& memb_name) const | bool RTLIL::Design::selected_member(const RTLIL::IdString& mod_name, const RTLIL::IdString& memb_name) const | ||||||
| { | { | ||||||
| 	if (!selected_active_module.empty() && mod_name != selected_active_module) | 	if (!selected_active_module.empty() && mod_name != selected_active_module) | ||||||
| 		return false; | 		return false; | ||||||
| 	return selection().selected_member(mod_name, memb_name); | 	if (selection_stack.size() == 0) | ||||||
|  | 		return true; | ||||||
|  | 	return selection_stack.back().selected_member(mod_name, memb_name); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Design::selected_module(RTLIL::Module *mod) const | bool RTLIL::Design::selected_module(RTLIL::Module *mod) const | ||||||
|  | @ -1161,86 +1122,37 @@ bool RTLIL::Design::selected_whole_module(RTLIL::Module *mod) const | ||||||
| 	return selected_whole_module(mod->name); | 	return selected_whole_module(mod->name); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void RTLIL::Design::push_selection(RTLIL::Selection sel) | std::vector<RTLIL::Module*> RTLIL::Design::selected_modules() const | ||||||
| { | { | ||||||
| 	sel.current_design = this; |  | ||||||
| 	selection_stack.push_back(sel); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void RTLIL::Design::push_empty_selection() |  | ||||||
| { |  | ||||||
| 	push_selection(RTLIL::Selection::EmptySelection(this)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void RTLIL::Design::push_full_selection() |  | ||||||
| { |  | ||||||
| 	push_selection(RTLIL::Selection::FullSelection(this)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void RTLIL::Design::push_complete_selection() |  | ||||||
| { |  | ||||||
| 	push_selection(RTLIL::Selection::CompleteSelection(this)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void RTLIL::Design::pop_selection() |  | ||||||
| { |  | ||||||
| 	selection_stack.pop_back(); |  | ||||||
| 	// Default to a full_selection if we ran out of stack
 |  | ||||||
| 	if (selection_stack.empty()) |  | ||||||
| 		push_full_selection(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::vector<RTLIL::Module*> RTLIL::Design::selected_modules(RTLIL::SelectPartials partials, RTLIL::SelectBoxes boxes) const |  | ||||||
| { |  | ||||||
| 	bool include_partials = partials == RTLIL::SELECT_ALL; |  | ||||||
| 	bool exclude_boxes = (partials & RTLIL::SB_UNBOXED_ONLY) != 0; |  | ||||||
| 	bool ignore_wb = (partials & RTLIL::SB_INCL_WB) != 0; |  | ||||||
| 	std::vector<RTLIL::Module*> result; | 	std::vector<RTLIL::Module*> result; | ||||||
| 	result.reserve(modules_.size()); | 	result.reserve(modules_.size()); | ||||||
| 	for (auto &it : modules_) | 	for (auto &it : modules_) | ||||||
| 		if (selected_whole_module(it.first) || (include_partials && selected_module(it.first))) { | 		if (selected_module(it.first) && !it.second->get_blackbox_attribute()) | ||||||
| 			if (!(exclude_boxes && it.second->get_blackbox_attribute(ignore_wb))) | 			result.push_back(it.second); | ||||||
| 				result.push_back(it.second); | 	return result; | ||||||
| 			else | } | ||||||
| 				switch (boxes) | 
 | ||||||
| 				{ | std::vector<RTLIL::Module*> RTLIL::Design::selected_whole_modules() const | ||||||
| 				case RTLIL::SB_UNBOXED_WARN: | { | ||||||
| 					log_warning("Ignoring boxed module %s.\n", log_id(it.first)); | 	std::vector<RTLIL::Module*> result; | ||||||
| 					break; | 	result.reserve(modules_.size()); | ||||||
| 				case RTLIL::SB_EXCL_BB_WARN: | 	for (auto &it : modules_) | ||||||
| 					log_warning("Ignoring blackbox module %s.\n", log_id(it.first)); | 		if (selected_whole_module(it.first) && !it.second->get_blackbox_attribute()) | ||||||
| 					break; | 			result.push_back(it.second); | ||||||
| 				case RTLIL::SB_UNBOXED_ERR: | 	return result; | ||||||
| 					log_error("Unsupported boxed module %s.\n", log_id(it.first)); | } | ||||||
| 					break; | 
 | ||||||
| 				case RTLIL::SB_EXCL_BB_ERR: | std::vector<RTLIL::Module*> RTLIL::Design::selected_whole_modules_warn(bool include_wb) const | ||||||
| 					log_error("Unsupported blackbox module %s.\n", log_id(it.first)); | { | ||||||
| 					break; | 	std::vector<RTLIL::Module*> result; | ||||||
| 				case RTLIL::SB_UNBOXED_CMDERR: | 	result.reserve(modules_.size()); | ||||||
| 					log_cmd_error("Unsupported boxed module %s.\n", log_id(it.first)); | 	for (auto &it : modules_) | ||||||
| 					break; | 		if (it.second->get_blackbox_attribute(include_wb)) | ||||||
| 				case RTLIL::SB_EXCL_BB_CMDERR: | 			continue; | ||||||
| 					log_cmd_error("Unsupported blackbox module %s.\n", log_id(it.first)); | 		else if (selected_whole_module(it.first)) | ||||||
| 					break; | 			result.push_back(it.second); | ||||||
| 				default: | 		else if (selected_module(it.first)) | ||||||
| 					break; | 			log_warning("Ignoring partially selected module %s.\n", log_id(it.first)); | ||||||
| 				} |  | ||||||
| 		} else if (!include_partials && selected_module(it.first)) { |  | ||||||
| 			switch(partials) |  | ||||||
| 			{ |  | ||||||
| 			case RTLIL::SELECT_WHOLE_WARN: |  | ||||||
| 				log_warning("Ignoring partially selected module %s.\n", log_id(it.first)); |  | ||||||
| 				break; |  | ||||||
| 			case RTLIL::SELECT_WHOLE_ERR: |  | ||||||
| 				log_error("Unsupported partially selected module %s.\n", log_id(it.first)); |  | ||||||
| 				break; |  | ||||||
| 			case RTLIL::SELECT_WHOLE_CMDERR: |  | ||||||
| 				log_cmd_error("Unsupported partially selected module %s.\n", log_id(it.first)); |  | ||||||
| 				break; |  | ||||||
| 			default: |  | ||||||
| 				break; |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -2372,13 +2284,6 @@ void RTLIL::Module::check() | ||||||
| 			log_assert(!packed_memids.count(memid)); | 			log_assert(!packed_memids.count(memid)); | ||||||
| 			packed_memids.insert(memid); | 			packed_memids.insert(memid); | ||||||
| 		} | 		} | ||||||
| 		auto cell_mod = design->module(it.first); |  | ||||||
| 		if (cell_mod != nullptr) { |  | ||||||
| 			// assertion check below to make sure that there are no
 |  | ||||||
| 			// cases where a cell has a blackbox attribute since
 |  | ||||||
| 			// that is deprecated
 |  | ||||||
| 			log_assert(!it.second->get_blackbox_attribute()); |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for (auto &it : processes) { | 	for (auto &it : processes) { | ||||||
|  | @ -2506,16 +2411,6 @@ bool RTLIL::Module::has_processes_warn() const | ||||||
| 	return !processes.empty(); | 	return !processes.empty(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RTLIL::Module::is_selected() const |  | ||||||
| { |  | ||||||
| 	return design->selected_module(this->name); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool RTLIL::Module::is_selected_whole() const |  | ||||||
| { |  | ||||||
| 	return design->selected_whole_module(this->name); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::vector<RTLIL::Wire*> RTLIL::Module::selected_wires() const | std::vector<RTLIL::Wire*> RTLIL::Module::selected_wires() const | ||||||
| { | { | ||||||
| 	std::vector<RTLIL::Wire*> result; | 	std::vector<RTLIL::Wire*> result; | ||||||
|  | @ -2536,40 +2431,6 @@ std::vector<RTLIL::Cell*> RTLIL::Module::selected_cells() const | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::vector<RTLIL::Memory*> RTLIL::Module::selected_memories() const |  | ||||||
| { |  | ||||||
| 	std::vector<RTLIL::Memory*> result; |  | ||||||
| 	result.reserve(memories.size()); |  | ||||||
| 	for (auto &it : memories) |  | ||||||
| 		if (design->selected(this, it.second)) |  | ||||||
| 			result.push_back(it.second); |  | ||||||
| 	return result; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::vector<RTLIL::Process*> RTLIL::Module::selected_processes() const |  | ||||||
| { |  | ||||||
| 	std::vector<RTLIL::Process*> result; |  | ||||||
| 	result.reserve(processes.size()); |  | ||||||
| 	for (auto &it : processes) |  | ||||||
| 		if (design->selected(this, it.second)) |  | ||||||
| 			result.push_back(it.second); |  | ||||||
| 	return result; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::vector<RTLIL::NamedObject*> RTLIL::Module::selected_members() const |  | ||||||
| { |  | ||||||
| 	std::vector<RTLIL::NamedObject*> result; |  | ||||||
| 	auto cells = selected_cells(); |  | ||||||
| 	auto memories = selected_memories(); |  | ||||||
| 	auto wires = selected_wires(); |  | ||||||
| 	auto processes = selected_processes(); |  | ||||||
| 	result.insert(result.end(), cells.begin(), cells.end()); |  | ||||||
| 	result.insert(result.end(), memories.begin(), memories.end()); |  | ||||||
| 	result.insert(result.end(), wires.begin(), wires.end()); |  | ||||||
| 	result.insert(result.end(), processes.begin(), processes.end()); |  | ||||||
| 	return result; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void RTLIL::Module::add(RTLIL::Wire *wire) | void RTLIL::Module::add(RTLIL::Wire *wire) | ||||||
| { | { | ||||||
| 	log_assert(!wire->name.empty()); | 	log_assert(!wire->name.empty()); | ||||||
|  |  | ||||||
							
								
								
									
										208
									
								
								kernel/rtlil.h
									
										
									
									
									
								
							
							
						
						
									
										208
									
								
								kernel/rtlil.h
									
										
									
									
									
								
							|  | @ -56,33 +56,8 @@ namespace RTLIL | ||||||
| 		CONST_FLAG_REAL   = 4   // only used for parameters
 | 		CONST_FLAG_REAL   = 4   // only used for parameters
 | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	enum SelectPartials : unsigned char { |  | ||||||
| 		SELECT_ALL = 0,          // include partial modules
 |  | ||||||
| 		SELECT_WHOLE_ONLY = 1,   // ignore partial modules
 |  | ||||||
| 		SELECT_WHOLE_WARN = 2,   // call log_warning on partial module
 |  | ||||||
| 		SELECT_WHOLE_ERR = 3,    // call log_error on partial module
 |  | ||||||
| 		SELECT_WHOLE_CMDERR = 4  // call log_cmd_error on partial module
 |  | ||||||
| 	}; |  | ||||||
| 
 |  | ||||||
| 	enum SelectBoxes : unsigned char { |  | ||||||
| 		SB_ALL = 0,            // include boxed modules
 |  | ||||||
| 		SB_WARN = 1,           // helper for log_warning
 |  | ||||||
| 		SB_ERR = 2,            // helper for log_error
 |  | ||||||
| 		SB_CMDERR = 3,         // helper for log_cmd_error
 |  | ||||||
| 		SB_UNBOXED_ONLY = 4,   // ignore boxed modules
 |  | ||||||
| 		SB_UNBOXED_WARN = 5,   // call log_warning on boxed module
 |  | ||||||
| 		SB_UNBOXED_ERR = 6,    // call log_error on boxed module
 |  | ||||||
| 		SB_UNBOXED_CMDERR = 7, // call log_cmd_error on boxed module
 |  | ||||||
| 		SB_INCL_WB = 8,        // helper for white boxes
 |  | ||||||
| 		SB_EXCL_BB_ONLY = 12,  // ignore black boxes, but not white boxes
 |  | ||||||
| 		SB_EXCL_BB_WARN = 13,  // call log_warning on black boxed module
 |  | ||||||
| 		SB_EXCL_BB_ERR = 14,   // call log_error on black boxed module
 |  | ||||||
| 		SB_EXCL_BB_CMDERR = 15 // call log_cmd_error on black boxed module
 |  | ||||||
| 	}; |  | ||||||
| 
 |  | ||||||
| 	struct Const; | 	struct Const; | ||||||
| 	struct AttrObject; | 	struct AttrObject; | ||||||
| 	struct NamedObject; |  | ||||||
| 	struct Selection; | 	struct Selection; | ||||||
| 	struct Monitor; | 	struct Monitor; | ||||||
| 	struct Design; | 	struct Design; | ||||||
|  | @ -894,11 +869,6 @@ struct RTLIL::AttrObject | ||||||
| 	vector<int> get_intvec_attribute(const RTLIL::IdString &id) const; | 	vector<int> get_intvec_attribute(const RTLIL::IdString &id) const; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RTLIL::NamedObject : public RTLIL::AttrObject |  | ||||||
| { |  | ||||||
| 	RTLIL::IdString name; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct RTLIL::SigChunk | struct RTLIL::SigChunk | ||||||
| { | { | ||||||
| 	RTLIL::Wire *wire; | 	RTLIL::Wire *wire; | ||||||
|  | @ -1164,94 +1134,32 @@ public: | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Selection | struct RTLIL::Selection | ||||||
| { | { | ||||||
| 	// selection includes boxed modules
 |  | ||||||
| 	bool selects_boxes; |  | ||||||
| 	// selection covers full design, including boxed modules
 |  | ||||||
| 	bool complete_selection; |  | ||||||
| 	// selection covers full design, not including boxed modules
 |  | ||||||
| 	bool full_selection; | 	bool full_selection; | ||||||
| 	pool<RTLIL::IdString> selected_modules; | 	pool<RTLIL::IdString> selected_modules; | ||||||
| 	dict<RTLIL::IdString, pool<RTLIL::IdString>> selected_members; | 	dict<RTLIL::IdString, pool<RTLIL::IdString>> selected_members; | ||||||
| 	RTLIL::Design *current_design; |  | ||||||
| 
 | 
 | ||||||
| 	// create a new selection
 | 	Selection(bool full = true) : full_selection(full) { } | ||||||
| 	Selection( |  | ||||||
| 		// should the selection cover the full design
 |  | ||||||
| 		bool full = true, |  | ||||||
| 		// should the selection include boxed modules
 |  | ||||||
| 		bool boxes = false, |  | ||||||
| 		// the design to select from
 |  | ||||||
| 		RTLIL::Design *design = nullptr |  | ||||||
| 	) :  |  | ||||||
| 		full_selection(full && !boxes), selects_boxes(boxes), complete_selection(full && boxes), current_design(design) { } |  | ||||||
| 
 | 
 | ||||||
| 	// checks if the given module exists in the current design and is a
 |  | ||||||
| 	// boxed module, warning the user if the current design is not set
 |  | ||||||
| 	bool boxed_module(const RTLIL::IdString &mod_name) const; |  | ||||||
| 
 |  | ||||||
| 	// checks if the given module is included in this selection
 |  | ||||||
| 	bool selected_module(const RTLIL::IdString &mod_name) const; | 	bool selected_module(const RTLIL::IdString &mod_name) const; | ||||||
| 
 |  | ||||||
| 	// checks if the given module is wholly included in this selection,
 |  | ||||||
| 	// i.e. not partially selected
 |  | ||||||
| 	bool selected_whole_module(const RTLIL::IdString &mod_name) const; | 	bool selected_whole_module(const RTLIL::IdString &mod_name) const; | ||||||
| 
 |  | ||||||
| 	// checks if the given member from the given module is included in this
 |  | ||||||
| 	// selection
 |  | ||||||
| 	bool selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const; | 	bool selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const; | ||||||
| 
 |  | ||||||
| 	// optimizes this selection for the given design by:
 |  | ||||||
| 	// - removing non-existent modules and members, any boxed modules and
 |  | ||||||
| 	//   their members (if selection does not include boxes), and any
 |  | ||||||
| 	//   partially selected modules with no selected members;
 |  | ||||||
| 	// - marking partially selected modules as wholly selected if all
 |  | ||||||
| 	//   members of that module are selected; and
 |  | ||||||
| 	// - marking selection as a complete_selection if all modules in the
 |  | ||||||
| 	//   given design are selected, or a full_selection if it does not
 |  | ||||||
| 	//   include boxes.
 |  | ||||||
| 	void optimize(RTLIL::Design *design); | 	void optimize(RTLIL::Design *design); | ||||||
| 
 | 
 | ||||||
| 	// checks if selection covers full design (may or may not include
 |  | ||||||
| 	// boxed-modules)
 |  | ||||||
| 	bool selects_all() const { |  | ||||||
| 		return full_selection || complete_selection; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// add whole module to this selection
 |  | ||||||
| 	template<typename T1> void select(T1 *module) { | 	template<typename T1> void select(T1 *module) { | ||||||
| 		if (!selects_all() && selected_modules.count(module->name) == 0) { | 		if (!full_selection && selected_modules.count(module->name) == 0) { | ||||||
| 			selected_modules.insert(module->name); | 			selected_modules.insert(module->name); | ||||||
| 			selected_members.erase(module->name); | 			selected_members.erase(module->name); | ||||||
| 			if (module->get_blackbox_attribute()) |  | ||||||
| 				selects_boxes = true; |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// add member of module to this selection
 |  | ||||||
| 	template<typename T1, typename T2> void select(T1 *module, T2 *member) { | 	template<typename T1, typename T2> void select(T1 *module, T2 *member) { | ||||||
| 		if (!selects_all() && selected_modules.count(module->name) == 0) { | 		if (!full_selection && selected_modules.count(module->name) == 0) | ||||||
| 			selected_members[module->name].insert(member->name); | 			selected_members[module->name].insert(member->name); | ||||||
| 			if (module->get_blackbox_attribute()) |  | ||||||
| 				selects_boxes = true; |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// checks if selection is empty
 |  | ||||||
| 	bool empty() const { | 	bool empty() const { | ||||||
| 		return !selects_all() && selected_modules.empty() && selected_members.empty(); | 		return !full_selection && selected_modules.empty() && selected_members.empty(); | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	// clear this selection, leaving it empty
 |  | ||||||
| 	void clear(); |  | ||||||
| 
 |  | ||||||
| 	// create a new selection which is empty
 |  | ||||||
| 	static Selection EmptySelection(RTLIL::Design *design = nullptr) { return Selection(false, false, design); }; |  | ||||||
| 
 |  | ||||||
| 	// create a new selection with all non-boxed modules
 |  | ||||||
| 	static Selection FullSelection(RTLIL::Design *design = nullptr) { return Selection(true, false, design); }; |  | ||||||
| 
 |  | ||||||
| 	// create a new selection with all modules, including boxes
 |  | ||||||
| 	static Selection CompleteSelection(RTLIL::Design *design = nullptr) { return Selection(true, true, design); }; |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Monitor | struct RTLIL::Monitor | ||||||
|  | @ -1305,7 +1213,7 @@ struct RTLIL::Design | ||||||
| 	RTLIL::ObjRange<RTLIL::Module*> modules(); | 	RTLIL::ObjRange<RTLIL::Module*> modules(); | ||||||
| 	RTLIL::Module *module(const RTLIL::IdString &name); | 	RTLIL::Module *module(const RTLIL::IdString &name); | ||||||
| 	const RTLIL::Module *module(const RTLIL::IdString &name) const; | 	const RTLIL::Module *module(const RTLIL::IdString &name) const; | ||||||
| 	RTLIL::Module *top_module() const; | 	RTLIL::Module *top_module(); | ||||||
| 
 | 
 | ||||||
| 	bool has(const RTLIL::IdString &id) const { | 	bool has(const RTLIL::IdString &id) const { | ||||||
| 		return modules_.count(id) != 0; | 		return modules_.count(id) != 0; | ||||||
|  | @ -1332,118 +1240,57 @@ struct RTLIL::Design | ||||||
| 	void check(); | 	void check(); | ||||||
| 	void optimize(); | 	void optimize(); | ||||||
| 
 | 
 | ||||||
| 	// checks if the given module is included in the current selection
 |  | ||||||
| 	bool selected_module(const RTLIL::IdString &mod_name) const; | 	bool selected_module(const RTLIL::IdString &mod_name) const; | ||||||
| 
 |  | ||||||
| 	// checks if the given module is wholly included in the current
 |  | ||||||
| 	// selection, i.e. not partially selected
 |  | ||||||
| 	bool selected_whole_module(const RTLIL::IdString &mod_name) const; | 	bool selected_whole_module(const RTLIL::IdString &mod_name) const; | ||||||
| 
 |  | ||||||
| 	// checks if the given member from the given module is included in the
 |  | ||||||
| 	// current selection
 |  | ||||||
| 	bool selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const; | 	bool selected_member(const RTLIL::IdString &mod_name, const RTLIL::IdString &memb_name) const; | ||||||
| 
 | 
 | ||||||
| 	// checks if the given module is included in the current selection
 |  | ||||||
| 	bool selected_module(RTLIL::Module *mod) const; | 	bool selected_module(RTLIL::Module *mod) const; | ||||||
| 
 |  | ||||||
| 	// checks if the given module is wholly included in the current
 |  | ||||||
| 	// selection, i.e. not partially selected
 |  | ||||||
| 	bool selected_whole_module(RTLIL::Module *mod) const; | 	bool selected_whole_module(RTLIL::Module *mod) const; | ||||||
| 
 | 
 | ||||||
| 	// push the given selection to the selection stack
 |  | ||||||
| 	void push_selection(RTLIL::Selection sel); |  | ||||||
| 	// push a new selection to the selection stack, with nothing selected
 |  | ||||||
| 	void push_empty_selection(); |  | ||||||
| 	// push a new selection to the selection stack, with all non-boxed
 |  | ||||||
| 	// modules selected
 |  | ||||||
| 	void push_full_selection(); |  | ||||||
| 	// push a new selection to the selection stack, with all modules
 |  | ||||||
| 	// selected including boxes
 |  | ||||||
| 	void push_complete_selection(); |  | ||||||
| 	// pop the current selection from the stack, returning to a full
 |  | ||||||
| 	// selection (no boxes) if the stack is empty
 |  | ||||||
| 	void pop_selection(); |  | ||||||
| 
 |  | ||||||
| 	// get the current selection
 |  | ||||||
| 	RTLIL::Selection &selection() { | 	RTLIL::Selection &selection() { | ||||||
| 		return selection_stack.back(); | 		return selection_stack.back(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// get the current selection
 |  | ||||||
| 	const RTLIL::Selection &selection() const { | 	const RTLIL::Selection &selection() const { | ||||||
| 		return selection_stack.back(); | 		return selection_stack.back(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// is the current selection a full selection (no boxes)
 |  | ||||||
| 	bool full_selection() const { | 	bool full_selection() const { | ||||||
| 		return selection().full_selection; | 		return selection_stack.back().full_selection; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// is the given module in the current selection
 |  | ||||||
| 	template<typename T1> bool selected(T1 *module) const { | 	template<typename T1> bool selected(T1 *module) const { | ||||||
| 		return selected_module(module->name); | 		return selected_module(module->name); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// is the given member of the given module in the current selection
 |  | ||||||
| 	template<typename T1, typename T2> bool selected(T1 *module, T2 *member) const { | 	template<typename T1, typename T2> bool selected(T1 *module, T2 *member) const { | ||||||
| 		return selected_member(module->name, member->name); | 		return selected_member(module->name, member->name); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// add whole module to the current selection
 |  | ||||||
| 	template<typename T1> void select(T1 *module) { | 	template<typename T1> void select(T1 *module) { | ||||||
| 		RTLIL::Selection &sel = selection(); | 		if (selection_stack.size() > 0) { | ||||||
| 		sel.select(module); | 			RTLIL::Selection &sel = selection_stack.back(); | ||||||
|  | 			sel.select(module); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// add member of module to the current selection
 |  | ||||||
| 	template<typename T1, typename T2> void select(T1 *module, T2 *member) { | 	template<typename T1, typename T2> void select(T1 *module, T2 *member) { | ||||||
| 		RTLIL::Selection &sel = selection(); | 		if (selection_stack.size() > 0) { | ||||||
| 		sel.select(module, member); | 			RTLIL::Selection &sel = selection_stack.back(); | ||||||
|  | 			sel.select(module, member); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	// returns all selected modules
 | 	std::vector<RTLIL::Module*> selected_modules() const; | ||||||
| 	std::vector<RTLIL::Module*> selected_modules( | 	std::vector<RTLIL::Module*> selected_whole_modules() const; | ||||||
| 		// controls if partially selected modules are included
 | 	std::vector<RTLIL::Module*> selected_whole_modules_warn(bool include_wb = false) const; | ||||||
| 		RTLIL::SelectPartials partials = SELECT_ALL, |  | ||||||
| 		// controls if boxed modules are included
 |  | ||||||
| 		RTLIL::SelectBoxes boxes = SB_UNBOXED_WARN |  | ||||||
| 	) const; |  | ||||||
| 
 |  | ||||||
| 	// returns all selected modules, and may include boxes
 |  | ||||||
| 	std::vector<RTLIL::Module*> all_selected_modules() const { return selected_modules(SELECT_ALL, SB_ALL); } |  | ||||||
| 	// returns all selected unboxed modules, silently ignoring any boxed
 |  | ||||||
| 	// modules in the selection
 |  | ||||||
| 	std::vector<RTLIL::Module*> selected_unboxed_modules() const { return selected_modules(SELECT_ALL, SB_UNBOXED_ONLY); } |  | ||||||
| 	// returns all selected unboxed modules, warning the user if any boxed
 |  | ||||||
| 	// modules have been ignored
 |  | ||||||
| 	std::vector<RTLIL::Module*> selected_unboxed_modules_warn() const { return selected_modules(SELECT_ALL, SB_UNBOXED_WARN); } |  | ||||||
| 
 |  | ||||||
| 	[[deprecated("Use select_unboxed_whole_modules() to maintain prior behaviour, or consider one of the other selected whole module helpers.")]] |  | ||||||
| 	std::vector<RTLIL::Module*> selected_whole_modules() const { return selected_modules(SELECT_WHOLE_ONLY, SB_UNBOXED_WARN); } |  | ||||||
| 	// returns all selected whole modules, silently ignoring partially
 |  | ||||||
| 	// selected modules, and may include boxes
 |  | ||||||
| 	std::vector<RTLIL::Module*> all_selected_whole_modules() const { return selected_modules(SELECT_WHOLE_ONLY, SB_ALL); } |  | ||||||
| 	// returns all selected whole modules, warning the user if any partially
 |  | ||||||
| 	// selected or boxed modules have been ignored; optionally includes
 |  | ||||||
| 	// selected whole modules with the 'whitebox' attribute
 |  | ||||||
| 	std::vector<RTLIL::Module*> selected_whole_modules_warn( |  | ||||||
| 		// should whole modules with the 'whitebox' attribute be
 |  | ||||||
| 		// included
 |  | ||||||
| 		bool include_wb = false |  | ||||||
| 	) const { return selected_modules(SELECT_WHOLE_WARN, include_wb ? SB_EXCL_BB_WARN : SB_UNBOXED_WARN); } |  | ||||||
| 	// returns all selected unboxed whole modules, silently ignoring
 |  | ||||||
| 	// partially selected or boxed modules
 |  | ||||||
| 	std::vector<RTLIL::Module*> selected_unboxed_whole_modules() const { return selected_modules(SELECT_WHOLE_ONLY, SB_UNBOXED_ONLY); } |  | ||||||
| 	// returns all selected unboxed whole modules, warning the user if any
 |  | ||||||
| 	// partially selected or boxed modules have been ignored
 |  | ||||||
| 	std::vector<RTLIL::Module*> selected_unboxed_whole_modules_warn() const { return selected_modules(SELECT_WHOLE_WARN, SB_UNBOXED_WARN); } |  | ||||||
| #ifdef WITH_PYTHON | #ifdef WITH_PYTHON | ||||||
| 	static std::map<unsigned int, RTLIL::Design*> *get_all_designs(void); | 	static std::map<unsigned int, RTLIL::Design*> *get_all_designs(void); | ||||||
| #endif | #endif | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Module : public RTLIL::NamedObject | struct RTLIL::Module : public RTLIL::AttrObject | ||||||
| { | { | ||||||
| 	Hasher::hash_t hashidx_; | 	Hasher::hash_t hashidx_; | ||||||
| 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | ||||||
|  | @ -1466,6 +1313,7 @@ public: | ||||||
| 	std::vector<RTLIL::SigSig>   connections_; | 	std::vector<RTLIL::SigSig>   connections_; | ||||||
| 	std::vector<RTLIL::Binding*> bindings_; | 	std::vector<RTLIL::Binding*> bindings_; | ||||||
| 
 | 
 | ||||||
|  | 	RTLIL::IdString name; | ||||||
| 	idict<RTLIL::IdString> avail_parameters; | 	idict<RTLIL::IdString> avail_parameters; | ||||||
| 	dict<RTLIL::IdString, RTLIL::Const> parameter_default_values; | 	dict<RTLIL::IdString, RTLIL::Const> parameter_default_values; | ||||||
| 	dict<RTLIL::IdString, RTLIL::Memory*> memories; | 	dict<RTLIL::IdString, RTLIL::Memory*> memories; | ||||||
|  | @ -1510,14 +1358,8 @@ public: | ||||||
| 	bool has_memories_warn() const; | 	bool has_memories_warn() const; | ||||||
| 	bool has_processes_warn() const; | 	bool has_processes_warn() const; | ||||||
| 
 | 
 | ||||||
| 	bool is_selected() const; |  | ||||||
| 	bool is_selected_whole() const; |  | ||||||
| 
 |  | ||||||
| 	std::vector<RTLIL::Wire*> selected_wires() const; | 	std::vector<RTLIL::Wire*> selected_wires() const; | ||||||
| 	std::vector<RTLIL::Cell*> selected_cells() const; | 	std::vector<RTLIL::Cell*> selected_cells() const; | ||||||
| 	std::vector<RTLIL::Memory*> selected_memories() const; |  | ||||||
| 	std::vector<RTLIL::Process*> selected_processes() const; |  | ||||||
| 	std::vector<RTLIL::NamedObject*> selected_members() const; |  | ||||||
| 
 | 
 | ||||||
| 	template<typename T> bool selected(T *member) const { | 	template<typename T> bool selected(T *member) const { | ||||||
| 		return design->selected_member(name, member->name); | 		return design->selected_member(name, member->name); | ||||||
|  | @ -1803,7 +1645,7 @@ namespace RTLIL_BACKEND { | ||||||
| void dump_wire(std::ostream &f, std::string indent, const RTLIL::Wire *wire); | void dump_wire(std::ostream &f, std::string indent, const RTLIL::Wire *wire); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Wire : public RTLIL::NamedObject | struct RTLIL::Wire : public RTLIL::AttrObject | ||||||
| { | { | ||||||
| 	Hasher::hash_t hashidx_; | 	Hasher::hash_t hashidx_; | ||||||
| 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | ||||||
|  | @ -1826,6 +1668,7 @@ public: | ||||||
| 	void operator=(RTLIL::Wire &other) = delete; | 	void operator=(RTLIL::Wire &other) = delete; | ||||||
| 
 | 
 | ||||||
| 	RTLIL::Module *module; | 	RTLIL::Module *module; | ||||||
|  | 	RTLIL::IdString name; | ||||||
| 	int width, start_offset, port_id; | 	int width, start_offset, port_id; | ||||||
| 	bool port_input, port_output, upto, is_signed; | 	bool port_input, port_output, upto, is_signed; | ||||||
| 
 | 
 | ||||||
|  | @ -1854,13 +1697,14 @@ inline int GetSize(RTLIL::Wire *wire) { | ||||||
| 	return wire->width; | 	return wire->width; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Memory : public RTLIL::NamedObject | struct RTLIL::Memory : public RTLIL::AttrObject | ||||||
| { | { | ||||||
| 	Hasher::hash_t hashidx_; | 	Hasher::hash_t hashidx_; | ||||||
| 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | ||||||
| 
 | 
 | ||||||
| 	Memory(); | 	Memory(); | ||||||
| 
 | 
 | ||||||
|  | 	RTLIL::IdString name; | ||||||
| 	int width, start_offset, size; | 	int width, start_offset, size; | ||||||
| #ifdef WITH_PYTHON | #ifdef WITH_PYTHON | ||||||
| 	~Memory(); | 	~Memory(); | ||||||
|  | @ -1868,7 +1712,7 @@ struct RTLIL::Memory : public RTLIL::NamedObject | ||||||
| #endif | #endif | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Cell : public RTLIL::NamedObject | struct RTLIL::Cell : public RTLIL::AttrObject | ||||||
| { | { | ||||||
| 	Hasher::hash_t hashidx_; | 	Hasher::hash_t hashidx_; | ||||||
| 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | ||||||
|  | @ -1885,6 +1729,7 @@ public: | ||||||
| 	void operator=(RTLIL::Cell &other) = delete; | 	void operator=(RTLIL::Cell &other) = delete; | ||||||
| 
 | 
 | ||||||
| 	RTLIL::Module *module; | 	RTLIL::Module *module; | ||||||
|  | 	RTLIL::IdString name; | ||||||
| 	RTLIL::IdString type; | 	RTLIL::IdString type; | ||||||
| 	dict<RTLIL::IdString, RTLIL::SigSpec> connections_; | 	dict<RTLIL::IdString, RTLIL::SigSpec> connections_; | ||||||
| 	dict<RTLIL::IdString, RTLIL::Const> parameters; | 	dict<RTLIL::IdString, RTLIL::Const> parameters; | ||||||
|  | @ -1977,7 +1822,7 @@ struct RTLIL::SyncRule | ||||||
| 	RTLIL::SyncRule *clone() const; | 	RTLIL::SyncRule *clone() const; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RTLIL::Process : public RTLIL::NamedObject | struct RTLIL::Process : public RTLIL::AttrObject | ||||||
| { | { | ||||||
| 	Hasher::hash_t hashidx_; | 	Hasher::hash_t hashidx_; | ||||||
| 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | 	[[nodiscard]] Hasher hash_into(Hasher h) const { h.eat(hashidx_); return h; } | ||||||
|  | @ -1989,6 +1834,7 @@ protected: | ||||||
| 	~Process(); | 	~Process(); | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|  | 	RTLIL::IdString name; | ||||||
| 	RTLIL::Module *module; | 	RTLIL::Module *module; | ||||||
| 	RTLIL::CaseRule root_case; | 	RTLIL::CaseRule root_case; | ||||||
| 	std::vector<RTLIL::SyncRule*> syncs; | 	std::vector<RTLIL::SyncRule*> syncs; | ||||||
|  |  | ||||||
|  | @ -114,7 +114,7 @@ static int tcl_yosys_cmd(ClientData, Tcl_Interp *interp, int argc, const char *a | ||||||
| 		if (in_repl) { | 		if (in_repl) { | ||||||
| 			auto design = yosys_get_design(); | 			auto design = yosys_get_design(); | ||||||
| 			while (design->selection_stack.size() > 1) | 			while (design->selection_stack.size() > 1) | ||||||
| 				design->pop_selection(); | 				design->selection_stack.pop_back(); | ||||||
| 			log_reset_stack(); | 			log_reset_stack(); | ||||||
| 		} | 		} | ||||||
| 		Tcl_SetResult(interp, (char *)"Yosys command produced an error", TCL_STATIC); | 		Tcl_SetResult(interp, (char *)"Yosys command produced an error", TCL_STATIC); | ||||||
|  |  | ||||||
|  | @ -312,11 +312,11 @@ const char *create_prompt(RTLIL::Design *design, int recursion_counter) | ||||||
| 	str += "yosys"; | 	str += "yosys"; | ||||||
| 	if (!design->selected_active_module.empty()) | 	if (!design->selected_active_module.empty()) | ||||||
| 		str += stringf(" [%s]", RTLIL::unescape_id(design->selected_active_module).c_str()); | 		str += stringf(" [%s]", RTLIL::unescape_id(design->selected_active_module).c_str()); | ||||||
| 	if (!design->full_selection()) { | 	if (!design->selection_stack.empty() && !design->selection_stack.back().full_selection) { | ||||||
| 		if (design->selected_active_module.empty()) | 		if (design->selected_active_module.empty()) | ||||||
| 			str += "*"; | 			str += "*"; | ||||||
| 		else if (design->selection().selected_modules.size() != 1 || design->selection().selected_members.size() != 0 || | 		else if (design->selection_stack.back().selected_modules.size() != 1 || design->selection_stack.back().selected_members.size() != 0 || | ||||||
| 				design->selection().selected_modules.count(design->selected_active_module) == 0) | 				design->selection_stack.back().selected_modules.count(design->selected_active_module) == 0) | ||||||
| 			str += "*"; | 			str += "*"; | ||||||
| 	} | 	} | ||||||
| 	snprintf(buffer, 100, "%s> ", str.c_str()); | 	snprintf(buffer, 100, "%s> ", str.c_str()); | ||||||
|  | @ -979,7 +979,7 @@ void shell(RTLIL::Design *design) | ||||||
| 			Pass::call(design, command); | 			Pass::call(design, command); | ||||||
| 		} catch (log_cmd_error_exception) { | 		} catch (log_cmd_error_exception) { | ||||||
| 			while (design->selection_stack.size() > 1) | 			while (design->selection_stack.size() > 1) | ||||||
| 				design->pop_selection(); | 				design->selection_stack.pop_back(); | ||||||
| 			log_reset_stack(); | 			log_reset_stack(); | ||||||
| 		} | 		} | ||||||
| 		design->check(); | 		design->check(); | ||||||
|  |  | ||||||
|  | @ -102,7 +102,7 @@ static void add_wire(RTLIL::Design *design, RTLIL::Module *module, std::string n | ||||||
| 		RTLIL::Module *mod = design->module(cell->type); | 		RTLIL::Module *mod = design->module(cell->type); | ||||||
| 		if (mod == nullptr) | 		if (mod == nullptr) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!mod->is_selected_whole()) | 		if (!design->selected_whole_module(mod->name)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (mod->get_blackbox_attribute()) | 		if (mod->get_blackbox_attribute()) | ||||||
| 			continue; | 			continue; | ||||||
|  |  | ||||||
|  | @ -216,8 +216,8 @@ struct DesignPass : public Pass { | ||||||
| 			RTLIL::Selection sel; | 			RTLIL::Selection sel; | ||||||
| 			if (argidx != args.size()) { | 			if (argidx != args.size()) { | ||||||
| 				handle_extra_select_args(this, args, argidx, args.size(), copy_from_design); | 				handle_extra_select_args(this, args, argidx, args.size(), copy_from_design); | ||||||
| 				sel = copy_from_design->selection(); | 				sel = copy_from_design->selection_stack.back(); | ||||||
| 				copy_from_design->pop_selection(); | 				copy_from_design->selection_stack.pop_back(); | ||||||
| 				argidx = args.size(); | 				argidx = args.size(); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  | @ -368,7 +368,7 @@ struct DesignPass : public Pass { | ||||||
| 			design->selection_vars.clear(); | 			design->selection_vars.clear(); | ||||||
| 			design->selected_active_module.clear(); | 			design->selected_active_module.clear(); | ||||||
| 
 | 
 | ||||||
| 			design->push_full_selection(); | 			design->selection_stack.push_back(RTLIL::Selection()); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (reset_mode || reset_vlog_mode || !load_name.empty() || push_mode || pop_mode) | 		if (reset_mode || reset_vlog_mode || !load_name.empty() || push_mode || pop_mode) | ||||||
|  |  | ||||||
|  | @ -340,7 +340,7 @@ struct SccPass : public Pass { | ||||||
| 		int origSelectPos = design->selection_stack.size() - 1; | 		int origSelectPos = design->selection_stack.size() - 1; | ||||||
| 		extra_args(args, argidx, design); | 		extra_args(args, argidx, design); | ||||||
| 
 | 
 | ||||||
| 		auto newSelection = RTLIL::Selection::EmptySelection(design); | 		RTLIL::Selection newSelection(false); | ||||||
| 		int scc_counter = 0; | 		int scc_counter = 0; | ||||||
| 
 | 
 | ||||||
| 		for (auto mod : design->selected_modules()) | 		for (auto mod : design->selected_modules()) | ||||||
|  |  | ||||||
|  | @ -141,42 +141,24 @@ static bool match_attr(const dict<RTLIL::IdString, RTLIL::Const> &attributes, co | ||||||
| 	return match_attr(attributes, match_expr, std::string(), 0); | 	return match_attr(attributes, match_expr, std::string(), 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void select_all(RTLIL::Design *design, RTLIL::Selection &lhs) |  | ||||||
| { |  | ||||||
| 	if (!lhs.selects_all()) |  | ||||||
| 		return; |  | ||||||
| 	lhs.current_design = design; |  | ||||||
| 	lhs.selected_modules.clear(); |  | ||||||
| 	for (auto mod : design->modules()) { |  | ||||||
| 		if (!lhs.selects_boxes && mod->get_blackbox_attribute()) |  | ||||||
| 			continue; |  | ||||||
| 		lhs.selected_modules.insert(mod->name); |  | ||||||
| 	} |  | ||||||
| 	lhs.full_selection = false; |  | ||||||
| 	lhs.complete_selection = false; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static void select_op_neg(RTLIL::Design *design, RTLIL::Selection &lhs) | static void select_op_neg(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| { | { | ||||||
| 	if (lhs.selects_all()) { | 	if (lhs.full_selection) { | ||||||
| 		lhs.clear(); | 		lhs.full_selection = false; | ||||||
|  | 		lhs.selected_modules.clear(); | ||||||
|  | 		lhs.selected_members.clear(); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (lhs.selected_modules.size() == 0 && lhs.selected_members.size() == 0) { | 	if (lhs.selected_modules.size() == 0 && lhs.selected_members.size() == 0) { | ||||||
| 		if (lhs.selects_boxes) | 		lhs.full_selection = true; | ||||||
| 			lhs.complete_selection = true; |  | ||||||
| 		else |  | ||||||
| 			lhs.full_selection = true; |  | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	auto new_sel = RTLIL::Selection::EmptySelection(); | 	RTLIL::Selection new_sel(false); | ||||||
| 
 | 
 | ||||||
| 	for (auto mod : design->modules()) | 	for (auto mod : design->modules()) | ||||||
| 	{ | 	{ | ||||||
| 		if (!lhs.selects_boxes && mod->get_blackbox_attribute()) |  | ||||||
| 			continue; |  | ||||||
| 		if (lhs.selected_whole_module(mod->name)) | 		if (lhs.selected_whole_module(mod->name)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!lhs.selected_module(mod->name)) { | 		if (!lhs.selected_module(mod->name)) { | ||||||
|  | @ -230,7 +212,7 @@ static void select_op_random(RTLIL::Design *design, RTLIL::Selection &lhs, int c | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	lhs = RTLIL::Selection(false, lhs.selects_boxes, design); | 	lhs = RTLIL::Selection(false); | ||||||
| 
 | 
 | ||||||
| 	while (!objects.empty() && count-- > 0) | 	while (!objects.empty() && count-- > 0) | ||||||
| 	{ | 	{ | ||||||
|  | @ -261,7 +243,7 @@ static void select_op_submod(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| 
 | 
 | ||||||
| static void select_op_cells_to_modules(RTLIL::Design *design, RTLIL::Selection &lhs) | static void select_op_cells_to_modules(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| { | { | ||||||
| 	RTLIL::Selection new_sel(false, lhs.selects_boxes, design); | 	RTLIL::Selection new_sel(false); | ||||||
| 	for (auto mod : design->modules()) | 	for (auto mod : design->modules()) | ||||||
| 		if (lhs.selected_module(mod->name)) | 		if (lhs.selected_module(mod->name)) | ||||||
| 			for (auto cell : mod->cells()) | 			for (auto cell : mod->cells()) | ||||||
|  | @ -272,7 +254,7 @@ static void select_op_cells_to_modules(RTLIL::Design *design, RTLIL::Selection & | ||||||
| 
 | 
 | ||||||
| static void select_op_module_to_cells(RTLIL::Design *design, RTLIL::Selection &lhs) | static void select_op_module_to_cells(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| { | { | ||||||
| 	RTLIL::Selection new_sel(false, lhs.selects_boxes, design); | 	RTLIL::Selection new_sel(false); | ||||||
| 	for (auto mod : design->modules()) | 	for (auto mod : design->modules()) | ||||||
| 		for (auto cell : mod->cells()) | 		for (auto cell : mod->cells()) | ||||||
| 			if ((design->module(cell->type) != nullptr) && lhs.selected_whole_module(cell->type)) | 			if ((design->module(cell->type) != nullptr) && lhs.selected_whole_module(cell->type)) | ||||||
|  | @ -292,8 +274,6 @@ static void select_op_alias(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| { | { | ||||||
| 	for (auto mod : design->modules()) | 	for (auto mod : design->modules()) | ||||||
| 	{ | 	{ | ||||||
| 		if (!lhs.selects_boxes && mod->get_blackbox_attribute()) |  | ||||||
| 			continue; |  | ||||||
| 		if (lhs.selected_whole_module(mod->name)) | 		if (lhs.selected_whole_module(mod->name)) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!lhs.selected_module(mod->name)) | 		if (!lhs.selected_module(mod->name)) | ||||||
|  | @ -312,38 +292,18 @@ static void select_op_alias(RTLIL::Design *design, RTLIL::Selection &lhs) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void select_op_union(RTLIL::Design* design, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | static void select_op_union(RTLIL::Design*, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | ||||||
| { | { | ||||||
| 	if (lhs.complete_selection) |  | ||||||
| 		return; |  | ||||||
| 	else if (rhs.complete_selection) { |  | ||||||
| 		lhs.complete_selection = true; |  | ||||||
| 		lhs.optimize(design); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (rhs.selects_boxes) { |  | ||||||
| 		if (lhs.full_selection) { |  | ||||||
| 			select_all(design, lhs); |  | ||||||
| 		} |  | ||||||
| 		lhs.selects_boxes = true; |  | ||||||
| 	} |  | ||||||
| 	else if (lhs.full_selection) |  | ||||||
| 		return; |  | ||||||
| 
 |  | ||||||
| 	if (rhs.full_selection) { | 	if (rhs.full_selection) { | ||||||
| 		if (lhs.selects_boxes) { | 		lhs.full_selection = true; | ||||||
| 			auto new_rhs = RTLIL::Selection(rhs); | 		lhs.selected_modules.clear(); | ||||||
| 			select_all(design, new_rhs); | 		lhs.selected_members.clear(); | ||||||
| 			for (auto mod : new_rhs.selected_modules) |  | ||||||
| 				lhs.selected_modules.insert(mod); |  | ||||||
| 		} else { |  | ||||||
| 			lhs.clear(); |  | ||||||
| 			lhs.full_selection = true; |  | ||||||
| 		} |  | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (lhs.full_selection) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
| 	for (auto &it : rhs.selected_members) | 	for (auto &it : rhs.selected_members) | ||||||
| 		for (auto &it2 : it.second) | 		for (auto &it2 : it.second) | ||||||
| 			lhs.selected_members[it.first].insert(it2); | 			lhs.selected_members[it.first].insert(it2); | ||||||
|  | @ -356,30 +316,20 @@ static void select_op_union(RTLIL::Design* design, RTLIL::Selection &lhs, const | ||||||
| 
 | 
 | ||||||
| static void select_op_diff(RTLIL::Design *design, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | static void select_op_diff(RTLIL::Design *design, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | ||||||
| { | { | ||||||
| 	if (rhs.complete_selection) { |  | ||||||
| 		lhs.clear(); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (rhs.full_selection) { | 	if (rhs.full_selection) { | ||||||
| 		if (lhs.selects_boxes) { | 		lhs.full_selection = false; | ||||||
| 			auto new_rhs = RTLIL::Selection(rhs); | 		lhs.selected_modules.clear(); | ||||||
| 			select_all(design, new_rhs); | 		lhs.selected_members.clear(); | ||||||
| 			select_all(design, lhs); |  | ||||||
| 			for (auto mod : new_rhs.selected_modules) { |  | ||||||
| 				lhs.selected_modules.erase(mod); |  | ||||||
| 				lhs.selected_members.erase(mod); |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			lhs.clear(); |  | ||||||
| 		} |  | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (rhs.empty() || lhs.empty()) | 	if (lhs.full_selection) { | ||||||
| 		return; | 		if (!rhs.full_selection && rhs.selected_modules.size() == 0 && rhs.selected_members.size() == 0) | ||||||
| 
 | 			return; | ||||||
| 	select_all(design, lhs); | 		lhs.full_selection = false; | ||||||
|  | 		for (auto mod : design->modules()) | ||||||
|  | 			lhs.selected_modules.insert(mod->name); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	for (auto &it : rhs.selected_modules) { | 	for (auto &it : rhs.selected_modules) { | ||||||
| 		lhs.selected_modules.erase(it); | 		lhs.selected_modules.erase(it); | ||||||
|  | @ -416,46 +366,38 @@ static void select_op_diff(RTLIL::Design *design, RTLIL::Selection &lhs, const R | ||||||
| 
 | 
 | ||||||
| static void select_op_intersect(RTLIL::Design *design, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | static void select_op_intersect(RTLIL::Design *design, RTLIL::Selection &lhs, const RTLIL::Selection &rhs) | ||||||
| { | { | ||||||
| 	if (rhs.complete_selection) | 	if (rhs.full_selection) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	if (rhs.full_selection && !lhs.selects_boxes) | 	if (lhs.full_selection) { | ||||||
| 		return; | 		lhs.full_selection = false; | ||||||
| 
 | 		for (auto mod : design->modules()) | ||||||
| 	if (lhs.empty()) | 			lhs.selected_modules.insert(mod->name); | ||||||
| 		return; |  | ||||||
| 
 |  | ||||||
| 	if (rhs.empty()) { |  | ||||||
| 		lhs.clear(); |  | ||||||
| 		return; |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	select_all(design, lhs); |  | ||||||
| 
 |  | ||||||
| 	std::vector<RTLIL::IdString> del_list; | 	std::vector<RTLIL::IdString> del_list; | ||||||
| 
 | 
 | ||||||
| 	for (auto mod_name : lhs.selected_modules) { | 	for (auto &it : lhs.selected_modules) | ||||||
| 		if (rhs.selected_whole_module(mod_name)) | 		if (rhs.selected_modules.count(it) == 0) { | ||||||
| 			continue; | 			if (rhs.selected_members.count(it) > 0) | ||||||
| 		if (rhs.selected_module(mod_name)) | 				for (auto &it2 : rhs.selected_members.at(it)) | ||||||
| 			for (auto memb_name : rhs.selected_members.at(mod_name)) | 					lhs.selected_members[it].insert(it2); | ||||||
| 				lhs.selected_members[mod_name].insert(memb_name); | 			del_list.push_back(it); | ||||||
| 		del_list.push_back(mod_name); | 		} | ||||||
| 	} |  | ||||||
| 	for (auto &it : del_list) | 	for (auto &it : del_list) | ||||||
| 		lhs.selected_modules.erase(it); | 		lhs.selected_modules.erase(it); | ||||||
| 
 | 
 | ||||||
| 	del_list.clear(); | 	del_list.clear(); | ||||||
| 	for (auto &it : lhs.selected_members) { | 	for (auto &it : lhs.selected_members) { | ||||||
| 		if (rhs.selected_whole_module(it.first)) | 		if (rhs.selected_modules.count(it.first) > 0) | ||||||
| 			continue; | 			continue; | ||||||
| 		if (!rhs.selected_module(it.first)) { | 		if (rhs.selected_members.count(it.first) == 0) { | ||||||
| 			del_list.push_back(it.first); | 			del_list.push_back(it.first); | ||||||
| 			continue; | 			continue; | ||||||
| 		} | 		} | ||||||
| 		std::vector<RTLIL::IdString> del_list2; | 		std::vector<RTLIL::IdString> del_list2; | ||||||
| 		for (auto &it2 : it.second) | 		for (auto &it2 : it.second) | ||||||
| 			if (!rhs.selected_member(it.first, it2)) | 			if (rhs.selected_members.at(it.first).count(it2) == 0) | ||||||
| 				del_list2.push_back(it2); | 				del_list2.push_back(it2); | ||||||
| 		for (auto &it2 : del_list2) | 		for (auto &it2 : del_list2) | ||||||
| 			it.second.erase(it2); | 			it.second.erase(it2); | ||||||
|  | @ -668,7 +610,9 @@ static void select_filter_active_mod(RTLIL::Design *design, RTLIL::Selection &se | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	if (sel.full_selection) { | 	if (sel.full_selection) { | ||||||
| 		sel.clear(); | 		sel.full_selection = false; | ||||||
|  | 		sel.selected_modules.clear(); | ||||||
|  | 		sel.selected_members.clear(); | ||||||
| 		sel.selected_modules.insert(design->selected_active_module); | 		sel.selected_modules.insert(design->selected_active_module); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
|  | @ -701,7 +645,8 @@ static void select_stmt(RTLIL::Design *design, std::string arg, bool disable_emp | ||||||
| 
 | 
 | ||||||
| 	if (arg[0] == '%') { | 	if (arg[0] == '%') { | ||||||
| 		if (arg == "%") { | 		if (arg == "%") { | ||||||
| 			work_stack.push_back(design->selection()); | 			if (design->selection_stack.size() > 0) | ||||||
|  | 				work_stack.push_back(design->selection_stack.back()); | ||||||
| 		} else | 		} else | ||||||
| 		if (arg == "%%") { | 		if (arg == "%%") { | ||||||
| 			while (work_stack.size() > 1) { | 			while (work_stack.size() > 1) { | ||||||
|  | @ -851,16 +796,15 @@ static void select_stmt(RTLIL::Design *design, std::string arg, bool disable_emp | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	bool full_selection = (arg == "*" && arg_mod == "*"); | 	work_stack.push_back(RTLIL::Selection()); | ||||||
| 	work_stack.push_back(RTLIL::Selection(full_selection, select_blackboxes, design)); |  | ||||||
| 	RTLIL::Selection &sel = work_stack.back(); | 	RTLIL::Selection &sel = work_stack.back(); | ||||||
| 
 | 
 | ||||||
| 	if (sel.full_selection) { | 	if (arg == "*" && arg_mod == "*" && select_blackboxes) { | ||||||
| 		if (sel.selects_boxes) sel.optimize(design); |  | ||||||
| 		select_filter_active_mod(design, work_stack.back()); | 		select_filter_active_mod(design, work_stack.back()); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	sel.full_selection = false; | ||||||
| 	for (auto mod : design->modules()) | 	for (auto mod : design->modules()) | ||||||
| 	{ | 	{ | ||||||
| 		if (!select_blackboxes && mod->get_blackbox_attribute()) | 		if (!select_blackboxes && mod->get_blackbox_attribute()) | ||||||
|  | @ -1001,33 +945,38 @@ static void select_stmt(RTLIL::Design *design, std::string arg, bool disable_emp | ||||||
| 
 | 
 | ||||||
| 	for (auto &it : arg_mod_found) { | 	for (auto &it : arg_mod_found) { | ||||||
| 		if (it.second == false && !disable_empty_warning) { | 		if (it.second == false && !disable_empty_warning) { | ||||||
| 			std::string selection_str = select_blackboxes ? "=" : ""; | 			log_warning("Selection \"%s\" did not match any module.\n", it.first.c_str()); | ||||||
| 			selection_str += it.first; |  | ||||||
| 			log_warning("Selection \"%s\" did not match any module.\n", selection_str.c_str()); |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	for (auto &it : arg_memb_found) { | 	for (auto &it : arg_memb_found) { | ||||||
| 		if (it.second == false && !disable_empty_warning) { | 		if (it.second == false && !disable_empty_warning) { | ||||||
| 			std::string selection_str = select_blackboxes ? "=" : ""; | 			log_warning("Selection \"%s\" did not match any object.\n", it.first.c_str()); | ||||||
| 			selection_str += it.first; |  | ||||||
| 			log_warning("Selection \"%s\" did not match any object.\n", selection_str.c_str()); |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static std::string describe_selection_for_assert(RTLIL::Design *design, RTLIL::Selection *sel, bool whole_modules = false) | static std::string describe_selection_for_assert(RTLIL::Design *design, RTLIL::Selection *sel, bool whole_modules = false) | ||||||
| { | { | ||||||
| 	bool push_selection = &design->selection() != sel; |  | ||||||
| 	if (push_selection) design->push_selection(*sel); |  | ||||||
| 	std::string desc = "Selection contains:\n"; | 	std::string desc = "Selection contains:\n"; | ||||||
| 	for (auto mod : design->all_selected_modules()) | 	for (auto mod : design->modules()) | ||||||
| 	{ | 	{ | ||||||
| 		if (whole_modules && sel->selected_whole_module(mod->name)) | 		if (sel->selected_module(mod->name)) { | ||||||
| 			desc += stringf("%s\n", id2cstr(mod->name)); | 			if (whole_modules && sel->selected_whole_module(mod->name)) | ||||||
| 		for (auto it : mod->selected_members()) | 					desc += stringf("%s\n", id2cstr(mod->name)); | ||||||
| 			desc += stringf("%s/%s\n", id2cstr(mod->name), id2cstr(it->name)); | 			for (auto wire : mod->wires()) | ||||||
|  | 				if (sel->selected_member(mod->name, wire->name)) | ||||||
|  | 					desc += stringf("%s/%s\n", id2cstr(mod->name), id2cstr(wire->name)); | ||||||
|  | 			for (auto &it : mod->memories) | ||||||
|  | 				if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 					desc += stringf("%s/%s\n", id2cstr(mod->name), id2cstr(it.first)); | ||||||
|  | 			for (auto cell : mod->cells()) | ||||||
|  | 				if (sel->selected_member(mod->name, cell->name)) | ||||||
|  | 					desc += stringf("%s/%s\n", id2cstr(mod->name), id2cstr(cell->name)); | ||||||
|  | 			for (auto &it : mod->processes) | ||||||
|  | 				if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 					desc += stringf("%s/%s\n", id2cstr(mod->name), id2cstr(it.first)); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	if (push_selection) design->pop_selection(); |  | ||||||
| 	return desc; | 	return desc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -1052,9 +1001,9 @@ void handle_extra_select_args(Pass *pass, const vector<string> &args, size_t arg | ||||||
| 		work_stack.pop_back(); | 		work_stack.pop_back(); | ||||||
| 	} | 	} | ||||||
| 	if (work_stack.empty()) | 	if (work_stack.empty()) | ||||||
| 		design->push_empty_selection(); | 		design->selection_stack.push_back(RTLIL::Selection(false)); | ||||||
| 	else | 	else | ||||||
| 		design->push_selection(work_stack.back()); | 		design->selection_stack.push_back(work_stack.back()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // extern decl. in register.h
 | // extern decl. in register.h
 | ||||||
|  | @ -1068,7 +1017,7 @@ RTLIL::Selection eval_select_args(const vector<string> &args, RTLIL::Design *des | ||||||
| 		work_stack.pop_back(); | 		work_stack.pop_back(); | ||||||
| 	} | 	} | ||||||
| 	if (work_stack.empty()) | 	if (work_stack.empty()) | ||||||
| 		return RTLIL::Selection::EmptySelection(design); | 		return RTLIL::Selection(false); | ||||||
| 	return work_stack.back(); | 	return work_stack.back(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -1441,7 +1390,7 @@ struct SelectPass : public Pass { | ||||||
| 			if (f.fail()) | 			if (f.fail()) | ||||||
| 				log_error("Can't open '%s' for reading: %s\n", read_file.c_str(), strerror(errno)); | 				log_error("Can't open '%s' for reading: %s\n", read_file.c_str(), strerror(errno)); | ||||||
| 
 | 
 | ||||||
| 			auto sel = RTLIL::Selection::EmptySelection(design); | 			RTLIL::Selection sel(false); | ||||||
| 			string line; | 			string line; | ||||||
| 
 | 
 | ||||||
| 			while (std::getline(f, line)) { | 			while (std::getline(f, line)) { | ||||||
|  | @ -1482,7 +1431,7 @@ struct SelectPass : public Pass { | ||||||
| 			log_cmd_error("Option -unset can not be combined with -list, -write, -count, -set, %s.\n", common_flagset); | 			log_cmd_error("Option -unset can not be combined with -list, -write, -count, -set, %s.\n", common_flagset); | ||||||
| 
 | 
 | ||||||
| 		if (work_stack.size() == 0 && got_module) { | 		if (work_stack.size() == 0 && got_module) { | ||||||
| 			auto sel = RTLIL::Selection::FullSelection(design); | 			RTLIL::Selection sel; | ||||||
| 			select_filter_active_mod(design, sel); | 			select_filter_active_mod(design, sel); | ||||||
| 			work_stack.push_back(sel); | 			work_stack.push_back(sel); | ||||||
| 		} | 		} | ||||||
|  | @ -1492,16 +1441,16 @@ struct SelectPass : public Pass { | ||||||
| 			work_stack.pop_back(); | 			work_stack.pop_back(); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		log_assert(!design->selection_stack.empty()); | 		log_assert(design->selection_stack.size() > 0); | ||||||
| 
 | 
 | ||||||
| 		if (clear_mode) { | 		if (clear_mode) { | ||||||
| 			design->selection() = RTLIL::Selection::FullSelection(design); | 			design->selection_stack.back() = RTLIL::Selection(true); | ||||||
| 			design->selected_active_module = std::string(); | 			design->selected_active_module = std::string(); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (none_mode) { | 		if (none_mode) { | ||||||
| 			design->selection() = RTLIL::Selection::EmptySelection(design); | 			design->selection_stack.back() = RTLIL::Selection(false); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -1516,17 +1465,28 @@ struct SelectPass : public Pass { | ||||||
| 				if (f == nullptr) | 				if (f == nullptr) | ||||||
| 					log_error("Can't open '%s' for writing: %s\n", write_file.c_str(), strerror(errno)); | 					log_error("Can't open '%s' for writing: %s\n", write_file.c_str(), strerror(errno)); | ||||||
| 			} | 			} | ||||||
|  | 			RTLIL::Selection *sel = &design->selection_stack.back(); | ||||||
| 			if (work_stack.size() > 0) | 			if (work_stack.size() > 0) | ||||||
| 				design->push_selection(work_stack.back()); | 				sel = &work_stack.back(); | ||||||
| 			RTLIL::Selection *sel = &design->selection(); |  | ||||||
| 			sel->optimize(design); | 			sel->optimize(design); | ||||||
| 			for (auto mod : design->all_selected_modules()) | 			for (auto mod : design->modules()) | ||||||
| 			{ | 			{ | ||||||
| 				if (sel->selected_whole_module(mod->name) && list_mode) | 				if (sel->selected_whole_module(mod->name) && list_mode) | ||||||
| 					log("%s\n", id2cstr(mod->name)); | 					log("%s\n", id2cstr(mod->name)); | ||||||
| 				if (!list_mod_mode) | 				if (sel->selected_module(mod->name) && !list_mod_mode) { | ||||||
| 					for (auto it : mod->selected_members()) | 					for (auto wire : mod->wires()) | ||||||
| 						LOG_OBJECT("%s/%s\n", id2cstr(mod->name), id2cstr(it->name)) | 						if (sel->selected_member(mod->name, wire->name)) | ||||||
|  | 							LOG_OBJECT("%s/%s\n", id2cstr(mod->name), id2cstr(wire->name)) | ||||||
|  | 					for (auto &it : mod->memories) | ||||||
|  | 						if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 							LOG_OBJECT("%s/%s\n", id2cstr(mod->name), id2cstr(it.first)) | ||||||
|  | 					for (auto cell : mod->cells()) | ||||||
|  | 						if (sel->selected_member(mod->name, cell->name)) | ||||||
|  | 							LOG_OBJECT("%s/%s\n", id2cstr(mod->name), id2cstr(cell->name)) | ||||||
|  | 					for (auto &it : mod->processes) | ||||||
|  | 						if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 							LOG_OBJECT("%s/%s\n", id2cstr(mod->name), id2cstr(it.first)) | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 			if (count_mode) | 			if (count_mode) | ||||||
| 			{ | 			{ | ||||||
|  | @ -1535,8 +1495,6 @@ struct SelectPass : public Pass { | ||||||
| 			} | 			} | ||||||
| 			if (f != nullptr) | 			if (f != nullptr) | ||||||
| 				fclose(f); | 				fclose(f); | ||||||
| 			if (work_stack.size() > 0) |  | ||||||
| 				design->pop_selection(); |  | ||||||
| 		#undef LOG_OBJECT | 		#undef LOG_OBJECT | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
|  | @ -1545,8 +1503,8 @@ struct SelectPass : public Pass { | ||||||
| 		{ | 		{ | ||||||
| 			if (work_stack.size() == 0) | 			if (work_stack.size() == 0) | ||||||
| 				log_cmd_error("Nothing to add to selection.\n"); | 				log_cmd_error("Nothing to add to selection.\n"); | ||||||
| 			select_op_union(design, design->selection(), work_stack.back()); | 			select_op_union(design, design->selection_stack.back(), work_stack.back()); | ||||||
| 			design->selection().optimize(design); | 			design->selection_stack.back().optimize(design); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -1554,8 +1512,8 @@ struct SelectPass : public Pass { | ||||||
| 		{ | 		{ | ||||||
| 			if (work_stack.size() == 0) | 			if (work_stack.size() == 0) | ||||||
| 				log_cmd_error("Nothing to delete from selection.\n"); | 				log_cmd_error("Nothing to delete from selection.\n"); | ||||||
| 			select_op_diff(design, design->selection(), work_stack.back()); | 			select_op_diff(design, design->selection_stack.back(), work_stack.back()); | ||||||
| 			design->selection().optimize(design); | 			design->selection_stack.back().optimize(design); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -1595,13 +1553,23 @@ struct SelectPass : public Pass { | ||||||
| 			if (work_stack.size() == 0) | 			if (work_stack.size() == 0) | ||||||
| 				log_cmd_error("No selection to check.\n"); | 				log_cmd_error("No selection to check.\n"); | ||||||
| 			RTLIL::Selection *sel = &work_stack.back(); | 			RTLIL::Selection *sel = &work_stack.back(); | ||||||
| 			design->push_selection(*sel); |  | ||||||
| 			sel->optimize(design); | 			sel->optimize(design); | ||||||
| 			for (auto mod : design->all_selected_modules()) { | 			for (auto mod : design->modules()) | ||||||
| 				module_count++; | 				if (sel->selected_module(mod->name)) { | ||||||
| 				for ([[maybe_unused]] auto member_name : mod->selected_members()) | 					module_count++; | ||||||
| 					total_count++; | 					for (auto wire : mod->wires()) | ||||||
| 			} | 						if (sel->selected_member(mod->name, wire->name)) | ||||||
|  | 							total_count++; | ||||||
|  | 					for (auto &it : mod->memories) | ||||||
|  | 						if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 							total_count++; | ||||||
|  | 					for (auto cell : mod->cells()) | ||||||
|  | 						if (sel->selected_member(mod->name, cell->name)) | ||||||
|  | 							total_count++; | ||||||
|  | 					for (auto &it : mod->processes) | ||||||
|  | 						if (sel->selected_member(mod->name, it.first)) | ||||||
|  | 							total_count++; | ||||||
|  | 				} | ||||||
| 			if (assert_modcount >= 0 && assert_modcount != module_count) | 			if (assert_modcount >= 0 && assert_modcount != module_count) | ||||||
| 			{ | 			{ | ||||||
| 				log_error("Assertion failed: selection contains %d modules instead of the asserted %d:%s\n", | 				log_error("Assertion failed: selection contains %d modules instead of the asserted %d:%s\n", | ||||||
|  | @ -1625,14 +1593,13 @@ struct SelectPass : public Pass { | ||||||
| 				log_error("Assertion failed: selection contains %d elements, less than the minimum number %d:%s\n%s", | 				log_error("Assertion failed: selection contains %d elements, less than the minimum number %d:%s\n%s", | ||||||
| 						total_count, assert_min, sel_str.c_str(), desc.c_str()); | 						total_count, assert_min, sel_str.c_str(), desc.c_str()); | ||||||
| 			} | 			} | ||||||
| 			design->pop_selection(); |  | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (!set_name.empty()) | 		if (!set_name.empty()) | ||||||
| 		{ | 		{ | ||||||
| 			if (work_stack.size() == 0) | 			if (work_stack.size() == 0) | ||||||
| 				design->selection_vars[set_name] = RTLIL::Selection::EmptySelection(design); | 				design->selection_vars[set_name] = RTLIL::Selection(false); | ||||||
| 			else | 			else | ||||||
| 				design->selection_vars[set_name] = work_stack.back(); | 				design->selection_vars[set_name] = work_stack.back(); | ||||||
| 			return; | 			return; | ||||||
|  | @ -1646,7 +1613,7 @@ struct SelectPass : public Pass { | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (work_stack.size() == 0) { | 		if (work_stack.size() == 0) { | ||||||
| 			RTLIL::Selection &sel = design->selection(); | 			RTLIL::Selection &sel = design->selection_stack.back(); | ||||||
| 			if (sel.full_selection) | 			if (sel.full_selection) | ||||||
| 				log("*\n"); | 				log("*\n"); | ||||||
| 			for (auto &it : sel.selected_modules) | 			for (auto &it : sel.selected_modules) | ||||||
|  | @ -1657,8 +1624,8 @@ struct SelectPass : public Pass { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		design->selection() = work_stack.back(); | 		design->selection_stack.back() = work_stack.back(); | ||||||
| 		design->selection().optimize(design); | 		design->selection_stack.back().optimize(design); | ||||||
| 	} | 	} | ||||||
| } SelectPass; | } SelectPass; | ||||||
| 
 | 
 | ||||||
|  | @ -1698,8 +1665,7 @@ struct CdPass : public Pass { | ||||||
| 			log_cmd_error("Invalid number of arguments.\n"); | 			log_cmd_error("Invalid number of arguments.\n"); | ||||||
| 
 | 
 | ||||||
| 		if (args.size() == 1 || args[1] == "/") { | 		if (args.size() == 1 || args[1] == "/") { | ||||||
| 			design->pop_selection(); | 			design->selection_stack.back() = RTLIL::Selection(true); | ||||||
| 			design->push_full_selection(); |  | ||||||
| 			design->selected_active_module = std::string(); | 			design->selected_active_module = std::string(); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
|  | @ -1708,8 +1674,7 @@ struct CdPass : public Pass { | ||||||
| 		{ | 		{ | ||||||
| 			string modname = design->selected_active_module; | 			string modname = design->selected_active_module; | ||||||
| 
 | 
 | ||||||
| 			design->pop_selection(); | 			design->selection_stack.back() = RTLIL::Selection(true); | ||||||
| 			design->push_full_selection(); |  | ||||||
| 			design->selected_active_module = std::string(); | 			design->selected_active_module = std::string(); | ||||||
| 
 | 
 | ||||||
| 			while (1) | 			while (1) | ||||||
|  | @ -1726,10 +1691,9 @@ struct CdPass : public Pass { | ||||||
| 					continue; | 					continue; | ||||||
| 
 | 
 | ||||||
| 				design->selected_active_module = modname; | 				design->selected_active_module = modname; | ||||||
| 				design->pop_selection(); | 				design->selection_stack.back() = RTLIL::Selection(); | ||||||
| 				design->push_full_selection(); | 				select_filter_active_mod(design, design->selection_stack.back()); | ||||||
| 				select_filter_active_mod(design, design->selection()); | 				design->selection_stack.back().optimize(design); | ||||||
| 				design->selection().optimize(design); |  | ||||||
| 				return; | 				return; | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  | @ -1746,10 +1710,9 @@ struct CdPass : public Pass { | ||||||
| 
 | 
 | ||||||
| 		if (design->module(modname) != nullptr) { | 		if (design->module(modname) != nullptr) { | ||||||
| 			design->selected_active_module = modname; | 			design->selected_active_module = modname; | ||||||
| 			design->pop_selection(); | 			design->selection_stack.back() = RTLIL::Selection(); | ||||||
| 			design->push_full_selection(); | 			select_filter_active_mod(design, design->selection_stack.back()); | ||||||
| 			select_filter_active_mod(design, design->selection()); | 			design->selection_stack.back().optimize(design); | ||||||
| 			design->selection().optimize(design); |  | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -1796,7 +1759,7 @@ struct LsPass : public Pass { | ||||||
| 		{ | 		{ | ||||||
| 			std::vector<IdString> matches; | 			std::vector<IdString> matches; | ||||||
| 
 | 
 | ||||||
| 			for (auto mod : design->all_selected_modules()) | 			for (auto mod : design->selected_modules()) | ||||||
| 				matches.push_back(mod->name); | 				matches.push_back(mod->name); | ||||||
| 
 | 
 | ||||||
| 			if (!matches.empty()) { | 			if (!matches.empty()) { | ||||||
|  |  | ||||||
|  | @ -802,8 +802,8 @@ struct ShowPass : public Pass { | ||||||
| 				std::pair<std::string, RTLIL::Selection> data; | 				std::pair<std::string, RTLIL::Selection> data; | ||||||
| 				data.first = args[++argidx], argidx++; | 				data.first = args[++argidx], argidx++; | ||||||
| 				handle_extra_select_args(this, args, argidx, argidx+1, design); | 				handle_extra_select_args(this, args, argidx, argidx+1, design); | ||||||
| 				data.second = design->selection(); | 				data.second = design->selection_stack.back(); | ||||||
| 				design->pop_selection(); | 				design->selection_stack.pop_back(); | ||||||
| 				color_selections.push_back(data); | 				color_selections.push_back(data); | ||||||
| 				continue; | 				continue; | ||||||
| 			} | 			} | ||||||
|  | @ -811,8 +811,8 @@ struct ShowPass : public Pass { | ||||||
| 				std::pair<std::string, RTLIL::Selection> data; | 				std::pair<std::string, RTLIL::Selection> data; | ||||||
| 				data.first = args[++argidx], argidx++; | 				data.first = args[++argidx], argidx++; | ||||||
| 				handle_extra_select_args(this, args, argidx, argidx+1, design); | 				handle_extra_select_args(this, args, argidx, argidx+1, design); | ||||||
| 				data.second = design->selection(); | 				data.second = design->selection_stack.back(); | ||||||
| 				design->pop_selection(); | 				design->selection_stack.pop_back(); | ||||||
| 				label_selections.push_back(data); | 				label_selections.push_back(data); | ||||||
| 				continue; | 				continue; | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | @ -468,7 +468,7 @@ struct StatPass : public Pass { | ||||||
| 				first_module = false; | 				first_module = false; | ||||||
| 			} else { | 			} else { | ||||||
| 				log("\n"); | 				log("\n"); | ||||||
| 				log("=== %s%s ===\n", log_id(mod->name), mod->is_selected_whole() ? "" : " (partially selected)"); | 				log("=== %s%s ===\n", log_id(mod->name), design->selected_whole_module(mod->name) ? "" : " (partially selected)"); | ||||||
| 				log("\n"); | 				log("\n"); | ||||||
| 				data.log_data(mod->name, false); | 				data.log_data(mod->name, false); | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | @ -950,8 +950,8 @@ struct VizPass : public Pass { | ||||||
| 				auto type = arg == "-g" || arg == "-G" ? VizConfig::TYPE_G : | 				auto type = arg == "-g" || arg == "-G" ? VizConfig::TYPE_G : | ||||||
| 					arg == "-u" || arg == "-U" ? VizConfig::TYPE_U : | 					arg == "-u" || arg == "-U" ? VizConfig::TYPE_U : | ||||||
| 					arg == "-x" || arg == "-X" ? VizConfig::TYPE_X : VizConfig::TYPE_S; | 					arg == "-x" || arg == "-X" ? VizConfig::TYPE_X : VizConfig::TYPE_S; | ||||||
| 				config.groups.push_back({type, design->selection()}); | 				config.groups.push_back({type, design->selection_stack.back()}); | ||||||
| 				design->pop_selection(); | 				design->selection_stack.pop_back(); | ||||||
| 				continue; | 				continue; | ||||||
| 			} | 			} | ||||||
| 			if (arg == "-0" || arg == "-1" || arg == "-2" || arg == "-3" || arg == "-4" || | 			if (arg == "-0" || arg == "-1" || arg == "-2" || arg == "-3" || arg == "-4" || | ||||||
|  |  | ||||||
|  | @ -246,7 +246,7 @@ struct SubmodWorker | ||||||
| 	SubmodWorker(RTLIL::Design *design, RTLIL::Module *module, bool copy_mode = false, bool hidden_mode = false, std::string opt_name = std::string()) : | 	SubmodWorker(RTLIL::Design *design, RTLIL::Module *module, bool copy_mode = false, bool hidden_mode = false, std::string opt_name = std::string()) : | ||||||
| 			design(design), module(module), sigmap(module), copy_mode(copy_mode), hidden_mode(hidden_mode), opt_name(opt_name) | 			design(design), module(module), sigmap(module), copy_mode(copy_mode), hidden_mode(hidden_mode), opt_name(opt_name) | ||||||
| 	{ | 	{ | ||||||
| 		if (!module->is_selected_whole() && opt_name.empty()) | 		if (!design->selected_whole_module(module->name) && opt_name.empty()) | ||||||
| 			return; | 			return; | ||||||
| 
 | 
 | ||||||
| 		if (module->processes.size() > 0) { | 		if (module->processes.size() > 0) { | ||||||
|  |  | ||||||
|  | @ -734,7 +734,7 @@ struct CleanPass : public Pass { | ||||||
| 		count_rm_cells = 0; | 		count_rm_cells = 0; | ||||||
| 		count_rm_wires = 0; | 		count_rm_wires = 0; | ||||||
| 
 | 
 | ||||||
| 		for (auto module : design->selected_unboxed_whole_modules()) { | 		for (auto module : design->selected_whole_modules()) { | ||||||
| 			if (module->has_processes()) | 			if (module->has_processes()) | ||||||
| 				continue; | 				continue; | ||||||
| 			rmunused_module(module, purge_mode, ys_debug(), true); | 			rmunused_module(module, purge_mode, ys_debug(), true); | ||||||
|  |  | ||||||
|  | @ -57,7 +57,7 @@ struct CutpointPass : public Pass { | ||||||
| 
 | 
 | ||||||
| 		for (auto module : design->selected_modules()) | 		for (auto module : design->selected_modules()) | ||||||
| 		{ | 		{ | ||||||
| 			if (module->is_selected_whole()) { | 			if (design->selected_whole_module(module->name)) { | ||||||
| 				log("Making all outputs of module %s cut points, removing module contents.\n", log_id(module)); | 				log("Making all outputs of module %s cut points, removing module contents.\n", log_id(module)); | ||||||
| 				module->new_connections(std::vector<RTLIL::SigSig>()); | 				module->new_connections(std::vector<RTLIL::SigSig>()); | ||||||
| 				for (auto cell : vector<Cell*>(module->cells())) | 				for (auto cell : vector<Cell*>(module->cells())) | ||||||
|  |  | ||||||
|  | @ -2887,7 +2887,7 @@ struct SimPass : public Pass { | ||||||
| 			if (!top_mod) | 			if (!top_mod) | ||||||
| 				log_cmd_error("Design has no top module, use the 'hierarchy' command to specify one.\n"); | 				log_cmd_error("Design has no top module, use the 'hierarchy' command to specify one.\n"); | ||||||
| 		} else { | 		} else { | ||||||
| 			auto mods = design->selected_unboxed_whole_modules(); | 			auto mods = design->selected_whole_modules(); | ||||||
| 			if (GetSize(mods) != 1) | 			if (GetSize(mods) != 1) | ||||||
| 				log_cmd_error("Only one top module must be selected.\n"); | 				log_cmd_error("Only one top module must be selected.\n"); | ||||||
| 			top_mod = mods.front(); | 			top_mod = mods.front(); | ||||||
|  | @ -3016,7 +3016,7 @@ struct Fst2TbPass : public Pass { | ||||||
| 			if (!top_mod) | 			if (!top_mod) | ||||||
| 				log_cmd_error("Design has no top module, use the 'hierarchy' command to specify one.\n"); | 				log_cmd_error("Design has no top module, use the 'hierarchy' command to specify one.\n"); | ||||||
| 		} else { | 		} else { | ||||||
| 			auto mods = design->selected_unboxed_whole_modules(); | 			auto mods = design->selected_whole_modules(); | ||||||
| 			if (GetSize(mods) != 1) | 			if (GetSize(mods) != 1) | ||||||
| 				log_cmd_error("Only one top module must be selected.\n"); | 				log_cmd_error("Only one top module must be selected.\n"); | ||||||
| 			top_mod = mods.front(); | 			top_mod = mods.front(); | ||||||
|  |  | ||||||
|  | @ -306,10 +306,9 @@ struct Abc9Pass : public ScriptPass | ||||||
| 			} | 			} | ||||||
| 			run("design -stash $abc9"); | 			run("design -stash $abc9"); | ||||||
| 			run("design -load $abc9_map"); | 			run("design -load $abc9_map"); | ||||||
| 			if (help_mode) run("select =*"); | 			run("proc"); | ||||||
| 			else active_design->push_complete_selection(); |  | ||||||
| 			run("wbflip"); | 			run("wbflip"); | ||||||
| 			run("techmap -autoproc -wb -map %$abc9 -map +/techmap.v A:abc9_flop"); | 			run("techmap -wb -map %$abc9 -map +/techmap.v A:abc9_flop"); | ||||||
| 			run("opt -nodffe -nosdff"); | 			run("opt -nodffe -nosdff"); | ||||||
| 			if (dff_mode || help_mode) { | 			if (dff_mode || help_mode) { | ||||||
| 				if (!help_mode) | 				if (!help_mode) | ||||||
|  | @ -370,8 +369,6 @@ struct Abc9Pass : public ScriptPass | ||||||
| 			if (saved_designs.count("$abc9_holes") || help_mode) { | 			if (saved_designs.count("$abc9_holes") || help_mode) { | ||||||
| 				run("design -stash $abc9"); | 				run("design -stash $abc9"); | ||||||
| 				run("design -load $abc9_holes"); | 				run("design -load $abc9_holes"); | ||||||
| 				if (help_mode) run("select =*"); |  | ||||||
| 				else active_design->push_complete_selection(); |  | ||||||
| 				run("techmap -wb -map %$abc9 -map +/techmap.v"); | 				run("techmap -wb -map %$abc9 -map +/techmap.v"); | ||||||
| 				run("opt -purge"); | 				run("opt -purge"); | ||||||
| 				run("aigmap"); | 				run("aigmap"); | ||||||
|  | @ -394,7 +391,7 @@ struct Abc9Pass : public ScriptPass | ||||||
| 			} | 			} | ||||||
| 			else { | 			else { | ||||||
| 				auto selected_modules = active_design->selected_modules(); | 				auto selected_modules = active_design->selected_modules(); | ||||||
| 				active_design->push_empty_selection(); | 				active_design->selection_stack.emplace_back(false); | ||||||
| 
 | 
 | ||||||
| 				for (auto mod : selected_modules) { | 				for (auto mod : selected_modules) { | ||||||
| 					if (mod->processes.size() > 0) { | 					if (mod->processes.size() > 0) { | ||||||
|  | @ -403,9 +400,8 @@ struct Abc9Pass : public ScriptPass | ||||||
| 					} | 					} | ||||||
| 
 | 
 | ||||||
| 					log_push(); | 					log_push(); | ||||||
| 					active_design->select(mod); | 					active_design->selection().select(mod); | ||||||
| 
 | 
 | ||||||
| 					// this check does nothing because the above line adds the whole module to the selection
 |  | ||||||
| 					if (!active_design->selected_whole_module(mod)) | 					if (!active_design->selected_whole_module(mod)) | ||||||
| 						log_error("Can't handle partially selected module %s!\n", log_id(mod)); | 						log_error("Can't handle partially selected module %s!\n", log_id(mod)); | ||||||
| 
 | 
 | ||||||
|  | @ -456,7 +452,7 @@ struct Abc9Pass : public ScriptPass | ||||||
| 					log_pop(); | 					log_pop(); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				active_design->pop_selection(); | 				active_design->selection_stack.pop_back(); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -454,7 +454,7 @@ void prep_bypass(RTLIL::Design *design) | ||||||
| 
 | 
 | ||||||
| void prep_dff(RTLIL::Design *design) | void prep_dff(RTLIL::Design *design) | ||||||
| { | { | ||||||
| 	auto r = design->selection_vars.insert(std::make_pair(ID($abc9_flops), RTLIL::Selection::EmptySelection(design))); | 	auto r = design->selection_vars.insert(std::make_pair(ID($abc9_flops), RTLIL::Selection(false))); | ||||||
| 	auto &modules_sel = r.first->second; | 	auto &modules_sel = r.first->second; | ||||||
| 
 | 
 | ||||||
| 	for (auto module : design->selected_modules()) | 	for (auto module : design->selected_modules()) | ||||||
|  |  | ||||||
|  | @ -139,7 +139,7 @@ struct AbcNewPass : public ScriptPass { | ||||||
| 			if (!help_mode) { | 			if (!help_mode) { | ||||||
| 				selected_modules = order_modules(active_design, | 				selected_modules = order_modules(active_design, | ||||||
| 												 active_design->selected_whole_modules_warn()); | 												 active_design->selected_whole_modules_warn()); | ||||||
| 				active_design->push_empty_selection(); | 				active_design->selection_stack.emplace_back(false); | ||||||
| 			} else { | 			} else { | ||||||
| 				selected_modules = {nullptr}; | 				selected_modules = {nullptr}; | ||||||
| 				run("foreach module in selection"); | 				run("foreach module in selection"); | ||||||
|  | @ -157,7 +157,7 @@ struct AbcNewPass : public ScriptPass { | ||||||
| 					exe_options = abc_exe_options; | 					exe_options = abc_exe_options; | ||||||
| 					log_header(active_design, "Mapping module '%s'.\n", log_id(mod)); | 					log_header(active_design, "Mapping module '%s'.\n", log_id(mod)); | ||||||
| 					log_push(); | 					log_push(); | ||||||
| 					active_design->select(mod); | 					active_design->selection().select(mod); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				std::string script_save; | 				std::string script_save; | ||||||
|  | @ -194,7 +194,7 @@ struct AbcNewPass : public ScriptPass { | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (!help_mode) { | 			if (!help_mode) { | ||||||
| 				active_design->pop_selection(); | 				active_design->selection_stack.pop_back(); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -171,7 +171,8 @@ struct AigmapPass : public Pass { | ||||||
| 				module->remove(cell); | 				module->remove(cell); | ||||||
| 
 | 
 | ||||||
| 			if (select_mode) { | 			if (select_mode) { | ||||||
| 				RTLIL::Selection& sel = design->selection(); | 				log_assert(!design->selection_stack.empty()); | ||||||
|  | 				RTLIL::Selection& sel = design->selection_stack.back(); | ||||||
| 				sel.selected_members[module->name] = std::move(new_sel); | 				sel.selected_members[module->name] = std::move(new_sel); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -333,7 +333,7 @@ struct ClockgatePass : public Pass { | ||||||
| 		dict<ClkNetInfo, GClkNetInfo> clk_nets; | 		dict<ClkNetInfo, GClkNetInfo> clk_nets; | ||||||
| 
 | 
 | ||||||
| 		int gated_flop_count = 0; | 		int gated_flop_count = 0; | ||||||
| 		for (auto module : design->selected_unboxed_whole_modules()) { | 		for (auto module : design->selected_whole_modules()) { | ||||||
| 			for (auto cell : module->cells()) { | 			for (auto cell : module->cells()) { | ||||||
| 				if (!RTLIL::builtin_ff_cell_types().count(cell->type)) | 				if (!RTLIL::builtin_ff_cell_types().count(cell->type)) | ||||||
| 					continue; | 					continue; | ||||||
|  |  | ||||||
|  | @ -42,7 +42,7 @@ struct NlutmapWorker | ||||||
| 
 | 
 | ||||||
| 	RTLIL::Selection get_selection() | 	RTLIL::Selection get_selection() | ||||||
| 	{ | 	{ | ||||||
| 		auto sel = RTLIL::Selection::EmptySelection(module->design); | 		RTLIL::Selection sel(false); | ||||||
| 		for (auto cell : module->cells()) | 		for (auto cell : module->cells()) | ||||||
| 			if (!mapped_cells.count(cell)) | 			if (!mapped_cells.count(cell)) | ||||||
| 				sel.select(module, cell); | 				sel.select(module, cell); | ||||||
|  |  | ||||||
|  | @ -13,5 +13,5 @@ run_subtest value | ||||||
| run_subtest value_fuzz | run_subtest value_fuzz | ||||||
| 
 | 
 | ||||||
| # Compile-only test. | # Compile-only test. | ||||||
| ../../yosys -p "read_verilog test_unconnected_output.v; select =*; proc; clean; write_cxxrtl cxxrtl-test-unconnected_output.cc" | ../../yosys -p "read_verilog test_unconnected_output.v; proc; clean; write_cxxrtl cxxrtl-test-unconnected_output.cc" | ||||||
| ${CC:-gcc} -std=c++11 -c -o cxxrtl-test-unconnected_output -I../../backends/cxxrtl/runtime cxxrtl-test-unconnected_output.cc | ${CC:-gcc} -std=c++11 -c -o cxxrtl-test-unconnected_output -I../../backends/cxxrtl/runtime cxxrtl-test-unconnected_output.cc | ||||||
|  |  | ||||||
|  | @ -1,3 +1,4 @@ | ||||||
|  | read_verilog -specify <<EOT | ||||||
| module top(input a, b, output o); | module top(input a, b, output o); | ||||||
| assign o = a & b; | assign o = a & b; | ||||||
| endmodule | endmodule | ||||||
|  | @ -14,3 +15,14 @@ endmodule | ||||||
| module wb(input a, b, output o); | module wb(input a, b, output o); | ||||||
| assign o = a ^ b; | assign o = a ^ b; | ||||||
| endmodule | endmodule | ||||||
|  | EOT | ||||||
|  | clean | ||||||
|  | 
 | ||||||
|  | select -assert-count 1 c:* | ||||||
|  | select -assert-none t:* t:$and %d | ||||||
|  | select -assert-count 3 w:* | ||||||
|  | select -assert-count 4 * | ||||||
|  | 
 | ||||||
|  | select -assert-count 3 =c:* | ||||||
|  | select -assert-count 10 =w:* | ||||||
|  | select -assert-count 13 =* | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| logger -expect-no-warnings |  | ||||||
| select -assert-count 5 =wb |  | ||||||
| clean =wb |  | ||||||
| select -assert-count 4 =wb |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| select -set top top |  | ||||||
| select -assert-count 4 @top |  | ||||||
| select -set boxes =?b |  | ||||||
| select -assert-count 9 @boxes |  | ||||||
|  | @ -1,38 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| # wb = 4w1c, bb = 3w1c, top = 3w1c |  | ||||||
| select =wb |  | ||||||
| select -assert-count 5 % |  | ||||||
| select -add =bb |  | ||||||
| select -assert-count 9 % |  | ||||||
| select -del =wb |  | ||||||
| select -assert-count 4 % |  | ||||||
| 
 |  | ||||||
| # unions |  | ||||||
| select -assert-count 8 =bb * %u |  | ||||||
| select -assert-count 8 * =bb %u |  | ||||||
| select -assert-count 13 top =* %u |  | ||||||
| select -assert-count 8 =bb top %u |  | ||||||
| select -assert-count 8 top =bb %u |  | ||||||
| 
 |  | ||||||
| # intersections |  | ||||||
| select -assert-count 3 =w:* =bb %i |  | ||||||
| select -assert-count 4 =* * %i |  | ||||||
| select -assert-count 4 * =* %i |  | ||||||
| 
 |  | ||||||
| # inverses |  | ||||||
| select -assert-count 8 =wb %n |  | ||||||
| select -assert-none top %n |  | ||||||
| select -assert-none * %n |  | ||||||
| select -assert-none =* %n |  | ||||||
| select -assert-count 9 =top %n |  | ||||||
| 
 |  | ||||||
| # differences |  | ||||||
| select -assert-count 9 =* top %d |  | ||||||
| select -assert-count 0 top =* %d |  | ||||||
| select -assert-count 9 =* * %d |  | ||||||
| select -assert-count 0 * =* %d |  | ||||||
| 
 |  | ||||||
| # random |  | ||||||
| select -assert-any =?b %R |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| select -assert-none ?b |  | ||||||
| select -assert-count 4 =bb |  | ||||||
| select -assert-count 5 =wb |  | ||||||
| select -assert-count 9 =?b |  | ||||||
|  | @ -1,6 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| select -assert-count 3 =c:* |  | ||||||
| select -assert-count 10 =w:* |  | ||||||
| select -assert-count 13 =* |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| select -assert-count 1 c:* |  | ||||||
| select -assert-none t:* t:$and %d |  | ||||||
| select -assert-count 3 w:* |  | ||||||
| select -assert-count 4 * |  | ||||||
|  | @ -1,18 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| 
 |  | ||||||
| logger -expect warning "did not match any module" 2 |  | ||||||
| clean wb |  | ||||||
| opt_clean wb |  | ||||||
| logger -check-expected |  | ||||||
| 
 |  | ||||||
| select -clear |  | ||||||
| logger -expect warning "did not match any module" 2 |  | ||||||
| clean wb |  | ||||||
| opt_clean wb |  | ||||||
| logger -check-expected |  | ||||||
| 
 |  | ||||||
| select -none |  | ||||||
| logger -expect warning "did not match any module" 2 |  | ||||||
| clean wb |  | ||||||
| opt_clean wb |  | ||||||
| logger -check-expected |  | ||||||
|  | @ -1,29 +0,0 @@ | ||||||
| read_verilog -specify boxes.v |  | ||||||
| clean |  | ||||||
| 
 |  | ||||||
| # default selection == select * |  | ||||||
| select -assert-count 4 * |  | ||||||
| select -assert-count 4 % |  | ||||||
| 
 |  | ||||||
| # -none replaces default selection |  | ||||||
| select -none |  | ||||||
| select -assert-none % |  | ||||||
| select -assert-count 13 =* |  | ||||||
| 
 |  | ||||||
| # select replaces current selection |  | ||||||
| select =* |  | ||||||
| select -assert-count 13 % |  | ||||||
| 
 |  | ||||||
| # -module changes module |  | ||||||
| select -module wb |  | ||||||
| select -assert-none % |  | ||||||
| select -assert-count 5 =* |  | ||||||
| 
 |  | ||||||
| # -none maintains module |  | ||||||
| select -none |  | ||||||
| select -assert-count 5 =* |  | ||||||
| 
 |  | ||||||
| # -clear clears current selection and module |  | ||||||
| select -clear |  | ||||||
| select -assert-count 4 % |  | ||||||
| select -assert-count 13 =* |  | ||||||
|  | @ -49,6 +49,6 @@ exec ${MAKE:-make} -f ../tools/autotest.mk $seed *.v *.sv EXTRA_FLAGS="-f \"veri | ||||||
|     clean; \ |     clean; \ | ||||||
|     check -assert * abc9_test037 %d; \ |     check -assert * abc9_test037 %d; \ | ||||||
|     select -assert-none t:${DOLLAR}_NOT_ t:${DOLLAR}_AND_ %%; \ |     select -assert-none t:${DOLLAR}_NOT_ t:${DOLLAR}_AND_ %%; \ | ||||||
|     setattr -mod -unset blackbox -unset whitebox =*'" |     setattr -mod -unset blackbox -unset whitebox'" | ||||||
| 
 | 
 | ||||||
| # NOTE: Skip 'check -assert' on abc9_test037 because it intentionally has a combinatorial loop | # NOTE: Skip 'check -assert' on abc9_test037 because it intentionally has a combinatorial loop | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue