mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-25 08:54:37 +00:00 
			
		
		
		
	Merge pull request #4946 from povik/cost-cc-enhance
cost: Add `$mem_v2`, `$macc_v2` estimates
This commit is contained in:
		
						commit
						38beae1e06
					
				
					 9 changed files with 143 additions and 123 deletions
				
			
		|  | @ -315,7 +315,7 @@ struct ConstEval | |||
| 			Macc macc; | ||||
| 			macc.from_cell(cell); | ||||
| 
 | ||||
| 			for (auto &port : macc.ports) { | ||||
| 			for (auto &port : macc.terms) { | ||||
| 				if (!eval(port.in_a, undef, cell)) | ||||
| 					return false; | ||||
| 				if (!eval(port.in_b, undef, cell)) | ||||
|  |  | |||
|  | @ -1,4 +1,5 @@ | |||
| #include "kernel/cost.h" | ||||
| #include "kernel/macc.h" | ||||
| 
 | ||||
| USING_YOSYS_NAMESPACE | ||||
| 
 | ||||
|  | @ -148,6 +149,9 @@ unsigned int CellCosts::get(RTLIL::Cell *cell) | |||
| 		log_assert(cell->hasPort(ID::Q) && "Weird flip flop"); | ||||
| 		log_debug("%s is ff\n", cell->name.c_str()); | ||||
| 		return cell->getParam(ID::WIDTH).as_int(); | ||||
| 	} else if (cell->type.in(ID($mem), ID($mem_v2))) { | ||||
| 		log_debug("%s is mem\n", cell->name.c_str()); | ||||
| 		return cell->getParam(ID::WIDTH).as_int() * cell->getParam(ID::SIZE).as_int(); | ||||
| 	} else if (y_coef(cell->type)) { | ||||
| 		// linear with Y_WIDTH or WIDTH
 | ||||
| 		log_assert((cell->hasParam(ID::Y_WIDTH) || cell->hasParam(ID::WIDTH)) && "Unknown width"); | ||||
|  | @ -173,6 +177,22 @@ unsigned int CellCosts::get(RTLIL::Cell *cell) | |||
| 		unsigned int coef = cell->type == ID($mul) ? 3 : 5; | ||||
| 		log_debug("%s coef*(sum**2) %d * %d\n", cell->name.c_str(), coef, sum * sum); | ||||
| 		return coef * sum * sum; | ||||
| 	} else if (cell->type.in(ID($macc), ID($macc_v2))) { | ||||
| 		// quadratic per term
 | ||||
| 		unsigned int cost_sum = 0; | ||||
| 		Macc macc; | ||||
| 		macc.from_cell(cell); | ||||
| 		unsigned int y_width = cell->getParam(ID::Y_WIDTH).as_int(); | ||||
| 		for (auto &term: macc.terms) { | ||||
| 			if (term.in_b.empty()) { | ||||
| 				// neglect addends
 | ||||
| 				continue; | ||||
| 			} | ||||
| 			unsigned a_width = term.in_a.size(), b_width = term.in_b.size(); | ||||
| 			unsigned int sum = a_width + b_width + std::min(y_width, a_width + b_width); | ||||
| 			cost_sum += 3 * sum * sum; | ||||
| 		} | ||||
| 		return cost_sum; | ||||
| 	} else if (cell->type == ID($lut)) { | ||||
| 		int width = cell->getParam(ID::WIDTH).as_int(); | ||||
| 		unsigned int cost = 1U << (unsigned int)width; | ||||
|  | @ -187,8 +207,8 @@ unsigned int CellCosts::get(RTLIL::Cell *cell) | |||
| 		log_debug("%s is free\n", cell->name.c_str()); | ||||
| 		return 0; | ||||
| 	} | ||||
| 	// TODO: $fsm $mem.* $macc
 | ||||
| 	// ignored: $pow
 | ||||
| 	// TODO: $fsm
 | ||||
| 	// ignored: $pow $memrd $memwr $meminit (and v2 counterparts)
 | ||||
| 
 | ||||
| 	log_warning("Can't determine cost of %s cell (%d parameters).\n", log_id(cell->type), GetSize(cell->parameters)); | ||||
| 	return 1; | ||||
|  |  | |||
|  | @ -26,18 +26,18 @@ YOSYS_NAMESPACE_BEGIN | |||
| 
 | ||||
| struct Macc | ||||
| { | ||||
| 	struct port_t { | ||||
| 	struct term_t { | ||||
| 		RTLIL::SigSpec in_a, in_b; | ||||
| 		bool is_signed, do_subtract; | ||||
| 	}; | ||||
| 	std::vector<port_t> ports; | ||||
| 	std::vector<term_t> terms; | ||||
| 
 | ||||
| 	void optimize(int width) | ||||
| 	{ | ||||
| 		std::vector<port_t> new_ports; | ||||
| 		std::vector<term_t> new_terms; | ||||
| 		RTLIL::Const off(0, width); | ||||
| 
 | ||||
| 		for (auto &port : ports) | ||||
| 		for (auto &port : terms) | ||||
| 		{ | ||||
| 			if (GetSize(port.in_a) == 0 && GetSize(port.in_b) == 0) | ||||
| 				continue; | ||||
|  | @ -68,25 +68,25 @@ struct Macc | |||
| 					port.in_b.remove(GetSize(port.in_b)-1); | ||||
| 			} | ||||
| 
 | ||||
| 			new_ports.push_back(port); | ||||
| 			new_terms.push_back(port); | ||||
| 		} | ||||
| 
 | ||||
| 		if (off.as_bool()) { | ||||
| 			port_t port; | ||||
| 			term_t port; | ||||
| 			port.in_a = off; | ||||
| 			port.is_signed = false; | ||||
| 			port.do_subtract = false; | ||||
| 			new_ports.push_back(port); | ||||
| 			new_terms.push_back(port); | ||||
| 		} | ||||
| 
 | ||||
| 		new_ports.swap(ports); | ||||
| 		new_terms.swap(terms); | ||||
| 	} | ||||
| 
 | ||||
| 	void from_cell_v1(RTLIL::Cell *cell) | ||||
| 	{ | ||||
| 		RTLIL::SigSpec port_a = cell->getPort(ID::A); | ||||
| 
 | ||||
| 		ports.clear(); | ||||
| 		terms.clear(); | ||||
| 
 | ||||
| 		auto config_bits = cell->getParam(ID::CONFIG); | ||||
| 		int config_cursor = 0; | ||||
|  | @ -105,7 +105,7 @@ struct Macc | |||
| 		{ | ||||
| 			log_assert(config_cursor + 2 + 2*num_bits <= config_width); | ||||
| 
 | ||||
| 			port_t this_port; | ||||
| 			term_t this_port; | ||||
| 			this_port.is_signed = config_bits[config_cursor++] == State::S1; | ||||
| 			this_port.do_subtract = config_bits[config_cursor++] == State::S1; | ||||
| 
 | ||||
|  | @ -126,11 +126,11 @@ struct Macc | |||
| 			port_a_cursor += size_b; | ||||
| 
 | ||||
| 			if (size_a || size_b) | ||||
| 				ports.push_back(this_port); | ||||
| 				terms.push_back(this_port); | ||||
| 		} | ||||
| 
 | ||||
| 		for (auto bit : cell->getPort(ID::B)) | ||||
| 			ports.push_back(port_t{{bit}, {}, false, false}); | ||||
| 			terms.push_back(term_t{{bit}, {}, false, false}); | ||||
| 
 | ||||
| 		log_assert(config_cursor == config_width); | ||||
| 		log_assert(port_a_cursor == GetSize(port_a)); | ||||
|  | @ -148,7 +148,7 @@ struct Macc | |||
| 		RTLIL::SigSpec port_b = cell->getPort(ID::B); | ||||
| 		RTLIL::SigSpec port_c = cell->getPort(ID::C); | ||||
| 
 | ||||
| 		ports.clear(); | ||||
| 		terms.clear(); | ||||
| 
 | ||||
| 		int nproducts = cell->getParam(ID::NPRODUCTS).as_int(); | ||||
| 		const Const &product_neg = cell->getParam(ID::PRODUCT_NEGATED); | ||||
|  | @ -158,7 +158,7 @@ struct Macc | |||
| 		const Const &b_signed = cell->getParam(ID::B_SIGNED); | ||||
| 		int ai = 0, bi = 0; | ||||
| 		for (int i = 0; i < nproducts; i++) { | ||||
| 			port_t term; | ||||
| 			term_t term; | ||||
| 
 | ||||
| 			log_assert(a_signed[i] == b_signed[i]); | ||||
| 			term.is_signed = (a_signed[i] == State::S1); | ||||
|  | @ -171,7 +171,7 @@ struct Macc | |||
| 			bi += b_width; | ||||
| 			term.do_subtract = (product_neg[i] == State::S1); | ||||
| 
 | ||||
| 			ports.push_back(term); | ||||
| 			terms.push_back(term); | ||||
| 		} | ||||
| 		log_assert(port_a.size() == ai); | ||||
| 		log_assert(port_b.size() == bi); | ||||
|  | @ -182,7 +182,7 @@ struct Macc | |||
| 		const Const &c_signed = cell->getParam(ID::C_SIGNED); | ||||
| 		int ci = 0; | ||||
| 		for (int i = 0; i < naddends; i++) { | ||||
| 			port_t term; | ||||
| 			term_t term; | ||||
| 
 | ||||
| 			term.is_signed = (c_signed[i] == State::S1); | ||||
| 			int c_width = c_widths.extract(16 * i, 16).as_int(false); | ||||
|  | @ -191,7 +191,7 @@ struct Macc | |||
| 			ci += c_width; | ||||
| 			term.do_subtract = (addend_neg[i] == State::S1); | ||||
| 
 | ||||
| 			ports.push_back(term); | ||||
| 			terms.push_back(term); | ||||
| 		} | ||||
| 		log_assert(port_c.size() == ci); | ||||
| 	} | ||||
|  | @ -205,23 +205,23 @@ struct Macc | |||
| 		Const c_signed, c_widths, addend_negated; | ||||
| 		SigSpec a, b, c; | ||||
| 
 | ||||
| 		for (int i = 0; i < (int) ports.size(); i++) { | ||||
| 			SigSpec term_a = ports[i].in_a, term_b = ports[i].in_b; | ||||
| 		for (int i = 0; i < (int) terms.size(); i++) { | ||||
| 			SigSpec term_a = terms[i].in_a, term_b = terms[i].in_b; | ||||
| 
 | ||||
| 			if (term_b.empty()) { | ||||
| 				// addend
 | ||||
| 				c_widths.append(Const(term_a.size(), 16)); | ||||
| 				c_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				addend_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||
| 				c_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				addend_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||
| 				c.append(term_a); | ||||
| 				naddends++; | ||||
| 			} else { | ||||
| 				// product
 | ||||
| 				a_widths.append(Const(term_a.size(), 16)); | ||||
| 				b_widths.append(Const(term_b.size(), 16)); | ||||
| 				a_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				b_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				product_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||
| 				a_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				b_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||
| 				product_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||
| 				a.append(term_a); | ||||
| 				b.append(term_b); | ||||
| 				nproducts++; | ||||
|  | @ -265,7 +265,7 @@ struct Macc | |||
| 		for (auto &bit : result.bits()) | ||||
| 			bit = State::S0; | ||||
| 
 | ||||
| 		for (auto &port : ports) | ||||
| 		for (auto &port : terms) | ||||
| 		{ | ||||
| 			if (!port.in_a.is_fully_const() || !port.in_b.is_fully_const()) | ||||
| 				return false; | ||||
|  | @ -287,9 +287,9 @@ struct Macc | |||
| 
 | ||||
| 	bool is_simple_product() | ||||
| 	{ | ||||
| 		return ports.size() == 1 && | ||||
| 				!ports[0].in_b.empty() && | ||||
| 				!ports[0].do_subtract; | ||||
| 		return terms.size() == 1 && | ||||
| 				!terms[0].in_b.empty() && | ||||
| 				!terms[0].do_subtract; | ||||
| 	} | ||||
| 
 | ||||
| 	Macc(RTLIL::Cell *cell = nullptr) | ||||
|  |  | |||
|  | @ -750,7 +750,7 @@ bool SatGen::importCell(RTLIL::Cell *cell, int timestep) | |||
| 
 | ||||
| 		std::vector<int> tmp(GetSize(y), ez->CONST_FALSE); | ||||
| 
 | ||||
| 		for (auto &port : macc.ports) | ||||
| 		for (auto &port : macc.terms) | ||||
| 		{ | ||||
| 			std::vector<int> in_a = importDefSigSpec(port.in_a, timestep); | ||||
| 			std::vector<int> in_b = importDefSigSpec(port.in_b, timestep); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue