mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-25 00:44:37 +00:00 
			
		
		
		
	s/((Claire|Xen|Xenia|Clifford)\s+)+(Wolf|Xen)\s+<(claire|clifford)@(symbioticeda.com|clifford.at|yosyshq.com)>/Claire Xenia Wolf <claire@yosyshq.com>/gi; s/((Nina|Nak|N\.)\s+)+Engelhardt\s+<nak@(symbioticeda.com|yosyshq.com)>/N. Engelhardt <nak@yosyshq.com>/gi; s/((David)\s+)+Shah\s+<(dave|david)@(symbioticeda.com|yosyshq.com|ds0.me)>/David Shah <dave@ds0.me>/gi; s/((Miodrag)\s+)+Milanovic\s+<(miodrag|micko)@(symbioticeda.com|yosyshq.com)>/Miodrag Milanovic <micko@yosyshq.com>/gi; s,https?://www.clifford.at/yosys/,http://yosyshq.net/yosys/,g;
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  *  yosys -- Yosys Open SYnthesis Suite
 | |
|  *
 | |
|  *  Copyright (C) 2012  Claire Xenia Wolf <claire@yosyshq.com>
 | |
|  *
 | |
|  *  Permission to use, copy, modify, and/or distribute this software for any
 | |
|  *  purpose with or without fee is hereby granted, provided that the above
 | |
|  *  copyright notice and this permission notice appear in all copies.
 | |
|  *
 | |
|  *  THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 | |
|  *  WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 | |
|  *  MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 | |
|  *  ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | |
|  *  WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | |
|  *  ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 | |
|  *  OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include "kernel/yosys.h"
 | |
| #include "kernel/sigtools.h"
 | |
| #include "passes/techmap/simplemap.h"
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| USING_YOSYS_NAMESPACE
 | |
| PRIVATE_NAMESPACE_BEGIN
 | |
| 
 | |
| static SigBit get_bit_or_zero(const SigSpec &sig)
 | |
| {
 | |
| 	if (GetSize(sig) == 0)
 | |
| 		return State::S0;
 | |
| 	return sig[0];
 | |
| }
 | |
| 
 | |
| static void run_ice40_opts(Module *module)
 | |
