mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-31 03:32:29 +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 macc; | ||||||
| 			macc.from_cell(cell); | 			macc.from_cell(cell); | ||||||
| 
 | 
 | ||||||
| 			for (auto &port : macc.ports) { | 			for (auto &port : macc.terms) { | ||||||
| 				if (!eval(port.in_a, undef, cell)) | 				if (!eval(port.in_a, undef, cell)) | ||||||
| 					return false; | 					return false; | ||||||
| 				if (!eval(port.in_b, undef, cell)) | 				if (!eval(port.in_b, undef, cell)) | ||||||
|  |  | ||||||
|  | @ -1,4 +1,5 @@ | ||||||
| #include "kernel/cost.h" | #include "kernel/cost.h" | ||||||
|  | #include "kernel/macc.h" | ||||||
| 
 | 
 | ||||||
| USING_YOSYS_NAMESPACE | USING_YOSYS_NAMESPACE | ||||||
| 
 | 
 | ||||||
|  | @ -148,6 +149,9 @@ unsigned int CellCosts::get(RTLIL::Cell *cell) | ||||||
| 		log_assert(cell->hasPort(ID::Q) && "Weird flip flop"); | 		log_assert(cell->hasPort(ID::Q) && "Weird flip flop"); | ||||||
| 		log_debug("%s is ff\n", cell->name.c_str()); | 		log_debug("%s is ff\n", cell->name.c_str()); | ||||||
| 		return cell->getParam(ID::WIDTH).as_int(); | 		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)) { | 	} else if (y_coef(cell->type)) { | ||||||
| 		// linear with Y_WIDTH or WIDTH
 | 		// linear with Y_WIDTH or WIDTH
 | ||||||
| 		log_assert((cell->hasParam(ID::Y_WIDTH) || cell->hasParam(ID::WIDTH)) && "Unknown 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; | 		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); | 		log_debug("%s coef*(sum**2) %d * %d\n", cell->name.c_str(), coef, sum * sum); | ||||||
| 		return 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)) { | 	} else if (cell->type == ID($lut)) { | ||||||
| 		int width = cell->getParam(ID::WIDTH).as_int(); | 		int width = cell->getParam(ID::WIDTH).as_int(); | ||||||
| 		unsigned int cost = 1U << (unsigned int)width; | 		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()); | 		log_debug("%s is free\n", cell->name.c_str()); | ||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 	// TODO: $fsm $mem.* $macc
 | 	// TODO: $fsm
 | ||||||
| 	// ignored: $pow
 | 	// 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)); | 	log_warning("Can't determine cost of %s cell (%d parameters).\n", log_id(cell->type), GetSize(cell->parameters)); | ||||||
| 	return 1; | 	return 1; | ||||||
|  |  | ||||||
|  | @ -26,18 +26,18 @@ YOSYS_NAMESPACE_BEGIN | ||||||
| 
 | 
 | ||||||
