mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-31 11:42:30 +00:00 
			
		
		
		
	There are some leftovers, but this is an easy regex-based approach that removes most of them.
		
			
				
	
	
		
			669 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			669 lines
		
	
	
	
		
			18 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/register.h"
 | |
| #include "kernel/sigtools.h"
 | |
| #include "kernel/log.h"
 | |
| #include "kernel/celltypes.h"
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <set>
 | |
| 
 | |
| USING_YOSYS_NAMESPACE
 | |
| PRIVATE_NAMESPACE_BEGIN
 | |
| 
 | |
| struct OptReduceWorker
 | |
| {
 | |
| 	RTLIL::Design *design;
 | |
| 	RTLIL::Module *module;
 | |
| 	SigMap assign_map;
 | |
| 
 | |
| 	int total_count;
 | |
| 	bool did_something;
 | |
| 
 | |
| 	void opt_reduce(pool<RTLIL::Cell*> &cells, SigSet<RTLIL::Cell*> &drivers, RTLIL::Cell *cell)
 | |
| 	{
 | |
| 		if (cells.count(cell) == 0)
 | |
| 			return;
 | |
| 		cells.erase(cell);
 | |
| 
 | |
| 		RTLIL::SigSpec sig_a = assign_map(cell->getPort(ID::A));
 | |
| 		sig_a.sort_and_unify();
 | |
| 		pool<RTLIL::SigBit> new_sig_a_bits;
 | |
| 
 | |
| 		for (auto &bit : sig_a)
 | |
| 		{
 | |
| 			if (bit == RTLIL::State::S0) {
 | |
| 				if (cell->type == ID($reduce_and)) {
 | |
| 					new_sig_a_bits.clear();
 | |
| 					new_sig_a_bits.insert(RTLIL::State::S0);
 | |
| 					break;
 | |
| 				}
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (bit == RTLIL::State::S1) {
 | |
| 				if (cell->type == ID($reduce_or)) {
 | |
| 					new_sig_a_bits.clear();
 | |
| 					new_sig_a_bits.insert(RTLIL::State::S1);
 | |
| 					break;
 | |
| 				}
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (bit.wire == NULL) {
 | |
| 				new_sig_a_bits.insert(bit);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			bool imported_children = false;
 | |
| 			for (auto child_cell : drivers.find(bit)) {
 | |
| 				if (child_cell->type == cell->type) {
 | |
| 					opt_reduce(cells, drivers, child_cell);
 | |
| 					if (child_cell->getPort(ID::Y)[0] == bit) {
 | |
| 						pool<RTLIL::SigBit> child_sig_a_bits = assign_map(child_cell->getPort(ID::A)).to_sigbit_pool();
 | |
| 						new_sig_a_bits.insert(child_sig_a_bits.begin(), child_sig_a_bits.end());
 | |
| 					} else
 | |
| 						new_sig_a_bits.insert(RTLIL::State::S0);
 | |
| 					imported_children = true;
 | |
| 				}
 | |
| 			}
 | |
| 			if (!imported_children)
 | |
| 				new_sig_a_bits.insert(bit);
 | |
| 		}
 | |
| 
 | |
| 		RTLIL::SigSpec new_sig_a(new_sig_a_bits);
 | |
| 		new_sig_a.sort_and_unify();
 | |
| 
 | |
| 		if (GetSize(new_sig_a) == 0)
 | |
| 			new_sig_a = (cell->type == ID($reduce_or)) ? State::S0 : State::S1;
 | |
| 
 | |
| 		if (new_sig_a != sig_a || sig_a.size() != cell->getPort(ID::A).size()) {
 | |
| 			log("    New input vector for %s cell %s: %s\n", cell->type, cell->name, log_signal(new_sig_a));
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 		}
 | |
| 
 | |
| 		cell->setPort(ID::A, new_sig_a);
 | |
| 		cell->parameters[ID::A_WIDTH] = RTLIL::Const(new_sig_a.size());
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	void opt_pmux(RTLIL::Cell *cell)
 | |
| 	{
 | |
| 		RTLIL::SigSpec sig_a = assign_map(cell->getPort(ID::A));
 | |
| 		RTLIL::SigSpec sig_b = assign_map(cell->getPort(ID::B));
 | |
| 		RTLIL::SigSpec sig_s = assign_map(cell->getPort(ID::S));
 | |
| 
 | |
| 		RTLIL::SigSpec new_sig_b, new_sig_s;
 | |
| 		pool<RTLIL::SigSpec> handled_sig;
 | |
| 
 | |
| 		handled_sig.insert(sig_a);
 | |
| 		for (int i = 0; i < sig_s.size(); i++)
 | |
| 		{
 | |
| 			RTLIL::SigSpec this_b = sig_b.extract(i*sig_a.size(), sig_a.size());
 | |
| 			if (handled_sig.count(this_b) > 0)
 | |
| 				continue;
 | |
| 
 | |
| 			RTLIL::SigSpec this_s = sig_s.extract(i, 1);
 | |
| 			for (int j = i+1; j < sig_s.size(); j++) {
 | |
| 				RTLIL::SigSpec that_b = sig_b.extract(j*sig_a.size(), sig_a.size());
 | |
| 				if (this_b == that_b)
 | |
| 					this_s.append(sig_s.extract(j, 1));
 | |
| 			}
 | |
| 
 | |
| 			if (this_s.size() > 1)
 | |
| 			{
 | |
| 				RTLIL::Cell *reduce_or_cell = module->addCell(NEW_ID, ID($reduce_or));
 | |
| 				reduce_or_cell->setPort(ID::A, this_s);
 | |
| 				reduce_or_cell->parameters[ID::A_SIGNED] = RTLIL::Const(0);
 | |
| 				reduce_or_cell->parameters[ID::A_WIDTH] = RTLIL::Const(this_s.size());
 | |
| 				reduce_or_cell->parameters[ID::Y_WIDTH] = RTLIL::Const(1);
 | |
| 
 | |
| 				RTLIL::Wire *reduce_or_wire = module->addWire(NEW_ID);
 | |
| 				this_s = RTLIL::SigSpec(reduce_or_wire);
 | |
| 				reduce_or_cell->setPort(ID::Y, this_s);
 | |
| 			}
 | |
| 
 | |
| 			new_sig_b.append(this_b);
 | |
| 			new_sig_s.append(this_s);
 | |
| 			handled_sig.insert(this_b);
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() == 0)
 | |
| 		{
 | |
| 			module->connect(cell->getPort(ID::Y), cell->getPort(ID::A));
 | |
| 			assign_map.add(cell->getPort(ID::Y), cell->getPort(ID::A));
 | |
| 			module->remove(cell);
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() != sig_s.size() || (new_sig_s.size() == 1 && cell->type == ID($pmux))) {
 | |
| 			log("    New ctrl vector for %s cell %s: %s\n", cell->type, cell->name, log_signal(new_sig_s));
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 			cell->setPort(ID::B, new_sig_b);
 | |
| 			cell->setPort(ID::S, new_sig_s);
 | |
| 			if (new_sig_s.size() > 1) {
 | |
| 				cell->parameters[ID::S_WIDTH] = RTLIL::Const(new_sig_s.size());
 | |
| 			} else {
 | |
| 				cell->type = ID($mux);
 | |
| 				cell->parameters.erase(ID::S_WIDTH);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void opt_bmux(RTLIL::Cell *cell)
 | |
| 	{
 | |
| 		RTLIL::SigSpec sig_a = assign_map(cell->getPort(ID::A));
 | |
| 		RTLIL::SigSpec sig_s = assign_map(cell->getPort(ID::S));
 | |
| 		int width = cell->getParam(ID::WIDTH).as_int();
 | |
| 
 | |
| 		RTLIL::SigSpec new_sig_a, new_sig_s;
 | |
| 		dict<RTLIL::SigBit, int> handled_bits;
 | |
| 
 | |
| 		// 0 and up: index of new_sig_s bit
 | |
| 		// -1: const 0
 | |
| 		// -2: const 1
 | |
| 		std::vector<int> swizzle;
 | |
| 
 | |
| 		for (int i = 0; i < sig_s.size(); i++)
 | |
| 		{
 | |
| 			SigBit bit = sig_s[i];
 | |
| 			if (bit == State::S0) {
 | |
| 				swizzle.push_back(-1);
 | |
| 			} else if (bit == State::S1) {
 | |
| 				swizzle.push_back(-2);
 | |
| 			} else {
 | |
| 				auto it = handled_bits.find(bit);
 | |
| 				if (it == handled_bits.end()) {
 | |
| 					int new_idx = GetSize(new_sig_s);
 | |
| 					new_sig_s.append(bit);
 | |
| 					handled_bits[bit] = new_idx;
 | |
| 					swizzle.push_back(new_idx);
 | |
| 				} else {
 | |
| 					swizzle.push_back(it->second);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		for (int i = 0; i < (1 << GetSize(new_sig_s)); i++) {
 | |
| 			int idx = 0;
 | |
| 			for (int j = 0; j < GetSize(sig_s); j++) {
 | |
| 				if (swizzle[j] == -1) {
 | |
| 					// const 0.
 | |
| 				} else if (swizzle[j] == -2) {
 | |
| 					// const 1.
 | |
| 					idx |= 1 << j;
 | |
| 				} else {
 | |
| 					if (i & 1 << swizzle[j])
 | |
| 						idx |= 1 << j;
 | |
| 				}
 | |
| 			}
 | |
| 			new_sig_a.append(sig_a.extract(idx * width, width));
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() == 0)
 | |
| 		{
 | |
| 			module->connect(cell->getPort(ID::Y), new_sig_a);
 | |
| 			assign_map.add(cell->getPort(ID::Y), new_sig_a);
 | |
| 			module->remove(cell);
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() == 1)
 | |
| 		{
 | |
| 			cell->type = ID($mux);
 | |
| 			cell->setPort(ID::A, new_sig_a.extract(0, width));
 | |
| 			cell->setPort(ID::B, new_sig_a.extract(width, width));
 | |
| 			cell->setPort(ID::S, new_sig_s);
 | |
| 			cell->parameters.erase(ID::S_WIDTH);
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() != sig_s.size()) {
 | |
| 			log("    New ctrl vector for %s cell %s: %s\n", cell->type, cell->name, log_signal(new_sig_s));
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 			cell->setPort(ID::A, new_sig_a);
 | |
| 			cell->setPort(ID::S, new_sig_s);
 | |
| 			cell->parameters[ID::S_WIDTH] = RTLIL::Const(new_sig_s.size());
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void opt_demux(RTLIL::Cell *cell)
 | |
| 	{
 | |
| 		RTLIL::SigSpec sig_y = assign_map(cell->getPort(ID::Y));
 | |
| 		RTLIL::SigSpec sig_s = assign_map(cell->getPort(ID::S));
 | |
| 		int width = cell->getParam(ID::WIDTH).as_int();
 | |
| 
 | |
| 		RTLIL::SigSpec new_sig_y, new_sig_s;
 | |
| 		dict<RTLIL::SigBit, int> handled_bits;
 | |
| 
 | |
| 		// 0 and up: index of new_sig_s bit
 | |
| 		// -1: const 0
 | |
| 		// -2: const 1
 | |
| 		std::vector<int> swizzle;
 | |
| 
 | |
| 		for (int i = 0; i < sig_s.size(); i++)
 | |
| 		{
 | |
| 			SigBit bit = sig_s[i];
 | |
| 			if (bit == State::S0) {
 | |
| 				swizzle.push_back(-1);
 | |
| 			} else if (bit == State::S1) {
 | |
| 				swizzle.push_back(-2);
 | |
| 			} else {
 | |
| 				auto it = handled_bits.find(bit);
 | |
| 				if (it == handled_bits.end()) {
 | |
| 					int new_idx = GetSize(new_sig_s);
 | |
| 					new_sig_s.append(bit);
 | |
| 					handled_bits[bit] = new_idx;
 | |
| 					swizzle.push_back(new_idx);
 | |
| 				} else {
 | |
| 					swizzle.push_back(it->second);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		pool<int> nonzero_idx;
 | |
| 
 | |
| 		for (int i = 0; i < (1 << GetSize(new_sig_s)); i++) {
 | |
| 			int idx = 0;
 | |
| 			for (int j = 0; j < GetSize(sig_s); j++) {
 | |
| 				if (swizzle[j] == -1) {
 | |
| 					// const 0.
 | |
| 				} else if (swizzle[j] == -2) {
 | |
| 					// const 1.
 | |
| 					idx |= 1 << j;
 | |
| 				} else {
 | |
| 					if (i & 1 << swizzle[j])
 | |
| 						idx |= 1 << j;
 | |
| 				}
 | |
| 			}
 | |
| 			log_assert(!nonzero_idx.count(idx));
 | |
| 			nonzero_idx.insert(idx);
 | |
| 			new_sig_y.append(sig_y.extract(idx * width, width));
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() == sig_s.size() && sig_s.size() > 0)
 | |
| 			return;
 | |
| 
 | |
| 		log("    New ctrl vector for %s cell %s: %s\n", cell->type, cell->name, log_signal(new_sig_s));
 | |
| 		did_something = true;
 | |
| 		total_count++;
 | |
| 
 | |
| 		for (int i = 0; i < (1 << GetSize(sig_s)); i++) {
 | |
| 			if (!nonzero_idx.count(i)) {
 | |
| 				SigSpec slice = sig_y.extract(i * width, width);
 | |
| 				module->connect(slice, Const(State::S0, width));
 | |
| 				assign_map.add(slice, Const(State::S0, width));
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (new_sig_s.size() == 0)
 | |
| 		{
 | |
| 			module->connect(new_sig_y, cell->getPort(ID::A));
 | |
| 			assign_map.add(new_sig_y, cell->getPort(ID::A));
 | |
| 			module->remove(cell);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			cell->setPort(ID::S, new_sig_s);
 | |
| 			cell->setPort(ID::Y, new_sig_y);
 | |
| 			cell->parameters[ID::S_WIDTH] = RTLIL::Const(new_sig_s.size());
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	bool opt_mux_bits(RTLIL::Cell *cell)
 | |
| 	{
 | |
| 		SigSpec sig_a = assign_map(cell->getPort(ID::A));
 | |
| 		SigSpec sig_b;
 | |
| 		SigSpec sig_y = assign_map(cell->getPort(ID::Y));
 | |
| 		int width = GetSize(sig_y);
 | |
| 
 | |
| 		if (cell->type != ID($bmux))
 | |
| 			sig_b = assign_map(cell->getPort(ID::B));
 | |
| 
 | |
| 		RTLIL::SigSig old_sig_conn;
 | |
| 
 | |
| 		dict<SigSpec, SigBit> consolidated_in_tuples;
 | |
| 		std::vector<int> swizzle;
 | |
| 
 | |
| 		for (int i = 0; i < width; i++)
 | |
| 		{
 | |
| 			SigSpec in_tuple;
 | |
| 			bool all_tuple_bits_same = true;
 | |
| 
 | |
| 			in_tuple.append(sig_a[i]);
 | |
| 			for (int j = i; j < GetSize(sig_a); j += width) {
 | |
| 				in_tuple.append(sig_a[j]);
 | |
| 				if (sig_a[j] != in_tuple[0])
 | |
| 					all_tuple_bits_same = false;
 | |
| 			}
 | |
| 			for (int j = i; j < GetSize(sig_b); j += width) {
 | |
| 				in_tuple.append(sig_b[j]);
 | |
| 				if (sig_b[j] != in_tuple[0])
 | |
| 					all_tuple_bits_same = false;
 | |
| 			}
 | |
| 
 | |
| 			if (all_tuple_bits_same)
 | |
| 			{
 | |
| 				old_sig_conn.first.append(sig_y[i]);
 | |
| 				old_sig_conn.second.append(sig_a[i]);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			auto it = consolidated_in_tuples.find(in_tuple);
 | |
| 			if (it == consolidated_in_tuples.end())
 | |
| 			{
 | |
| 				consolidated_in_tuples[in_tuple] = sig_y[i];
 | |
| 				swizzle.push_back(i);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				old_sig_conn.first.append(sig_y[i]);
 | |
| 				old_sig_conn.second.append(it->second);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (GetSize(swizzle) != width)
 | |
| 		{
 | |
| 			log("    Consolidated identical input bits for %s cell %s:\n", cell->type, cell->name);
 | |
| 			if (cell->type != ID($bmux)) {
 | |
| 				log("      Old ports: A=%s, B=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 						log_signal(cell->getPort(ID::B)), log_signal(cell->getPort(ID::Y)));
 | |
| 			} else {
 | |
| 				log("      Old ports: A=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 						log_signal(cell->getPort(ID::Y)));
 | |
| 			}
 | |
| 
 | |
| 			if (swizzle.empty()) {
 | |
| 				module->remove(cell);
 | |
| 			} else {
 | |
| 				SigSpec new_sig_a;
 | |
| 				for (int i = 0; i < GetSize(sig_a); i += width)
 | |
| 					for (int j: swizzle)
 | |
| 						new_sig_a.append(sig_a[i+j]);
 | |
| 				cell->setPort(ID::A, new_sig_a);
 | |
| 
 | |
| 				if (cell->type != ID($bmux)) {
 | |
| 					SigSpec new_sig_b;
 | |
| 					for (int i = 0; i < GetSize(sig_b); i += width)
 | |
| 						for (int j: swizzle)
 | |
| 							new_sig_b.append(sig_b[i+j]);
 | |
| 					cell->setPort(ID::B, new_sig_b);
 | |
| 				}
 | |
| 
 | |
| 				SigSpec new_sig_y;
 | |
| 				for (int j: swizzle)
 | |
| 					new_sig_y.append(sig_y[j]);
 | |
| 				cell->setPort(ID::Y, new_sig_y);
 | |
| 
 | |
| 				cell->parameters[ID::WIDTH] = RTLIL::Const(GetSize(swizzle));
 | |
| 
 | |
| 				if (cell->type != ID($bmux)) {
 | |
| 					log("      New ports: A=%s, B=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 							log_signal(cell->getPort(ID::B)), log_signal(cell->getPort(ID::Y)));
 | |
| 				} else {
 | |
| 					log("      New ports: A=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 							log_signal(cell->getPort(ID::Y)));
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			log("      New connections: %s = %s\n", log_signal(old_sig_conn.first), log_signal(old_sig_conn.second));
 | |
| 			module->connect(old_sig_conn);
 | |
| 
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 		}
 | |
| 		return swizzle.empty();
 | |
| 	}
 | |
| 
 | |
| 	bool opt_demux_bits(RTLIL::Cell *cell) {
 | |
| 		SigSpec sig_a = assign_map(cell->getPort(ID::A));
 | |
| 		SigSpec sig_y = assign_map(cell->getPort(ID::Y));
 | |
| 		int width = GetSize(sig_a);
 | |
| 
 | |
| 		RTLIL::SigSig old_sig_conn;
 | |
| 
 | |
| 		dict<SigBit, int> handled_bits;
 | |
| 		std::vector<int> swizzle;
 | |
| 
 | |
| 		for (int i = 0; i < width; i++)
 | |
| 		{
 | |
| 			if (sig_a[i] == State::S0)
 | |
| 			{
 | |
| 				for (int j = i; j < GetSize(sig_y); j += width)
 | |
| 				{
 | |
| 					old_sig_conn.first.append(sig_y[j]);
 | |
| 					old_sig_conn.second.append(State::S0);
 | |
| 				}
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			auto it = handled_bits.find(sig_a[i]);
 | |
| 			if (it == handled_bits.end())
 | |
| 			{
 | |
| 				handled_bits[sig_a[i]] = i;
 | |
| 				swizzle.push_back(i);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				for (int j = 0; j < GetSize(sig_y); j += width)
 | |
| 				{
 | |
| 					old_sig_conn.first.append(sig_y[i+j]);
 | |
| 					old_sig_conn.second.append(sig_y[it->second+j]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (GetSize(swizzle) != width)
 | |
| 		{
 | |
| 			log("    Consolidated identical input bits for %s cell %s:\n", cell->type, cell->name);
 | |
| 			log("      Old ports: A=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 					log_signal(cell->getPort(ID::Y)));
 | |
| 
 | |
| 			if (swizzle.empty()) {
 | |
| 				module->remove(cell);
 | |
| 			} else {
 | |
| 				SigSpec new_sig_a;
 | |
| 				for (int j: swizzle)
 | |
| 					new_sig_a.append(sig_a[j]);
 | |
| 				cell->setPort(ID::A, new_sig_a);
 | |
| 
 | |
| 				SigSpec new_sig_y;
 | |
| 				for (int i = 0; i < GetSize(sig_y); i += width)
 | |
| 					for (int j: swizzle)
 | |
| 						new_sig_y.append(sig_y[i+j]);
 | |
| 				cell->setPort(ID::Y, new_sig_y);
 | |
| 
 | |
| 				cell->parameters[ID::WIDTH] = RTLIL::Const(GetSize(swizzle));
 | |
| 
 | |
| 				log("      New ports: A=%s, Y=%s\n", log_signal(cell->getPort(ID::A)),
 | |
| 						log_signal(cell->getPort(ID::Y)));
 | |
| 			}
 | |
| 
 | |
| 			log("      New connections: %s = %s\n", log_signal(old_sig_conn.first), log_signal(old_sig_conn.second));
 | |
| 			module->connect(old_sig_conn);
 | |
| 
 | |
| 			did_something = true;
 | |
| 			total_count++;
 | |
| 		}
 | |
| 		return swizzle.empty();
 | |
| 	}
 | |
| 
 | |
| 	OptReduceWorker(RTLIL::Design *design, RTLIL::Module *module, bool do_fine) :
 | |
| 			design(design), module(module), assign_map(module)
 | |
| 	{
 | |
| 		log("  Optimizing cells in module %s.\n", module->name);
 | |
| 
 | |
| 		total_count = 0;
 | |
| 		did_something = true;
 | |
| 
 | |
| 		SigPool mem_wren_sigs;
 | |
| 		for (auto &cell_it : module->cells_) {
 | |
| 			RTLIL::Cell *cell = cell_it.second;
 | |
| 			if (cell->type.in(ID($mem), ID($mem_v2)))
 | |
| 				mem_wren_sigs.add(assign_map(cell->getPort(ID::WR_EN)));
 | |
| 			if (cell->type.in(ID($memwr), ID($memwr_v2)))
 | |
| 				mem_wren_sigs.add(assign_map(cell->getPort(ID::EN)));
 | |
| 		}
 | |
| 		for (auto &cell_it : module->cells_) {
 | |
| 			RTLIL::Cell *cell = cell_it.second;
 | |
| 			if (cell->type == ID($dff) && mem_wren_sigs.check_any(assign_map(cell->getPort(ID::Q))))
 | |
| 				mem_wren_sigs.add(assign_map(cell->getPort(ID::D)));
 | |
| 		}
 | |
| 
 | |
| 		bool keep_expanding_mem_wren_sigs = true;
 | |
| 		while (keep_expanding_mem_wren_sigs) {
 | |
| 			keep_expanding_mem_wren_sigs = false;
 | |
| 			for (auto &cell_it : module->cells_) {
 | |
| 				RTLIL::Cell *cell = cell_it.second;
 | |
| 				if (cell->type == ID($mux) && mem_wren_sigs.check_any(assign_map(cell->getPort(ID::Y)))) {
 | |
| 					if (!mem_wren_sigs.check_all(assign_map(cell->getPort(ID::A))) ||
 | |
| 							!mem_wren_sigs.check_all(assign_map(cell->getPort(ID::B))))
 | |
| 						keep_expanding_mem_wren_sigs = true;
 | |
| 					mem_wren_sigs.add(assign_map(cell->getPort(ID::A)));
 | |
| 					mem_wren_sigs.add(assign_map(cell->getPort(ID::B)));
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		while (did_something)
 | |
| 		{
 | |
| 			did_something = false;
 | |
| 
 | |
| 			// merge trees of reduce_* cells to one single cell and unify input vectors
 | |
| 			// (only handle reduce_and and reduce_or for various reasons)
 | |
| 
 | |
| 			const IdString type_list[] = { ID($reduce_or), ID($reduce_and) };
 | |
| 			for (auto type : type_list)
 | |
| 			{
 | |
| 				SigSet<RTLIL::Cell*> drivers;
 | |
| 				pool<RTLIL::Cell*> cells;
 | |
| 
 | |
| 				for (auto &cell_it : module->cells_) {
 | |
| 					RTLIL::Cell *cell = cell_it.second;
 | |
| 					if (cell->type != type || !design->selected(module, cell))
 | |
| 						continue;
 | |
| 					drivers.insert(assign_map(cell->getPort(ID::Y)), cell);
 | |
| 					cells.insert(cell);
 | |
| 				}
 | |
| 
 | |
| 				while (cells.size() > 0) {
 | |
| 					RTLIL::Cell *cell = *cells.begin();
 | |
| 					opt_reduce(cells, drivers, cell);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// merge identical inputs on $mux and $pmux cells
 | |
| 
 | |
| 			for (auto cell : module->selected_cells())
 | |
| 			{
 | |
| 				if (!cell->type.in(ID($mux), ID($pmux), ID($bmux), ID($demux)))
 | |
| 					continue;
 | |
| 
 | |
| 				// this optimization is to aggressive for most coarse-grain applications.
 | |
| 				// but we always want it for multiplexers driving write enable ports.
 | |
| 				if (do_fine || mem_wren_sigs.check_any(assign_map(cell->getPort(ID::Y)))) {
 | |
| 					if (cell->type == ID($demux)) {
 | |
| 						if (opt_demux_bits(cell))
 | |
| 							continue;
 | |
| 					} else {
 | |
| 						if (opt_mux_bits(cell))
 | |
| 							continue;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				if (cell->type.in(ID($mux), ID($pmux)))
 | |
| 					opt_pmux(cell);
 | |
| 				else if (cell->type == ID($bmux))
 | |
| 					opt_bmux(cell);
 | |
| 				else if (cell->type == ID($demux))
 | |
| 					opt_demux(cell);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		module->check();
 | |
| 	}
 | |
| };
 | |
| 
 | |
| struct OptReducePass : public Pass {
 | |
| 	OptReducePass() : Pass("opt_reduce", "simplify large MUXes and AND/OR gates") { }
 | |
| 	void help() override
 | |
| 	{
 | |
| 		//   |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
 | |
| 		log("\n");
 | |
| 		log("    opt_reduce [options] [selection]\n");
 | |
| 		log("\n");
 | |
| 		log("This pass performs two interlinked optimizations:\n");
 | |
| 		log("\n");
 | |
| 		log("1. it consolidates trees of large AND gates or OR gates and eliminates\n");
 | |
| 		log("duplicated inputs.\n");
 | |
| 		log("\n");
 | |
| 		log("2. it identifies duplicated inputs to MUXes and replaces them with a single\n");
 | |
| 		log("input with the original control signals OR'ed together.\n");
 | |
| 		log("\n");
 | |
| 		log("    -fine\n");
 | |
| 		log("      perform fine-grain optimizations\n");
 | |
| 		log("\n");
 | |
| 		log("    -full\n");
 | |
| 		log("      alias for -fine\n");
 | |
| 		log("\n");
 | |
| 	}
 | |
| 	void execute(std::vector<std::string> args, RTLIL::Design *design) override
 | |
| 	{
 | |
| 		bool do_fine = false;
 | |
| 
 | |
| 		log_header(design, "Executing OPT_REDUCE pass (consolidate $*mux and $reduce_* inputs).\n");
 | |
| 
 | |
| 		size_t argidx;
 | |
| 		for (argidx = 1; argidx < args.size(); argidx++) {
 | |
| 			if (args[argidx] == "-fine") {
 | |
| 				do_fine = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (args[argidx] == "-full") {
 | |
| 				do_fine = true;
 | |
| 				continue;
 | |
| 			}
 | |
| 			break;
 | |
| 		}
 | |
| 		extra_args(args, argidx, design);
 | |
| 
 | |
| 		int total_count = 0;
 | |
| 		for (auto module : design->selected_modules())
 | |
| 			while (1) {
 | |
| 				OptReduceWorker worker(design, module, do_fine);
 | |
| 				total_count += worker.total_count;
 | |
| 				if (worker.total_count == 0)
 | |
| 					break;
 | |
| 			}
 | |
| 
 | |
| 		if (total_count)
 | |
| 			design->scratchpad_set_bool("opt.did_something", true);
 | |
| 		log("Performed a total of %d changes.\n", total_count);
 | |
| 	}
 | |
| } OptReducePass;
 | |
| 
 | |
| PRIVATE_NAMESPACE_END
 |