| {
 | |
| 	pool<SigBit> optimized_co;
 | |
| 	vector<Cell*> sb_lut_cells;
 | |
| 	SigMap sigmap(module);
 | |
| 
 | |
| 	for (auto cell : module->selected_cells())
 | |
| 	{
 | |
| 		if (!cell->type.in(ID(SB_LUT4), ID(SB_CARRY), ID($__ICE40_CARRY_WRAPPER)))
 | |
| 			continue;
 | |
| 		if (cell->has_keep_attr())
 | |
| 			continue;
 | |
| 
 | |
| 		if (cell->type == ID(SB_LUT4))
 | |
| 		{
 | |
| 			sb_lut_cells.push_back(cell);
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (cell->type == ID(SB_CARRY))
 | |
| 		{
 | |
| 			SigSpec non_const_inputs, replacement_output;
 | |
| 			int count_zeros = 0, count_ones = 0;
 | |
| 
 | |
| 			SigBit inbit[3] = {
 | |
| 				get_bit_or_zero(cell->getPort(ID(I0))),
 | |
| 				get_bit_or_zero(cell->getPort(ID(I1))),
 | |
| 				get_bit_or_zero(cell->getPort(ID::CI))
 | |
| 			};
 | |
| 			for (int i = 0; i < 3; i++)
 | |
| 				if (inbit[i].wire == nullptr) {
 | |
| 					if (inbit[i] == State::S1)
 | |
| 						count_ones++;
 | |
| 					else
 | |
| 						count_zeros++;
 | |
| 				} else
 | |
| 					non_const_inputs.append(inbit[i]);
 | |
| 
 | |
| 			if (count_zeros >= 2)
 | |
| 				replacement_output = State::S0;
 | |
| 			else if (count_ones >= 2)
 | |
| 				replacement_output = State::S1;
 | |
| 			else if (GetSize(non_const_inputs) == 1)
 | |
| 				replacement_output = non_const_inputs;
 | |
| 
 | |
| 			if (GetSize(replacement_output)) {
 | |
| 				optimized_co.insert(sigmap(cell->getPort(ID::CO)[0]));
 | |
| 				module->connect(cell->getPort(ID::CO)[0], replacement_output);
 | |
| 				module->design->scratchpad_set_bool("opt.did_something", true);
 | |
| 				log("Optimized away SB_CARRY cell %s.%s: CO=%s\n",
 | |
| 						log_id(module), log_id(cell), log_signal(replacement_output));
 | |
| 				module->remove(cell);
 | |
| 			}
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (cell->type == ID($__ICE40_CARRY_WRAPPER))
 | |
| 		{
 | |
| 			SigSpec non_const_inputs, replacement_output;
 | |
| 			int count_zeros = 0, count_ones = 0;
 | |
| 
 | |
| 			SigBit inbit[3] = {
 | |
| 				cell->getPort(ID::A),
 | |
| 				cell->getPort(ID::B),
 | |
| 				cell->getPort(ID::CI)
 | |
| 			};
 | |
| 			for (int i = 0; i < 3; i++)
 | |
| 				if (inbit[i].wire == nullptr) {
 | |
| 					if (inbit[i] == State::S1)
 | |
| 						count_ones++;
 | |
| 					else
 | |
| 						count_zeros++;
 | |
| 				} else
 | |
| 					non_const_inputs.append(inbit[i]);
 | |
| 
 | |
| 			if (count_zeros >= 2)
 | |
| 				replacement_output = State::S0;
 | |
| 			else if (count_ones >= 2)
 | |
| 				replacement_output = State::S1;
 | |
| 			else if (GetSize(non_const_inputs) == 1)
 | |
| 				replacement_output = non_const_inputs;
 | |
| 
 | |
| 			if (GetSize(replacement_output)) {
 | |
| 				optimized_co.insert(sigmap(cell->getPort(ID::CO)[0]));
 | |
| 				auto it = cell->attributes.find(ID(SB_LUT4.name));
 | |
| 				if (it != cell->attributes.end()) {
 | |
| 					module->rename(cell, it->second.decode_string());
 | |
| 					decltype(Cell::attributes) new_attr;
 | |
| 					for (const auto &a : cell->attributes)
 | |
| 						if (a.first.begins_with("\\SB_LUT4.\\"))
 | |
| 							new_attr[a.first.c_str() + strlen("\\SB_LUT4.")] = a.second;
 | |
| 						else if (a.first == ID::src)
 | |
| 							new_attr.insert(std::make_pair(a.first, a.second));
 | |
| 						else if (a.first.in(ID(SB_LUT4.name), ID::keep, ID::module_not_derived))
 | |
| 							continue;
 | |
| 						else if (a.first.begins_with("\\SB_CARRY.\\"))
 | |
| 							continue;
 | |
| 						else
 | |
| 							log_abort();
 | |
| 					cell->attributes = std::move(new_attr);
 | |
| 				}
 | |
| 				module->connect(cell->getPort(ID::CO)[0], replacement_output);
 | |
| 				module->design->scratchpad_set_bool("opt.did_something", true);
 | |
| 				log("Optimized $__ICE40_CARRY_WRAPPER cell back to logic (without SB_CARRY) %s.%s: CO=%s\n",
 | |
| 						log_id(module), log_id(cell), log_signal(replacement_output));
 | |
| 				cell->type = ID($lut);
 | |
| 				auto I3 = get_bit_or_zero(cell->getPort(cell->getParam(ID(I3_IS_CI)).as_bool() ? ID::CI : ID(I3)));
 | |
| 				cell->setPort(ID::A, { I3, inbit[1], inbit[0], get_bit_or_zero(cell->getPort(ID(I0))) });
 | |
| 				cell->setPort(ID::Y, cell->getPort(ID::O));
 | |
| 				cell->unsetPort(ID::B);
 | |
| 				cell->unsetPort(ID::CI);
 | |
| 				cell->unsetPort(ID(I0));
 | |
| 				cell->unsetPort(ID(I3));
 | |
| 				cell->unsetPort(ID::CO);
 | |
| 				cell->unsetPort(ID::O);
 | |
| 				cell->setParam(ID::WIDTH, 4);
 | |
| 				cell->unsetParam(ID(I3_IS_CI));
 | |
| 			}
 | |
| 			continue;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for (auto cell : sb_lut_cells)
 | |
| 	{
 | |
| 		SigSpec inbits;
 | |
| 
 | |
| 		inbits.append(get_bit_or_zero(cell->getPort(ID(I0))));
 | |
| 		inbits.append(get_bit_or_zero(cell->getPort(ID(I1))));
 | |
| 		inbits.append(get_bit_or_zero(cell->getPort(ID(I2))));
 | |
| 		inbits.append(get_bit_or_zero(cell->getPort(ID(I3))));
 | |
| 		sigmap.apply(inbits);
 | |
| 
 | |
| 		if (optimized_co.count(inbits[0])) goto remap_lut;
 | |
| 		if (optimized_co.count(inbits[1])) goto remap_lut;
 | |
| 		if (optimized_co.count(inbits[2])) goto remap_lut;
 | |
| 		if (optimized_co.count(inbits[3])) goto remap_lut;
 | |
| 
 | |
| 		if (!sigmap(inbits).is_fully_const())
 | |
| 			continue;
 | |
| 
 | |
| 	remap_lut:
 | |
| 		module->design->scratchpad_set_bool("opt.did_something", true);
 | |
| 		log("Mapping SB_LUT4 cell %s.%s back to logic.\n", log_id(module), log_id(cell));
 | |
| 
 | |
| 		cell->type = ID($lut);
 | |
| 		cell->setParam(ID::WIDTH, 4);
 | |
| 		cell->setParam(ID::LUT, cell->getParam(ID(LUT_INIT)));
 | |
| 		cell->unsetParam(ID(LUT_INIT));
 | |
| 
 | |
| 		cell->setPort(ID::A, SigSpec({
 | |
| 			get_bit_or_zero(cell->getPort(ID(I3))),
 | |
| 			get_bit_or_zero(cell->getPort(ID(I2))),
 | |
| 			get_bit_or_zero(cell->getPort(ID(I1))),
 | |
| 			get_bit_or_zero(cell->getPort(ID(I0)))
 | |
| 		}));
 | |
| 		cell->setPort(ID::Y, cell->getPort(ID::O)[0]);
 | |
| 		cell->unsetPort(ID(I0));
 | |
| 		cell->unsetPort(ID(I1));
 | |
| 		cell->unsetPort(ID(I2));
 | |
| 		cell->unsetPort(ID(I3));
 | |
| 		cell->unsetPort(ID::O);
 | |
| 
 | |
| 		cell->check();
 | |
| 		simplemap_lut(module, cell);
 | |
| 		module->remove(cell);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| struct Ice40OptPass : public Pass {
 | |
| 	Ice40OptPass() : Pass("ice40_opt", "iCE40: perform simple optimizations") { }
 | |
| 	void help() override
 | |
| 	{
 | |
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | |
| 		log("\n");
 | |
| 		log("    ice40_opt [options] [selection]\n");
 | |
| 		log("\n");
 | |
| 		log("This command executes the following script:\n");
 | |
| 		log("\n");
 | |
| 		log("    do\n");
 | |
| 		log("        <ice40 specific optimizations>\n");
 | |
| 		log("        opt_expr -mux_undef -undriven [-full]\n");
 | |
| 		log("        opt_merge\n");
 | |
| 		log("        opt_dff\n");
 | |
| 		log("        opt_clean\n");
 | |
| 		log("    while <changed design>\n");
 | |
| 		log("\n");
 | |
| 	}
 | |
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) override
 | |
| 	{
 | |
| 		string opt_expr_args = "-mux_undef -undriven";
 | |
| 
 | |
| 		log_header(design, "Executing ICE40_OPT pass (performing simple optimizations).\n");
 | |
| 		log_push();
 | |
| 
 | |
| 		size_t argidx;
 | |
| 		for (argidx = 1; argidx < args.size(); argidx++) {
 | |
| 			if (args[argidx] == "-full") {
 | |
| 				opt_expr_args += " -full";
 | |
| 				continue;
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 		extra_args(args, argidx, design);
 | |
| 
 | |
| 		while (1)
 | |
| 		{
 | |
| 			design->scratchpad_unset("opt.did_something");
 | |
| 
 | |
| 			log_header(design, "Running ICE40 specific optimizations.\n");
 | |
| 			for (auto module : design->selected_modules())
 | |
| 				run_ice40_opts(module);
 | |
| 
 | |
| 			Pass::call(design, "opt_expr " + opt_expr_args);
 | |
| 			Pass::call(design, "opt_merge");
 | |
| 			Pass::call(design, "opt_dff");
 | |
| 			Pass::call(design, "opt_clean");
 | |
| 
 | |
| 			if (design->scratchpad_get_bool("opt.did_something") == false)
 | |
| 				break;
 | |
| 
 | |
| 			log_header(design, "Rerunning OPT passes. (Removed registers in this run.)\n");
 | |
| 		}
 | |
| 
 | |
| 		design->optimize();
 | |
| 		design->sort();
 | |
| 		design->check();
 | |
| 
 | |
| 		log_header(design, "Finished OPT passes. (There is nothing left to do.)\n");
 | |
| 		log_pop();
 | |
| 	}
 | |
| } Ice40OptPass;
 | |
| 
 | |
| PRIVATE_NAMESPACE_END
 |