| struct Macc | struct Macc | ||||||
| { | { | ||||||
| 	struct port_t { | 	struct term_t { | ||||||
| 		RTLIL::SigSpec in_a, in_b; | 		RTLIL::SigSpec in_a, in_b; | ||||||
| 		bool is_signed, do_subtract; | 		bool is_signed, do_subtract; | ||||||
| 	}; | 	}; | ||||||
| 	std::vector<port_t> ports; | 	std::vector<term_t> terms; | ||||||
| 
 | 
 | ||||||
| 	void optimize(int width) | 	void optimize(int width) | ||||||
| 	{ | 	{ | ||||||
| 		std::vector<port_t> new_ports; | 		std::vector<term_t> new_terms; | ||||||
| 		RTLIL::Const off(0, width); | 		RTLIL::Const off(0, width); | ||||||
| 
 | 
 | ||||||
| 		for (auto &port : ports) | 		for (auto &port : terms) | ||||||
| 		{ | 		{ | ||||||
| 			if (GetSize(port.in_a) == 0 && GetSize(port.in_b) == 0) | 			if (GetSize(port.in_a) == 0 && GetSize(port.in_b) == 0) | ||||||
| 				continue; | 				continue; | ||||||
|  | @ -68,25 +68,25 @@ struct Macc | ||||||
| 					port.in_b.remove(GetSize(port.in_b)-1); | 					port.in_b.remove(GetSize(port.in_b)-1); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			new_ports.push_back(port); | 			new_terms.push_back(port); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (off.as_bool()) { | 		if (off.as_bool()) { | ||||||
| 			port_t port; | 			term_t port; | ||||||
| 			port.in_a = off; | 			port.in_a = off; | ||||||
| 			port.is_signed = false; | 			port.is_signed = false; | ||||||
| 			port.do_subtract = 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) | 	void from_cell_v1(RTLIL::Cell *cell) | ||||||
| 	{ | 	{ | ||||||
| 		RTLIL::SigSpec port_a = cell->getPort(ID::A); | 		RTLIL::SigSpec port_a = cell->getPort(ID::A); | ||||||
| 
 | 
 | ||||||
| 		ports.clear(); | 		terms.clear(); | ||||||
| 
 | 
 | ||||||
| 		auto config_bits = cell->getParam(ID::CONFIG); | 		auto config_bits = cell->getParam(ID::CONFIG); | ||||||
| 		int config_cursor = 0; | 		int config_cursor = 0; | ||||||
|  | @ -105,7 +105,7 @@ struct Macc | ||||||
| 		{ | 		{ | ||||||
| 			log_assert(config_cursor + 2 + 2*num_bits <= config_width); | 			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.is_signed = config_bits[config_cursor++] == State::S1; | ||||||
| 			this_port.do_subtract = 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; | 			port_a_cursor += size_b; | ||||||
| 
 | 
 | ||||||
| 			if (size_a || size_b) | 			if (size_a || size_b) | ||||||
| 				ports.push_back(this_port); | 				terms.push_back(this_port); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for (auto bit : cell->getPort(ID::B)) | 		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(config_cursor == config_width); | ||||||
| 		log_assert(port_a_cursor == GetSize(port_a)); | 		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_b = cell->getPort(ID::B); | ||||||
| 		RTLIL::SigSpec port_c = cell->getPort(ID::C); | 		RTLIL::SigSpec port_c = cell->getPort(ID::C); | ||||||
| 
 | 
 | ||||||
| 		ports.clear(); | 		terms.clear(); | ||||||
| 
 | 
 | ||||||
| 		int nproducts = cell->getParam(ID::NPRODUCTS).as_int(); | 		int nproducts = cell->getParam(ID::NPRODUCTS).as_int(); | ||||||
| 		const Const &product_neg = cell->getParam(ID::PRODUCT_NEGATED); | 		const Const &product_neg = cell->getParam(ID::PRODUCT_NEGATED); | ||||||
|  | @ -158,7 +158,7 @@ struct Macc | ||||||
| 		const Const &b_signed = cell->getParam(ID::B_SIGNED); | 		const Const &b_signed = cell->getParam(ID::B_SIGNED); | ||||||
| 		int ai = 0, bi = 0; | 		int ai = 0, bi = 0; | ||||||
| 		for (int i = 0; i < nproducts; i++) { | 		for (int i = 0; i < nproducts; i++) { | ||||||
| 			port_t term; | 			term_t term; | ||||||
| 
 | 
 | ||||||
| 			log_assert(a_signed[i] == b_signed[i]); | 			log_assert(a_signed[i] == b_signed[i]); | ||||||
| 			term.is_signed = (a_signed[i] == State::S1); | 			term.is_signed = (a_signed[i] == State::S1); | ||||||
|  | @ -171,7 +171,7 @@ struct Macc | ||||||
| 			bi += b_width; | 			bi += b_width; | ||||||
| 			term.do_subtract = (product_neg[i] == State::S1); | 			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_a.size() == ai); | ||||||
| 		log_assert(port_b.size() == bi); | 		log_assert(port_b.size() == bi); | ||||||
|  | @ -182,7 +182,7 @@ struct Macc | ||||||
| 		const Const &c_signed = cell->getParam(ID::C_SIGNED); | 		const Const &c_signed = cell->getParam(ID::C_SIGNED); | ||||||
| 		int ci = 0; | 		int ci = 0; | ||||||
| 		for (int i = 0; i < naddends; i++) { | 		for (int i = 0; i < naddends; i++) { | ||||||
| 			port_t term; | 			term_t term; | ||||||
| 
 | 
 | ||||||
| 			term.is_signed = (c_signed[i] == State::S1); | 			term.is_signed = (c_signed[i] == State::S1); | ||||||
| 			int c_width = c_widths.extract(16 * i, 16).as_int(false); | 			int c_width = c_widths.extract(16 * i, 16).as_int(false); | ||||||
|  | @ -191,7 +191,7 @@ struct Macc | ||||||
| 			ci += c_width; | 			ci += c_width; | ||||||
| 			term.do_subtract = (addend_neg[i] == State::S1); | 			term.do_subtract = (addend_neg[i] == State::S1); | ||||||
| 
 | 
 | ||||||
| 			ports.push_back(term); | 			terms.push_back(term); | ||||||
| 		} | 		} | ||||||
| 		log_assert(port_c.size() == ci); | 		log_assert(port_c.size() == ci); | ||||||
| 	} | 	} | ||||||
|  | @ -205,23 +205,23 @@ struct Macc | ||||||
| 		Const c_signed, c_widths, addend_negated; | 		Const c_signed, c_widths, addend_negated; | ||||||
| 		SigSpec a, b, c; | 		SigSpec a, b, c; | ||||||
| 
 | 
 | ||||||
| 		for (int i = 0; i < (int) ports.size(); i++) { | 		for (int i = 0; i < (int) terms.size(); i++) { | ||||||
| 			SigSpec term_a = ports[i].in_a, term_b = ports[i].in_b; | 			SigSpec term_a = terms[i].in_a, term_b = terms[i].in_b; | ||||||
| 
 | 
 | ||||||
| 			if (term_b.empty()) { | 			if (term_b.empty()) { | ||||||
| 				// addend
 | 				// addend
 | ||||||
| 				c_widths.append(Const(term_a.size(), 16)); | 				c_widths.append(Const(term_a.size(), 16)); | ||||||
| 				c_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | 				c_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||||
| 				addend_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | 				addend_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||||
| 				c.append(term_a); | 				c.append(term_a); | ||||||
| 				naddends++; | 				naddends++; | ||||||
| 			} else { | 			} else { | ||||||
| 				// product
 | 				// product
 | ||||||
| 				a_widths.append(Const(term_a.size(), 16)); | 				a_widths.append(Const(term_a.size(), 16)); | ||||||
| 				b_widths.append(Const(term_b.size(), 16)); | 				b_widths.append(Const(term_b.size(), 16)); | ||||||
| 				a_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | 				a_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||||
| 				b_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0); | 				b_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0); | ||||||
| 				product_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | 				product_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0); | ||||||
| 				a.append(term_a); | 				a.append(term_a); | ||||||
| 				b.append(term_b); | 				b.append(term_b); | ||||||
| 				nproducts++; | 				nproducts++; | ||||||
|  | @ -265,7 +265,7 @@ struct Macc | ||||||
| 		for (auto &bit : result.bits()) | 		for (auto &bit : result.bits()) | ||||||
| 			bit = State::S0; | 			bit = State::S0; | ||||||
| 
 | 
 | ||||||
| 		for (auto &port : ports) | 		for (auto &port : terms) | ||||||
| 		{ | 		{ | ||||||
| 			if (!port.in_a.is_fully_const() || !port.in_b.is_fully_const()) | 			if (!port.in_a.is_fully_const() || !port.in_b.is_fully_const()) | ||||||
| 				return false; | 				return false; | ||||||
|  | @ -287,9 +287,9 @@ struct Macc | ||||||
| 
 | 
 | ||||||
| 	bool is_simple_product() | 	bool is_simple_product() | ||||||
| 	{ | 	{ | ||||||
| 		return ports.size() == 1 && | 		return terms.size() == 1 && | ||||||
| 				!ports[0].in_b.empty() && | 				!terms[0].in_b.empty() && | ||||||
| 				!ports[0].do_subtract; | 				!terms[0].do_subtract; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	Macc(RTLIL::Cell *cell = nullptr) | 	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); | 		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_a = importDefSigSpec(port.in_a, timestep); | ||||||
| 			std::vector<int> in_b = importDefSigSpec(port.in_b, timestep); | 			std::vector<int> in_b = importDefSigSpec(port.in_b, timestep); | ||||||
|  |  | ||||||
|  | @ -110,7 +110,7 @@ struct ShareWorker | ||||||
| 	// Code for sharing and comparing MACC cells
 | 	// Code for sharing and comparing MACC cells
 | ||||||
| 	// ---------------------------------------------------
 | 	// ---------------------------------------------------
 | ||||||
| 
 | 
 | ||||||
| 	static int bits_macc_port(const Macc::port_t &p, int width) | 	static int bits_macc_term(const Macc::term_t &p, int width) | ||||||
| 	{ | 	{ | ||||||
| 		if (GetSize(p.in_a) == 0 || GetSize(p.in_b) == 0) | 		if (GetSize(p.in_a) == 0 || GetSize(p.in_b) == 0) | ||||||
| 			return min(max(GetSize(p.in_a), GetSize(p.in_b)), width); | 			return min(max(GetSize(p.in_a), GetSize(p.in_b)), width); | ||||||
|  | @ -120,8 +120,8 @@ struct ShareWorker | ||||||
| 	static int bits_macc(const Macc &m, int width) | 	static int bits_macc(const Macc &m, int width) | ||||||
| 	{ | 	{ | ||||||
| 		int bits = 0; | 		int bits = 0; | ||||||
| 		for (auto &p : m.ports) | 		for (auto &p : m.terms) | ||||||
| 			bits += bits_macc_port(p, width); | 			bits += bits_macc_term(p, width); | ||||||
| 		return bits; | 		return bits; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -132,7 +132,7 @@ struct ShareWorker | ||||||
| 		return bits_macc(m, width); | 		return bits_macc(m, width); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	static bool cmp_macc_ports(const Macc::port_t &p1, const Macc::port_t &p2) | 	static bool cmp_macc_ports(const Macc::term_t &p1, const Macc::term_t &p2) | ||||||
| 	{ | 	{ | ||||||
| 		bool mul1 = GetSize(p1.in_a) && GetSize(p1.in_b); | 		bool mul1 = GetSize(p1.in_a) && GetSize(p1.in_b); | ||||||
| 		bool mul2 = GetSize(p2.in_a) && GetSize(p2.in_b); | 		bool mul2 = GetSize(p2.in_a) && GetSize(p2.in_b); | ||||||
|  | @ -161,7 +161,7 @@ struct ShareWorker | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	int share_macc_ports(Macc::port_t &p1, Macc::port_t &p2, int w1, int w2, | 	int share_macc_ports(Macc::term_t &p1, Macc::term_t &p2, int w1, int w2, | ||||||
| 			RTLIL::SigSpec act = RTLIL::SigSpec(), Macc *supermacc = nullptr, pool<RTLIL::Cell*> *supercell_aux = nullptr) | 			RTLIL::SigSpec act = RTLIL::SigSpec(), Macc *supermacc = nullptr, pool<RTLIL::Cell*> *supercell_aux = nullptr) | ||||||
| 	{ | 	{ | ||||||
| 		if (p1.do_subtract != p2.do_subtract) | 		if (p1.do_subtract != p2.do_subtract) | ||||||
|  | @ -216,12 +216,12 @@ struct ShareWorker | ||||||
| 				supercell_aux->insert(module->addMux(NEW_ID, sig_b2, sig_b1, act, sig_b)); | 				supercell_aux->insert(module->addMux(NEW_ID, sig_b2, sig_b1, act, sig_b)); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			Macc::port_t p; | 			Macc::term_t p; | ||||||
| 			p.in_a = sig_a; | 			p.in_a = sig_a; | ||||||
| 			p.in_b = sig_b; | 			p.in_b = sig_b; | ||||||
| 			p.is_signed = force_signed; | 			p.is_signed = force_signed; | ||||||
| 			p.do_subtract = p1.do_subtract; | 			p.do_subtract = p1.do_subtract; | ||||||
| 			supermacc->ports.push_back(p); | 			supermacc->terms.push_back(p); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		int score = 1000 + abs(GetSize(p1.in_a) - GetSize(p2.in_a)) * max(abs(GetSize(p1.in_b) - GetSize(p2.in_b)), 1); | 		int score = 1000 + abs(GetSize(p1.in_a) - GetSize(p2.in_a)) * max(abs(GetSize(p1.in_b) - GetSize(p2.in_b)), 1); | ||||||
|  | @ -248,15 +248,15 @@ struct ShareWorker | ||||||
| 		m1.optimize(w1); | 		m1.optimize(w1); | ||||||
| 		m2.optimize(w2); | 		m2.optimize(w2); | ||||||
| 
 | 
 | ||||||
| 		std::sort(m1.ports.begin(), m1.ports.end(), cmp_macc_ports); | 		std::sort(m1.terms.begin(), m1.terms.end(), cmp_macc_ports); | ||||||
| 		std::sort(m2.ports.begin(), m2.ports.end(), cmp_macc_ports); | 		std::sort(m2.terms.begin(), m2.terms.end(), cmp_macc_ports); | ||||||
| 
 | 
 | ||||||
| 		std::set<int> m1_unmapped, m2_unmapped; | 		std::set<int> m1_unmapped, m2_unmapped; | ||||||
| 
 | 
 | ||||||
| 		for (int i = 0; i < GetSize(m1.ports); i++) | 		for (int i = 0; i < GetSize(m1.terms); i++) | ||||||
| 			m1_unmapped.insert(i); | 			m1_unmapped.insert(i); | ||||||
| 
 | 
 | ||||||
| 		for (int i = 0; i < GetSize(m2.ports); i++) | 		for (int i = 0; i < GetSize(m2.terms); i++) | ||||||
| 			m2_unmapped.insert(i); | 			m2_unmapped.insert(i); | ||||||
| 
 | 
 | ||||||
| 		while (1) | 		while (1) | ||||||
|  | @ -265,7 +265,7 @@ struct ShareWorker | ||||||
| 
 | 
 | ||||||
| 			for (int i : m1_unmapped) | 			for (int i : m1_unmapped) | ||||||
| 			for (int j : m2_unmapped) { | 			for (int j : m2_unmapped) { | ||||||
| 				int score = share_macc_ports(m1.ports[i], m2.ports[j], w1, w2); | 				int score = share_macc_ports(m1.terms[i], m2.terms[j], w1, w2); | ||||||
| 				if (score >= 0 && (best_i < 0 || best_score > score)) | 				if (score >= 0 && (best_i < 0 || best_score > score)) | ||||||
| 					best_i = i, best_j = j, best_score = score; | 					best_i = i, best_j = j, best_score = score; | ||||||
| 			} | 			} | ||||||
|  | @ -273,55 +273,55 @@ struct ShareWorker | ||||||
| 			if (best_i >= 0) { | 			if (best_i >= 0) { | ||||||
| 				m1_unmapped.erase(best_i); | 				m1_unmapped.erase(best_i); | ||||||
| 				m2_unmapped.erase(best_j); | 				m2_unmapped.erase(best_j); | ||||||
| 				share_macc_ports(m1.ports[best_i], m2.ports[best_j], w1, w2, act, &supermacc, supercell_aux); | 				share_macc_ports(m1.terms[best_i], m2.terms[best_j], w1, w2, act, &supermacc, supercell_aux); | ||||||
| 			} else | 			} else | ||||||
| 				break; | 				break; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for (int i : m1_unmapped) | 		for (int i : m1_unmapped) | ||||||
| 		{ | 		{ | ||||||
| 			RTLIL::SigSpec sig_a = m1.ports[i].in_a; | 			RTLIL::SigSpec sig_a = m1.terms[i].in_a; | ||||||
| 			RTLIL::SigSpec sig_b = m1.ports[i].in_b; | 			RTLIL::SigSpec sig_b = m1.terms[i].in_b; | ||||||
| 
 | 
 | ||||||
| 			if (supercell_aux && GetSize(sig_a)) { | 			if (supercell_aux && GetSize(sig_a)) { | ||||||
| 				sig_a = module->addWire(NEW_ID, GetSize(sig_a)); | 				sig_a = module->addWire(NEW_ID, GetSize(sig_a)); | ||||||
| 				supercell_aux->insert(module->addMux(NEW_ID, RTLIL::SigSpec(0, GetSize(sig_a)), m1.ports[i].in_a, act, sig_a)); | 				supercell_aux->insert(module->addMux(NEW_ID, RTLIL::SigSpec(0, GetSize(sig_a)), m1.terms[i].in_a, act, sig_a)); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (supercell_aux && GetSize(sig_b)) { | 			if (supercell_aux && GetSize(sig_b)) { | ||||||
| 				sig_b = module->addWire(NEW_ID, GetSize(sig_b)); | 				sig_b = module->addWire(NEW_ID, GetSize(sig_b)); | ||||||
| 				supercell_aux->insert(module->addMux(NEW_ID, RTLIL::SigSpec(0, GetSize(sig_b)), m1.ports[i].in_b, act, sig_b)); | 				supercell_aux->insert(module->addMux(NEW_ID, RTLIL::SigSpec(0, GetSize(sig_b)), m1.terms[i].in_b, act, sig_b)); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			Macc::port_t p; | 			Macc::term_t p; | ||||||
| 			p.in_a = sig_a; | 			p.in_a = sig_a; | ||||||
| 			p.in_b = sig_b; | 			p.in_b = sig_b; | ||||||
| 			p.is_signed = m1.ports[i].is_signed; | 			p.is_signed = m1.terms[i].is_signed; | ||||||
| 			p.do_subtract = m1.ports[i].do_subtract; | 			p.do_subtract = m1.terms[i].do_subtract; | ||||||
| 			supermacc.ports.push_back(p); | 			supermacc.terms.push_back(p); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for (int i : m2_unmapped) | 		for (int i : m2_unmapped) | ||||||
| 		{ | 		{ | ||||||
| 			RTLIL::SigSpec sig_a = m2.ports[i].in_a; | 			RTLIL::SigSpec sig_a = m2.terms[i].in_a; | ||||||
| 			RTLIL::SigSpec sig_b = m2.ports[i].in_b; | 			RTLIL::SigSpec sig_b = m2.terms[i].in_b; | ||||||
| 
 | 
 | ||||||
| 			if (supercell_aux && GetSize(sig_a)) { | 			if (supercell_aux && GetSize(sig_a)) { | ||||||
| 				sig_a = module->addWire(NEW_ID, GetSize(sig_a)); | 				sig_a = module->addWire(NEW_ID, GetSize(sig_a)); | ||||||
| 				supercell_aux->insert(module->addMux(NEW_ID, m2.ports[i].in_a, RTLIL::SigSpec(0, GetSize(sig_a)), act, sig_a)); | 				supercell_aux->insert(module->addMux(NEW_ID, m2.terms[i].in_a, RTLIL::SigSpec(0, GetSize(sig_a)), act, sig_a)); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (supercell_aux && GetSize(sig_b)) { | 			if (supercell_aux && GetSize(sig_b)) { | ||||||
| 				sig_b = module->addWire(NEW_ID, GetSize(sig_b)); | 				sig_b = module->addWire(NEW_ID, GetSize(sig_b)); | ||||||
| 				supercell_aux->insert(module->addMux(NEW_ID, m2.ports[i].in_b, RTLIL::SigSpec(0, GetSize(sig_b)), act, sig_b)); | 				supercell_aux->insert(module->addMux(NEW_ID, m2.terms[i].in_b, RTLIL::SigSpec(0, GetSize(sig_b)), act, sig_b)); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			Macc::port_t p; | 			Macc::term_t p; | ||||||
| 			p.in_a = sig_a; | 			p.in_a = sig_a; | ||||||
| 			p.in_b = sig_b; | 			p.in_b = sig_b; | ||||||
| 			p.is_signed = m2.ports[i].is_signed; | 			p.is_signed = m2.terms[i].is_signed; | ||||||
| 			p.do_subtract = m2.ports[i].do_subtract; | 			p.do_subtract = m2.terms[i].do_subtract; | ||||||
| 			supermacc.ports.push_back(p); | 			supermacc.terms.push_back(p); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (supercell) | 		if (supercell) | ||||||
|  |  | ||||||
|  | @ -142,7 +142,7 @@ struct AlumaccWorker | ||||||
| 			log("  creating $macc model for %s (%s).\n", log_id(cell), log_id(cell->type)); | 			log("  creating $macc model for %s (%s).\n", log_id(cell), log_id(cell->type)); | ||||||
| 
 | 
 | ||||||
| 			maccnode_t *n = new maccnode_t; | 			maccnode_t *n = new maccnode_t; | ||||||
| 			Macc::port_t new_port; | 			Macc::term_t new_term; | ||||||
| 
 | 
 | ||||||
| 			n->cell = cell; | 			n->cell = cell; | ||||||
| 			n->y = sigmap(cell->getPort(ID::Y)); | 			n->y = sigmap(cell->getPort(ID::Y)); | ||||||
|  | @ -153,32 +153,32 @@ struct AlumaccWorker | ||||||
| 
 | 
 | ||||||
| 			if (cell->type.in(ID($pos), ID($neg))) | 			if (cell->type.in(ID($pos), ID($neg))) | ||||||
| 			{ | 			{ | ||||||
| 				new_port.in_a = sigmap(cell->getPort(ID::A)); | 				new_term.in_a = sigmap(cell->getPort(ID::A)); | ||||||
| 				new_port.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | ||||||
| 				new_port.do_subtract = cell->type == ID($neg); | 				new_term.do_subtract = cell->type == ID($neg); | ||||||
| 				n->macc.ports.push_back(new_port); | 				n->macc.terms.push_back(new_term); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (cell->type.in(ID($add), ID($sub))) | 			if (cell->type.in(ID($add), ID($sub))) | ||||||
| 			{ | 			{ | ||||||
| 				new_port.in_a = sigmap(cell->getPort(ID::A)); | 				new_term.in_a = sigmap(cell->getPort(ID::A)); | ||||||
| 				new_port.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | ||||||
| 				new_port.do_subtract = false; | 				new_term.do_subtract = false; | ||||||
| 				n->macc.ports.push_back(new_port); | 				n->macc.terms.push_back(new_term); | ||||||
| 
 | 
 | ||||||
| 				new_port.in_a = sigmap(cell->getPort(ID::B)); | 				new_term.in_a = sigmap(cell->getPort(ID::B)); | ||||||
| 				new_port.is_signed = cell->getParam(ID::B_SIGNED).as_bool(); | 				new_term.is_signed = cell->getParam(ID::B_SIGNED).as_bool(); | ||||||
| 				new_port.do_subtract = cell->type == ID($sub); | 				new_term.do_subtract = cell->type == ID($sub); | ||||||
| 				n->macc.ports.push_back(new_port); | 				n->macc.terms.push_back(new_term); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			if (cell->type.in(ID($mul))) | 			if (cell->type.in(ID($mul))) | ||||||
| 			{ | 			{ | ||||||
| 				new_port.in_a = sigmap(cell->getPort(ID::A)); | 				new_term.in_a = sigmap(cell->getPort(ID::A)); | ||||||
| 				new_port.in_b = sigmap(cell->getPort(ID::B)); | 				new_term.in_b = sigmap(cell->getPort(ID::B)); | ||||||
| 				new_port.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool(); | ||||||
| 				new_port.do_subtract = false; | 				new_term.do_subtract = false; | ||||||
| 				n->macc.ports.push_back(new_port); | 				n->macc.terms.push_back(new_term); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			log_assert(sig_macc.count(n->y) == 0); | 			log_assert(sig_macc.count(n->y) == 0); | ||||||
|  | @ -190,7 +190,7 @@ struct AlumaccWorker | ||||||
| 	{ | 	{ | ||||||
| 		std::vector<int> port_sizes; | 		std::vector<int> port_sizes; | ||||||
| 
 | 
 | ||||||
| 		for (auto &port : macc.ports) { | 		for (auto &port : macc.terms) { | ||||||
| 			if (port.is_signed != is_signed) | 			if (port.is_signed != is_signed) | ||||||
| 				return true; | 				return true; | ||||||
| 			if (!port.is_signed && port.do_subtract) | 			if (!port.is_signed && port.do_subtract) | ||||||
|  | @ -235,9 +235,9 @@ struct AlumaccWorker | ||||||
| 				if (delete_nodes.count(n)) | 				if (delete_nodes.count(n)) | ||||||
| 					continue; | 					continue; | ||||||
| 
 | 
 | ||||||
| 				for (int i = 0; i < GetSize(n->macc.ports); i++) | 				for (int i = 0; i < GetSize(n->macc.terms); i++) | ||||||
| 				{ | 				{ | ||||||
| 					auto &port = n->macc.ports[i]; | 					auto &port = n->macc.terms[i]; | ||||||
| 
 | 
 | ||||||
| 					if (GetSize(port.in_b) > 0 || sig_macc.count(port.in_a) == 0) | 					if (GetSize(port.in_b) > 0 || sig_macc.count(port.in_a) == 0) | ||||||
| 						continue; | 						continue; | ||||||
|  | @ -253,13 +253,13 @@ struct AlumaccWorker | ||||||
| 					log("  merging $macc model for %s into %s.\n", log_id(other_n->cell), log_id(n->cell)); | 					log("  merging $macc model for %s into %s.\n", log_id(other_n->cell), log_id(n->cell)); | ||||||
| 
 | 
 | ||||||
| 					bool do_subtract = port.do_subtract; | 					bool do_subtract = port.do_subtract; | ||||||
| 					for (int j = 0; j < GetSize(other_n->macc.ports); j++) { | 					for (int j = 0; j < GetSize(other_n->macc.terms); j++) { | ||||||
| 						if (do_subtract) | 						if (do_subtract) | ||||||
| 							other_n->macc.ports[j].do_subtract = !other_n->macc.ports[j].do_subtract; | 							other_n->macc.terms[j].do_subtract = !other_n->macc.terms[j].do_subtract; | ||||||
| 						if (j == 0) | 						if (j == 0) | ||||||
| 							n->macc.ports[i--] = other_n->macc.ports[j]; | 							n->macc.terms[i--] = other_n->macc.terms[j]; | ||||||
| 						else | 						else | ||||||
| 							n->macc.ports.push_back(other_n->macc.ports[j]); | 							n->macc.terms.push_back(other_n->macc.terms[j]); | ||||||
| 					} | 					} | ||||||
| 
 | 
 | ||||||
| 					delete_nodes.insert(other_n); | 					delete_nodes.insert(other_n); | ||||||
|  | @ -288,7 +288,7 @@ struct AlumaccWorker | ||||||
| 			bool subtract_b = false; | 			bool subtract_b = false; | ||||||
| 			alunode_t *alunode; | 			alunode_t *alunode; | ||||||
| 
 | 
 | ||||||
| 			for (auto &port : n->macc.ports) | 			for (auto &port : n->macc.terms) | ||||||
| 				if (GetSize(port.in_b) > 0) { | 				if (GetSize(port.in_b) > 0) { | ||||||
| 					goto next_macc; | 					goto next_macc; | ||||||
| 				} else if (GetSize(port.in_a) == 1 && !port.is_signed && !port.do_subtract) { | 				} else if (GetSize(port.in_a) == 1 && !port.is_signed && !port.do_subtract) { | ||||||
|  |  | ||||||
|  | @ -227,9 +227,9 @@ struct BoothPassWorker { | ||||||
| 					continue; | 					continue; | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				A = macc.ports[0].in_a; | 				A = macc.terms[0].in_a; | ||||||
| 				B = macc.ports[0].in_b; | 				B = macc.terms[0].in_b; | ||||||
| 				is_signed = macc.ports[0].is_signed; | 				is_signed = macc.terms[0].is_signed; | ||||||
| 				Y = cell->getPort(ID::Y); | 				Y = cell->getPort(ID::Y); | ||||||
| 			} else { | 			} else { | ||||||
| 				continue; | 				continue; | ||||||
|  |  | ||||||
|  | @ -278,42 +278,42 @@ void maccmap(RTLIL::Module *module, RTLIL::Cell *cell, bool unmap) | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for (auto &port : macc.ports) | 	for (auto &term : macc.terms) | ||||||
| 		if (GetSize(port.in_b) == 0) | 		if (GetSize(term.in_b) == 0) | ||||||
| 			log("  %s %s (%d bits, %s)\n", port.do_subtract ? "sub" : "add", log_signal(port.in_a), | 			log("  %s %s (%d bits, %s)\n", term.do_subtract ? "sub" : "add", log_signal(term.in_a), | ||||||
| 					GetSize(port.in_a), port.is_signed ? "signed" : "unsigned"); | 					GetSize(term.in_a), term.is_signed ? "signed" : "unsigned"); | ||||||
| 		else | 		else | ||||||
| 			log("  %s %s * %s (%dx%d bits, %s)\n", port.do_subtract ? "sub" : "add", log_signal(port.in_a), log_signal(port.in_b), | 			log("  %s %s * %s (%dx%d bits, %s)\n", term.do_subtract ? "sub" : "add", log_signal(term.in_a), log_signal(term.in_b), | ||||||
| 					GetSize(port.in_a), GetSize(port.in_b), port.is_signed ? "signed" : "unsigned"); | 					GetSize(term.in_a), GetSize(term.in_b), term.is_signed ? "signed" : "unsigned"); | ||||||
| 
 | 
 | ||||||
| 	if (unmap) | 	if (unmap) | ||||||
| 	{ | 	{ | ||||||
| 		typedef std::pair<RTLIL::SigSpec, bool> summand_t; | 		typedef std::pair<RTLIL::SigSpec, bool> summand_t; | ||||||
| 		std::vector<summand_t> summands; | 		std::vector<summand_t> summands; | ||||||
| 
 | 
 | ||||||
| 		RTLIL::SigSpec bit_ports; | 		RTLIL::SigSpec bit_terms; | ||||||
| 
 | 
 | ||||||
| 		for (auto &port : macc.ports) { | 		for (auto &term : macc.terms) { | ||||||
| 			summand_t this_summand; | 			summand_t this_summand; | ||||||
| 			if (GetSize(port.in_b)) { | 			if (GetSize(term.in_b)) { | ||||||
| 				this_summand.first = module->addWire(NEW_ID, width); | 				this_summand.first = module->addWire(NEW_ID, width); | ||||||
| 				module->addMul(NEW_ID, port.in_a, port.in_b, this_summand.first, port.is_signed); | 				module->addMul(NEW_ID, term.in_a, term.in_b, this_summand.first, term.is_signed); | ||||||
| 			} else if (GetSize(port.in_a) == 1 && GetSize(port.in_b) == 0 && !port.is_signed && !port.do_subtract) { | 			} else if (GetSize(term.in_a) == 1 && GetSize(term.in_b) == 0 && !term.is_signed && !term.do_subtract) { | ||||||
| 				// Mimic old 'bit_ports' treatment in case it's relevant for performance,
 | 				// Mimic old 'bit_terms' treatment in case it's relevant for performance,
 | ||||||
| 				// i.e. defer single-bit summands to be the last ones
 | 				// i.e. defer single-bit summands to be the last ones
 | ||||||
| 				bit_ports.append(port.in_a); | 				bit_terms.append(term.in_a); | ||||||
| 				continue; | 				continue; | ||||||
| 			} else if (GetSize(port.in_a) != width) { | 			} else if (GetSize(term.in_a) != width) { | ||||||
| 				this_summand.first = module->addWire(NEW_ID, width); | 				this_summand.first = module->addWire(NEW_ID, width); | ||||||
| 				module->addPos(NEW_ID, port.in_a, this_summand.first, port.is_signed); | 				module->addPos(NEW_ID, term.in_a, this_summand.first, term.is_signed); | ||||||
| 			} else { | 			} else { | ||||||
| 				this_summand.first = port.in_a; | 				this_summand.first = term.in_a; | ||||||
| 			} | 			} | ||||||
| 			this_summand.second = port.do_subtract; | 			this_summand.second = term.do_subtract; | ||||||
| 			summands.push_back(this_summand); | 			summands.push_back(this_summand); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for (auto &bit : bit_ports) | 		for (auto &bit : bit_terms) | ||||||
| 			summands.push_back(summand_t(bit, false)); | 			summands.push_back(summand_t(bit, false)); | ||||||
| 
 | 
 | ||||||
| 		if (GetSize(summands) == 0) | 		if (GetSize(summands) == 0) | ||||||
|  | @ -350,20 +350,20 @@ void maccmap(RTLIL::Module *module, RTLIL::Cell *cell, bool unmap) | ||||||
| 	else | 	else | ||||||
| 	{ | 	{ | ||||||
| 		MaccmapWorker worker(module, width); | 		MaccmapWorker worker(module, width); | ||||||
| 		RTLIL::SigSpec bit_ports; | 		RTLIL::SigSpec bit_terms; | ||||||
| 
 | 
 | ||||||
| 		for (auto &port : macc.ports) { | 		for (auto &term : macc.terms) { | ||||||
| 			// Mimic old 'bit_ports' treatment in case it's relevant for performance,
 | 			// Mimic old 'bit_terms' treatment in case it's relevant for performance,
 | ||||||
| 			// i.e. defer single-bit summands to be the last ones
 | 			// i.e. defer single-bit summands to be the last ones
 | ||||||
| 			if (GetSize(port.in_a) == 1 && GetSize(port.in_b) == 0 && !port.is_signed && !port.do_subtract) | 			if (GetSize(term.in_a) == 1 && GetSize(term.in_b) == 0 && !term.is_signed && !term.do_subtract) | ||||||
| 				bit_ports.append(port.in_a); | 				bit_terms.append(term.in_a); | ||||||
| 			else if (GetSize(port.in_b) == 0) | 			else if (GetSize(term.in_b) == 0) | ||||||
| 				worker.add(port.in_a, port.is_signed, port.do_subtract); | 				worker.add(term.in_a, term.is_signed, term.do_subtract); | ||||||
| 			else | 			else | ||||||
| 				worker.add(port.in_a, port.in_b, port.is_signed, port.do_subtract); | 				worker.add(term.in_a, term.in_b, term.is_signed, term.do_subtract); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for (auto bit : bit_ports) | 		for (auto bit : bit_terms) | ||||||
| 			worker.add(bit, 0); | 			worker.add(bit, 0); | ||||||
| 
 | 
 | ||||||
| 		module->connect(cell->getPort(ID::Y), worker.synth()); | 		module->connect(cell->getPort(ID::Y), worker.synth()); | ||||||
|  |  | ||||||
|  | @ -189,17 +189,17 @@ static RTLIL::Cell* create_gold_module(RTLIL::Design *design, RTLIL::IdString ce | ||||||
| 			} else | 			} else | ||||||
| 				size_b = 0; | 				size_b = 0; | ||||||
| 
 | 
 | ||||||
| 			Macc::port_t this_port; | 			Macc::term_t this_term; | ||||||
| 
 | 
 | ||||||
| 			wire_a->width += size_a; | 			wire_a->width += size_a; | ||||||
| 			this_port.in_a = RTLIL::SigSpec(wire_a, wire_a->width - size_a, size_a); | 			this_term.in_a = RTLIL::SigSpec(wire_a, wire_a->width - size_a, size_a); | ||||||
| 
 | 
 | ||||||
| 			wire_a->width += size_b; | 			wire_a->width += size_b; | ||||||
| 			this_port.in_b = RTLIL::SigSpec(wire_a, wire_a->width - size_b, size_b); | 			this_term.in_b = RTLIL::SigSpec(wire_a, wire_a->width - size_b, size_b); | ||||||
| 
 | 
 | ||||||
| 			this_port.is_signed = xorshift32(2) == 1; | 			this_term.is_signed = xorshift32(2) == 1; | ||||||
| 			this_port.do_subtract = xorshift32(2) == 1; | 			this_term.do_subtract = xorshift32(2) == 1; | ||||||
| 			macc.ports.push_back(this_port); | 			macc.terms.push_back(this_term); | ||||||
| 		} | 		} | ||||||
| 		// Macc::to_cell sets the input ports
 | 		// Macc::to_cell sets the input ports
 | ||||||
| 		macc.to_cell(cell); | 		macc.to_cell(cell); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue