mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-26 09:24:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			613 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			613 lines
		
	
	
	
		
			15 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 "kernel/macc.h"
 | |
| 
 | |
| USING_YOSYS_NAMESPACE
 | |
| PRIVATE_NAMESPACE_BEGIN
 | |
| 
 | |
| struct AlumaccWorker
 | |
| {
 | |
| 	RTLIL::Module *module;
 | |
| 	SigMap sigmap;
 | |
| 
 | |
| 	struct maccnode_t {
 | |
| 		Macc macc;
 | |
| 		RTLIL::Cell *cell;
 | |
| 		RTLIL::SigSpec y;
 | |
| 		int users;
 | |
| 	};
 | |
| 
 | |
| 	struct alunode_t
 | |
| 	{
 | |
| 		std::vector<RTLIL::Cell*> cells;
 | |
| 		RTLIL::SigSpec a, b, c, y;
 | |
| 		std::vector<tuple<bool, bool, bool, bool, bool, RTLIL::SigSpec>> cmp;
 | |
| 		bool is_signed, invert_b;
 | |
| 
 | |
| 		RTLIL::Cell *alu_cell;
 | |
| 		RTLIL::SigSpec cached_lt, cached_slt, cached_gt, cached_sgt, cached_eq, cached_ne;
 | |
| 		RTLIL::SigSpec cached_cf, cached_of, cached_sf;
 | |
| 
 | |
| 		RTLIL::SigSpec get_lt(bool is_signed) {
 | |
| 			if (is_signed) {
 | |
| 				if (GetSize(cached_slt) == 0) {
 | |
| 					get_of();
 | |
| 					get_sf();
 | |
| 					cached_slt = alu_cell->module->Xor(NEW_ID, cached_of, cached_sf);
 | |
| 				}
 | |
| 
 | |
| 				return cached_slt;
 | |
| 			} else {
 | |
| 				if (GetSize(cached_lt) == 0) {
 | |
| 					cached_lt = get_cf();
 | |
| 				}
 | |
| 
 | |
| 				return cached_lt;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_gt(bool is_signed) {
 | |
| 			if (is_signed) {
 | |
| 				if (GetSize(cached_sgt) == 0) {
 | |
| 					get_lt(is_signed);
 | |
| 					get_eq();
 | |
| 					SigSpec Or = alu_cell->module->Or(NEW_ID, cached_slt, cached_eq);
 | |
| 					cached_sgt = alu_cell->module->Not(NEW_ID, Or, false, alu_cell->get_src_attribute());
 | |
| 				}
 | |
| 
 | |
| 				return cached_sgt;
 | |
| 			} else {
 | |
| 				if (GetSize(cached_gt) == 0) {
 | |
| 					get_lt(is_signed);
 | |
| 					get_eq();
 | |
| 					SigSpec Or = alu_cell->module->Or(NEW_ID, cached_lt, cached_eq);
 | |
| 					cached_gt = alu_cell->module->Not(NEW_ID, Or, false, alu_cell->get_src_attribute());
 | |
| 				}
 | |
| 
 | |
| 				return cached_gt;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_eq() {
 | |
| 			if (GetSize(cached_eq) == 0)
 | |
| 				cached_eq = alu_cell->module->ReduceAnd(NEW_ID, alu_cell->getPort(ID::X), false, alu_cell->get_src_attribute());
 | |
| 			return cached_eq;
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_ne() {
 | |
| 			if (GetSize(cached_ne) == 0)
 | |
| 				cached_ne = alu_cell->module->Not(NEW_ID, get_eq(), false, alu_cell->get_src_attribute());
 | |
| 			return cached_ne;
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_cf() {
 | |
| 			if (GetSize(cached_cf) == 0) {
 | |
| 				cached_cf = alu_cell->getPort(ID::CO);
 | |
| 				log_assert(GetSize(cached_cf) >= 1);
 | |
| 				cached_cf = alu_cell->module->Not(NEW_ID, cached_cf[GetSize(cached_cf)-1], false, alu_cell->get_src_attribute());
 | |
| 			}
 | |
| 			return cached_cf;
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_of() {
 | |
| 			if (GetSize(cached_of) == 0) {
 | |
| 				cached_of = {alu_cell->getPort(ID::CO), alu_cell->getPort(ID::CI)};
 | |
| 				log_assert(GetSize(cached_of) >= 2);
 | |
| 				cached_of = alu_cell->module->Xor(NEW_ID, cached_of[GetSize(cached_of)-1], cached_of[GetSize(cached_of)-2]);
 | |
| 			}
 | |
| 			return cached_of;
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec get_sf() {
 | |
| 			if (GetSize(cached_sf) == 0) {
 | |
| 				cached_sf = alu_cell->getPort(ID::Y);
 | |
| 				cached_sf = cached_sf[GetSize(cached_sf)-1];
 | |
| 			}
 | |
| 			return cached_sf;
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	dict<RTLIL::SigBit, int> bit_users;
 | |
| 	dict<RTLIL::SigSpec, maccnode_t*> sig_macc;
 | |
| 	dict<RTLIL::SigSig, pool<alunode_t*>> sig_alu;
 | |
| 	int macc_counter, alu_counter;
 | |
| 
 | |
| 	AlumaccWorker(RTLIL::Module *module) : module(module), sigmap(module)
 | |
| 	{
 | |
| 		macc_counter = 0;
 | |
| 		alu_counter = 0;
 | |
| 	}
 | |
| 
 | |
| 	void count_bit_users()
 | |
| 	{
 | |
| 		for (auto port : module->ports)
 | |
| 			for (auto bit : sigmap(module->wire(port)))
 | |
| 				bit_users[bit]++;
 | |
| 
 | |
| 		for (auto cell : module->cells())
 | |
| 		for (auto &conn : cell->connections())
 | |
| 			for (auto bit : sigmap(conn.second))
 | |
| 				bit_users[bit]++;
 | |
| 	}
 | |
| 
 | |
| 	void extract_macc()
 | |
| 	{
 | |
| 		for (auto cell : module->selected_cells())
 | |
| 		{
 | |
| 			if (!cell->type.in(ID($pos), ID($neg), ID($add), ID($sub), ID($mul)))
 | |
| 				continue;
 | |
| 
 | |
| 			log("  creating $macc model for %s (%s).\n", log_id(cell), log_id(cell->type));
 | |
| 
 | |
| 			maccnode_t *n = new maccnode_t;
 | |
| 			Macc::term_t new_term;
 | |
| 
 | |
| 			n->cell = cell;
 | |
| 			n->y = sigmap(cell->getPort(ID::Y));
 | |
| 			n->users = 0;
 | |
| 
 | |
| 			for (auto bit : n->y)
 | |
| 				n->users = max(n->users, bit_users.at(bit) - 1);
 | |
| 
 | |
| 			if (cell->type.in(ID($pos), ID($neg)))
 | |
| 			{
 | |
| 				new_term.in_a = sigmap(cell->getPort(ID::A));
 | |
| 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool();
 | |
| 				new_term.do_subtract = cell->type == ID($neg);
 | |
| 				n->macc.terms.push_back(new_term);
 | |
| 			}
 | |
| 
 | |
| 			if (cell->type.in(ID($add), ID($sub)))
 | |
| 			{
 | |
| 				new_term.in_a = sigmap(cell->getPort(ID::A));
 | |
| 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool();
 | |
| 				new_term.do_subtract = false;
 | |
| 				n->macc.terms.push_back(new_term);
 | |
| 
 | |
| 				new_term.in_a = sigmap(cell->getPort(ID::B));
 | |
| 				new_term.is_signed = cell->getParam(ID::B_SIGNED).as_bool();
 | |
| 				new_term.do_subtract = cell->type == ID($sub);
 | |
| 				n->macc.terms.push_back(new_term);
 | |
| 			}
 | |
| 
 | |
| 			if (cell->type.in(ID($mul)))
 | |
| 			{
 | |
| 				new_term.in_a = sigmap(cell->getPort(ID::A));
 | |
| 				new_term.in_b = sigmap(cell->getPort(ID::B));
 | |
| 				new_term.is_signed = cell->getParam(ID::A_SIGNED).as_bool();
 | |
| 				new_term.do_subtract = false;
 | |
| 				n->macc.terms.push_back(new_term);
 | |
| 			}
 | |
| 
 | |
| 			log_assert(sig_macc.count(n->y) == 0);
 | |
| 			sig_macc[n->y] = n;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	static bool macc_may_overflow(Macc &macc, int width, bool is_signed)
 | |
| 	{
 | |
| 		std::vector<int> port_sizes;
 | |
| 
 | |
| 		for (auto &port : macc.terms) {
 | |
| 			if (port.is_signed != is_signed)
 | |
| 				return true;
 | |
| 			if (!port.is_signed && port.do_subtract)
 | |
| 				return true;
 | |
| 			if (GetSize(port.in_b))
 | |
| 				port_sizes.push_back(GetSize(port.in_a) + GetSize(port.in_b));
 | |
| 			else
 | |
| 				port_sizes.push_back(GetSize(port.in_a));
 | |
| 		}
 | |
| 
 | |
| 		std::sort(port_sizes.begin(), port_sizes.end());
 | |
| 
 | |
| 		int acc_sum = 0, acc_shift = 0;
 | |
| 		for (int sz : port_sizes) {
 | |
| 			while ((sz - acc_shift) > 20) {
 | |
| 				if (acc_sum & 1)
 | |
| 					acc_sum++;
 | |
| 				acc_sum = acc_sum >> 1;
 | |
| 				acc_shift++;
 | |
| 			}
 | |
| 			acc_sum += (1 << (sz - acc_shift)) - 1;
 | |
| 		}
 | |
| 
 | |
| 		while (acc_sum) {
 | |
| 			acc_sum = acc_sum >> 1;
 | |
| 			acc_shift++;
 | |
| 		}
 | |
| 
 | |
| 		return acc_shift > width;
 | |
| 	}
 | |
| 
 | |
| 	void merge_macc()
 | |
| 	{
 | |
| 		while (1)
 | |
| 		{
 | |
| 			pool<maccnode_t*> delete_nodes;
 | |
| 
 | |
| 			for (auto &it : sig_macc)
 | |
| 			{
 | |
| 				auto n = it.second;
 | |
| 
 | |
| 				if (delete_nodes.count(n))
 | |
| 					continue;
 | |
| 
 | |
| 				for (int i = 0; i < GetSize(n->macc.terms); i++)
 | |
| 				{
 | |
| 					auto &port = n->macc.terms[i];
 | |
| 
 | |
| 					if (GetSize(port.in_b) > 0 || sig_macc.count(port.in_a) == 0)
 | |
| 						continue;
 | |
| 
 | |
| 					auto other_n = sig_macc.at(port.in_a);
 | |
| 
 | |
| 					if (other_n->users > 1)
 | |
| 						continue;
 | |
| 
 | |
| 					if (GetSize(other_n->y) != GetSize(n->y) && macc_may_overflow(other_n->macc, GetSize(other_n->y), port.is_signed))
 | |
| 						continue;
 | |
| 
 | |
| 					log("  merging $macc model for %s into %s.\n", log_id(other_n->cell), log_id(n->cell));
 | |
| 
 | |
| 					bool do_subtract = port.do_subtract;
 | |
| 					for (int j = 0; j < GetSize(other_n->macc.terms); j++) {
 | |
| 						if (do_subtract)
 | |
| 							other_n->macc.terms[j].do_subtract = !other_n->macc.terms[j].do_subtract;
 | |
| 						if (j == 0)
 | |
| 							n->macc.terms[i--] = other_n->macc.terms[j];
 | |
| 						else
 | |
| 							n->macc.terms.push_back(other_n->macc.terms[j]);
 | |
| 					}
 | |
| 
 | |
| 					delete_nodes.insert(other_n);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (delete_nodes.empty())
 | |
| 				break;
 | |
| 
 | |
| 			for (auto n : delete_nodes) {
 | |
| 				sig_macc.erase(n->y);
 | |
| 				delete n;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void macc_to_alu()
 | |
| 	{
 | |
| 		pool<maccnode_t*> delete_nodes;
 | |
| 
 | |
| 		for (auto &it : sig_macc)
 | |
| 		{
 | |
| 			auto n = it.second;
 | |
| 			RTLIL::SigSpec A, B, C;
 | |
| 			bool a_signed = false, b_signed = false;
 | |
| 			bool subtract_b = false;
 | |
| 			alunode_t *alunode;
 | |
| 
 | |
| 			for (auto &port : n->macc.terms)
 | |
| 				if (GetSize(port.in_b) > 0) {
 | |
| 					goto next_macc;
 | |
| 				} else if (GetSize(port.in_a) == 1 && !port.is_signed && !port.do_subtract) {
 | |
| 					C.append(port.in_a);
 | |
| 				} else if (GetSize(A) || port.do_subtract) {
 | |
| 					if (GetSize(B))
 | |
| 						goto next_macc;
 | |
| 					B = port.in_a;
 | |
| 					b_signed = port.is_signed;
 | |
| 					subtract_b = port.do_subtract;
 | |
| 				} else {
 | |
| 					if (GetSize(A))
 | |
| 						goto next_macc;
 | |
| 					A = port.in_a;
 | |
| 					a_signed = port.is_signed;
 | |
| 				}
 | |
| 
 | |
| 			if (!a_signed || !b_signed) {
 | |
| 				if (GetSize(A) == GetSize(n->y))
 | |
| 					a_signed = false;
 | |
| 				if (GetSize(B) == GetSize(n->y))
 | |
| 					b_signed = false;
 | |
| 				if (a_signed != b_signed)
 | |
| 					goto next_macc;
 | |
| 			}
 | |
| 
 | |
| 			if (GetSize(A) == 0 && GetSize(C) > 0) {
 | |
| 				A = C[0];
 | |
| 				C.remove(0);
 | |
| 			}
 | |
| 
 | |
| 			if (GetSize(B) == 0 && GetSize(C) > 0) {
 | |
| 				B = C[0];
 | |
| 				C.remove(0);
 | |
| 			}
 | |
| 
 | |
| 			if (subtract_b)
 | |
| 				C.append(State::S1);
 | |
| 
 | |
| 			if (GetSize(C) > 1)
 | |
| 				goto next_macc;
 | |
| 
 | |
| 			if (!subtract_b && B < A && GetSize(B))
 | |
| 				std::swap(A, B);
 | |
| 
 | |
| 			log("  creating $alu model for $macc %s.\n", log_id(n->cell));
 | |
| 
 | |
| 			alunode = new alunode_t;
 | |
| 			alunode->cells.push_back(n->cell);
 | |
| 			alunode->is_signed = a_signed;
 | |
| 			alunode->invert_b = subtract_b;
 | |
| 
 | |
| 			alunode->a = A;
 | |
| 			alunode->b = B;
 | |
| 			alunode->c = C;
 | |
| 			alunode->y = n->y;
 | |
| 
 | |
| 			sig_alu[RTLIL::SigSig(A, B)].insert(alunode);
 | |
| 			delete_nodes.insert(n);
 | |
| 		next_macc:;
 | |
| 		}
 | |
| 
 | |
| 		for (auto n : delete_nodes) {
 | |
| 			sig_macc.erase(n->y);
 | |
| 			delete n;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void replace_macc()
 | |
| 	{
 | |
| 		for (auto &it : sig_macc)
 | |
| 		{
 | |
| 			auto n = it.second;
 | |
| 			auto cell = module->addCell(NEW_ID, ID($macc));
 | |
| 
 | |
| 			macc_counter++;
 | |
| 
 | |
| 			log("  creating $macc cell for %s: %s\n", log_id(n->cell), log_id(cell));
 | |
| 
 | |
| 			cell->set_src_attribute(n->cell->get_src_attribute());
 | |
| 
 | |
| 			n->macc.optimize(GetSize(n->y));
 | |
| 			n->macc.to_cell(cell);
 | |
| 			cell->setPort(ID::Y, n->y);
 | |
| 			cell->fixup_parameters();
 | |
| 			module->remove(n->cell);
 | |
| 			delete n;
 | |
| 		}
 | |
| 
 | |
| 		sig_macc.clear();
 | |
| 	}
 | |
| 
 | |
| 	void extract_cmp_alu()
 | |
| 	{
 | |
| 		std::vector<RTLIL::Cell*> lge_cells, eq_cells;
 | |
| 
 | |
| 		for (auto cell : module->selected_cells())
 | |
| 		{
 | |
| 			if (cell->type.in(ID($lt), ID($le), ID($ge), ID($gt)))
 | |
| 				lge_cells.push_back(cell);
 | |
| 			if (cell->type.in(ID($eq), ID($eqx), ID($ne), ID($nex)))
 | |
| 				eq_cells.push_back(cell);
 | |
| 		}
 | |
| 
 | |
| 		for (auto cell : lge_cells)
 | |
| 		{
 | |
| 			log("  creating $alu model for %s (%s):", log_id(cell), log_id(cell->type));
 | |
| 
 | |
| 			bool cmp_less = cell->type.in(ID($lt), ID($le));
 | |
| 			bool cmp_equal = cell->type.in(ID($le), ID($ge));
 | |
| 			bool is_signed = cell->getParam(ID::A_SIGNED).as_bool();
 | |
| 
 | |
| 			RTLIL::SigSpec A = sigmap(cell->getPort(ID::A));
 | |
| 			RTLIL::SigSpec B = sigmap(cell->getPort(ID::B));
 | |
| 			RTLIL::SigSpec Y = sigmap(cell->getPort(ID::Y));
 | |
| 
 | |
| 			alunode_t *n = nullptr;
 | |
| 
 | |
| 			for (auto node : sig_alu[RTLIL::SigSig(A, B)])
 | |
| 				if (node->invert_b && node->c == State::S1) {
 | |
| 					n = node;
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 			if (n == nullptr) {
 | |
| 				for (auto node : sig_alu[RTLIL::SigSig(B, A)])
 | |
| 					if (node->is_signed == is_signed && node->invert_b && node->c == State::S1) {
 | |
| 						n = node;
 | |
| 						cmp_less = !cmp_less;
 | |
| 						std::swap(A, B);
 | |
| 						break;
 | |
| 					}
 | |
| 			}
 | |
| 
 | |
| 			if (n == nullptr) {
 | |
| 				n = new alunode_t;
 | |
| 				n->a = A;
 | |
| 				n->b = B;
 | |
| 				n->c = State::S1;
 | |
| 				n->y = module->addWire(NEW_ID, max(GetSize(A), GetSize(B)));
 | |
| 				n->is_signed = is_signed;
 | |
| 				n->invert_b = true;
 | |
| 				sig_alu[RTLIL::SigSig(A, B)].insert(n);
 | |
| 				log(" new $alu\n");
 | |
| 			} else {
 | |
| 				log(" merged with %s.\n", log_id(n->cells.front()));
 | |
| 			}
 | |
| 
 | |
| 			n->cells.push_back(cell);
 | |
| 			n->cmp.push_back(std::make_tuple(cmp_less, !cmp_less, cmp_equal, false, is_signed, Y));
 | |
| 		}
 | |
| 
 | |
| 		for (auto cell : eq_cells)
 | |
| 		{
 | |
| 			bool cmp_equal = cell->type.in(ID($eq), ID($eqx));
 | |
| 			bool is_signed = cell->getParam(ID::A_SIGNED).as_bool();
 | |
| 
 | |
| 			RTLIL::SigSpec A = sigmap(cell->getPort(ID::A));
 | |
| 			RTLIL::SigSpec B = sigmap(cell->getPort(ID::B));
 | |
| 			RTLIL::SigSpec Y = sigmap(cell->getPort(ID::Y));
 | |
| 
 | |
| 			alunode_t *n = nullptr;
 | |
| 
 | |
| 			for (auto node : sig_alu[RTLIL::SigSig(A, B)])
 | |
| 				if (node->invert_b && node->c == State::S1) {
 | |
| 					n = node;
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 			if (n == nullptr) {
 | |
| 				for (auto node : sig_alu[RTLIL::SigSig(B, A)])
 | |
| 					if (node->is_signed == is_signed && node->invert_b && node->c == State::S1) {
 | |
| 						n = node;
 | |
| 						break;
 | |
| 					}
 | |
| 			}
 | |
| 
 | |
| 			if (n != nullptr) {
 | |
| 				log("  creating $alu model for %s (%s): merged with %s.\n", log_id(cell), log_id(cell->type), log_id(n->cells.front()));
 | |
| 				n->cells.push_back(cell);
 | |
| 				n->cmp.push_back(std::make_tuple(false, false, cmp_equal, !cmp_equal, false, Y));
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void replace_alu()
 | |
| 	{
 | |
| 		std::string src("");
 | |
| 		for (auto &it1 : sig_alu)
 | |
| 		for (auto n : it1.second)
 | |
| 		{
 | |
| 			if (GetSize(n->b) == 0 && GetSize(n->c) == 0 && GetSize(n->cmp) == 0)
 | |
| 			{
 | |
| 				n->alu_cell = module->addPos(NEW_ID, n->a, n->y, n->is_signed);
 | |
| 
 | |
| 				log("  creating $pos cell for ");
 | |
| 				for (int i = 0; i < GetSize(n->cells); i++)
 | |
| 					log("%s%s", i ? ", ": "", log_id(n->cells[i]));
 | |
| 				log(": %s\n", log_id(n->alu_cell));
 | |
| 
 | |
| 				goto delete_node;
 | |
| 			}
 | |
| 
 | |
| 			n->alu_cell = module->addCell(NEW_ID, ID($alu));
 | |
| 			alu_counter++;
 | |
| 
 | |
| 			log("  creating $alu cell for ");
 | |
| 			for (int i = 0; i < GetSize(n->cells); i++)
 | |
| 				log("%s%s", i ? ", ": "", log_id(n->cells[i]));
 | |
| 			log(": %s\n", log_id(n->alu_cell));
 | |
| 
 | |
| 			if (n->cells.size() > 0)
 | |
| 				n->alu_cell->set_src_attribute(n->cells[0]->get_src_attribute());
 | |
| 
 | |
| 			n->alu_cell->setPort(ID::A, n->a);
 | |
| 			n->alu_cell->setPort(ID::B, n->b);
 | |
| 			n->alu_cell->setPort(ID::CI, GetSize(n->c) ? n->c : State::S0);
 | |
| 			n->alu_cell->setPort(ID::BI, n->invert_b ? State::S1 : State::S0);
 | |
| 			n->alu_cell->setPort(ID::Y, n->y);
 | |
| 			n->alu_cell->setPort(ID::X, module->addWire(NEW_ID, GetSize(n->y)));
 | |
| 			n->alu_cell->setPort(ID::CO, module->addWire(NEW_ID, GetSize(n->y)));
 | |
| 			n->alu_cell->fixup_parameters(n->is_signed, n->is_signed);
 | |
| 
 | |
| 			for (auto &it : n->cmp)
 | |
| 			{
 | |
| 				bool cmp_lt = std::get<0>(it);
 | |
| 				bool cmp_gt = std::get<1>(it);
 | |
| 				bool cmp_eq = std::get<2>(it);
 | |
| 				bool cmp_ne = std::get<3>(it);
 | |
| 				bool is_signed = std::get<4>(it);
 | |
| 				RTLIL::SigSpec cmp_y = std::get<5>(it);
 | |
| 
 | |
| 				RTLIL::SigSpec sig;
 | |
| 				if (cmp_lt) sig.append(n->get_lt(is_signed));
 | |
| 				if (cmp_gt) sig.append(n->get_gt(is_signed));
 | |
| 				if (cmp_eq) sig.append(n->get_eq());
 | |
| 				if (cmp_ne) sig.append(n->get_ne());
 | |
| 
 | |
| 				if (GetSize(sig) > 1)
 | |
| 					sig = module->ReduceOr(NEW_ID, sig);
 | |
| 
 | |
| 				sig.extend_u0(GetSize(cmp_y));
 | |
| 				module->connect(cmp_y, sig);
 | |
| 			}
 | |
| 
 | |
| 		delete_node:
 | |
| 			for (auto c : n->cells)
 | |
| 				module->remove(c);
 | |
| 			delete n;
 | |
| 		}
 | |
| 
 | |
| 		sig_alu.clear();
 | |
| 	}
 | |
| 
 | |
| 	void run()
 | |
| 	{
 | |
| 		log("Extracting $alu and $macc cells in module %s:\n", log_id(module));
 | |
| 
 | |
| 		count_bit_users();
 | |
| 		extract_macc();
 | |
| 		merge_macc();
 | |
| 		macc_to_alu();
 | |
| 		replace_macc();
 | |
| 		extract_cmp_alu();
 | |
| 		replace_alu();
 | |
| 
 | |
| 		log("  created %d $alu and %d $macc cells.\n", alu_counter, macc_counter);
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct AlumaccPass : public Pass {
 | |
| 	AlumaccPass() : Pass("alumacc", "extract ALU and MACC cells") { }
 | |
| 	void help() override
 | |
| 	{
 | |
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | |
| 		log("\n");
 | |
| 		log("    alumacc [selection]\n");
 | |
| 		log("\n");
 | |
| 		log("This pass translates arithmetic operations like $add, $mul, $lt, etc. to $alu\n");
 | |
| 		log("and $macc cells.\n");
 | |
| 		log("\n");
 | |
| 	}
 | |
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) override
 | |
| 	{
 | |
| 		log_header(design, "Executing ALUMACC pass (create $alu and $macc cells).\n");
 | |
| 
 | |
| 		size_t argidx;
 | |
| 		for (argidx = 1; argidx < args.size(); argidx++) {
 | |
| 			// if (args[argidx] == "-foobar") {
 | |
| 			// 	foobar_mode = true;
 | |
| 			// 	continue;
 | |
| 			// }
 | |
| 			break;
 | |
| 		}
 | |
| 		extra_args(args, argidx, design);
 | |
| 
 | |
| 		for (auto mod : design->selected_modules())
 | |
| 			if (!mod->has_processes_warn()) {
 | |
| 				AlumaccWorker worker(mod);
 | |
| 				worker.run();
 | |
| 			}
 | |
| 	}
 | |
| } AlumaccPass;
 | |
| 
 | |
| PRIVATE_NAMESPACE_END
 |