mirror of
				https://github.com/YosysHQ/yosys
				synced 2025-10-25 00:44:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			471 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			471 lines
		
	
	
	
		
			10 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.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #ifndef SIGTOOLS_H
 | |
| #define SIGTOOLS_H
 | |
| 
 | |
| #include "kernel/yosys.h"
 | |
| 
 | |
| YOSYS_NAMESPACE_BEGIN
 | |
| 
 | |
| struct SigPool
 | |
| {
 | |
| 	struct bitDef_t : public std::pair<RTLIL::Wire*, int> {
 | |
| 		bitDef_t() : std::pair<RTLIL::Wire*, int>(NULL, 0) { }
 | |
| 		bitDef_t(const RTLIL::SigBit &bit) : std::pair<RTLIL::Wire*, int>(bit.wire, bit.offset) { }
 | |
| 		[[nodiscard]] Hasher hash_into(Hasher h) const {
 | |
| 			h.eat(first->name);
 | |
| 			h.eat(second);
 | |
| 			return h;
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	pool<bitDef_t> bits;
 | |
| 
 | |
| 	void clear()
 | |
| 	{
 | |
| 		bits.clear();
 | |
| 	}
 | |
| 
 | |
| 	void add(const RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits.insert(bit);
 | |
| 	}
 | |
| 
 | |
| 	void add(const SigPool &other)
 | |
| 	{
 | |
| 		for (auto &bit : other.bits)
 | |
| 			bits.insert(bit);
 | |
| 	}
 | |
| 
 | |
| 	void del(const RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits.erase(bit);
 | |
| 	}
 | |
| 
 | |
| 	void del(const SigPool &other)
 | |
| 	{
 | |
| 		for (auto &bit : other.bits)
 | |
| 			bits.erase(bit);
 | |
| 	}
 | |
| 
 | |
| 	void expand(const RTLIL::SigSpec &from, const RTLIL::SigSpec &to)
 | |
| 	{
 | |
| 		log_assert(GetSize(from) == GetSize(to));
 | |
| 		for (int i = 0; i < GetSize(from); i++) {
 | |
| 			bitDef_t bit_from(from[i]), bit_to(to[i]);
 | |
| 			if (bit_from.first != NULL && bit_to.first != NULL && bits.count(bit_from) > 0)
 | |
| 				bits.insert(bit_to);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec extract(const RTLIL::SigSpec &sig) const
 | |
| 	{
 | |
| 		RTLIL::SigSpec result;
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL && bits.count(bit))
 | |
| 				result.append(bit);
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec remove(const RTLIL::SigSpec &sig) const
 | |
| 	{
 | |
| 		RTLIL::SigSpec result;
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL && bits.count(bit) == 0)
 | |
| 				result.append(bit);
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	bool check(const RTLIL::SigBit &bit) const
 | |
| 	{
 | |
| 		return bit.wire != NULL && bits.count(bit);
 | |
| 	}
 | |
| 
 | |
| 	bool check_any(const RTLIL::SigSpec &sig) const
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL && bits.count(bit))
 | |
| 				return true;
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	bool check_all(const RTLIL::SigSpec &sig) const
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL && bits.count(bit) == 0)
 | |
| 				return false;
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec export_one() const
 | |
| 	{
 | |
| 		for (auto &bit : bits)
 | |
| 			return RTLIL::SigSpec(bit.first, bit.second);
 | |
| 		return RTLIL::SigSpec();
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec export_all() const
 | |
| 	{
 | |
| 		pool<RTLIL::SigBit> sig;
 | |
| 		for (auto &bit : bits)
 | |
| 			sig.insert(RTLIL::SigBit(bit.first, bit.second));
 | |
| 		return sig;
 | |
| 	}
 | |
| 
 | |
| 	size_t size() const
 | |
| 	{
 | |
| 		return bits.size();
 | |
| 	}
 | |
| };
 | |
| 
 | |
| template <typename T, class Compare = void>
 | |
| struct SigSet
 | |
| {
 | |
| 	static_assert(!std::is_same<Compare,void>::value, "Default value for `Compare' class not found for SigSet<T>. Please specify.");
 | |
| 
 | |
| 	struct bitDef_t : public std::pair<RTLIL::Wire*, int> {
 | |
| 		bitDef_t() : std::pair<RTLIL::Wire*, int>(NULL, 0) { }
 | |
| 		bitDef_t(const RTLIL::SigBit &bit) : std::pair<RTLIL::Wire*, int>(bit.wire, bit.offset) { }
 | |
| 		[[nodiscard]] Hasher hash_into(Hasher h) const {
 | |
| 			h.eat(first->name);
 | |
| 			h.eat(second);
 | |
| 			return h;
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	dict<bitDef_t, std::set<T, Compare>> bits;
 | |
| 
 | |
| 	void clear()
 | |
| 	{
 | |
| 		bits.clear();
 | |
| 	}
 | |
| 
 | |
| 	void insert(const RTLIL::SigSpec &sig, T data)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits[bit].insert(data);
 | |
| 	}
 | |
| 
 | |
| 	void insert(const RTLIL::SigSpec& sig, const std::set<T> &data)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits[bit].insert(data.begin(), data.end());
 | |
| 	}
 | |
| 
 | |
| 	void erase(const RTLIL::SigSpec& sig)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits[bit].clear();
 | |
| 	}
 | |
| 
 | |
| 	void erase(const RTLIL::SigSpec &sig, T data)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits[bit].erase(data);
 | |
| 	}
 | |
| 
 | |
| 	void erase(const RTLIL::SigSpec &sig, const std::set<T> &data)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL)
 | |
| 				bits[bit].erase(data.begin(), data.end());
 | |
| 	}
 | |
| 
 | |
| 	void find(const RTLIL::SigSpec &sig, std::set<T> &result)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL) {
 | |
| 				auto &data = bits[bit];
 | |
| 				result.insert(data.begin(), data.end());
 | |
| 			}
 | |
| 	}
 | |
| 
 | |
| 	void find(const RTLIL::SigSpec &sig, pool<T> &result)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			if (bit.wire != NULL) {
 | |
| 				auto &data = bits[bit];
 | |
| 				result.insert(data.begin(), data.end());
 | |
| 			}
 | |
| 	}
 | |
| 
 | |
| 	std::set<T> find(const RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		std::set<T> result;
 | |
| 		find(sig, result);
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	bool has(const RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			if (bit.wire != NULL && bits.count(bit))
 | |
| 				return true;
 | |
| 		return false;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| template<typename T>
 | |
| class SigSet<T, typename std::enable_if<!std::is_pointer<T>::value>::type> : public SigSet<T, std::less<T>> {};
 | |
| template<typename T>
 | |
| using sort_by_name_id_guard = typename std::enable_if<std::is_same<T,RTLIL::Cell*>::value>::type;
 | |
| template<typename T>
 | |
| class SigSet<T, sort_by_name_id_guard<T>> : public SigSet<T, RTLIL::sort_by_name_id<typename std::remove_pointer<T>::type>> {};
 | |
| 
 | |
| struct SigMapView
 | |
| {
 | |
| 	mfp<SigBit> database;
 | |
| 
 | |
| 	// Modify bit to its representative
 | |
| 	void apply(RTLIL::SigBit &bit) const
 | |
| 	{
 | |
| 		bit = database.find(bit);
 | |
| 	}
 | |
| 
 | |
| 	void apply(RTLIL::SigSpec &sig) const
 | |
| 	{
 | |
| 		for (auto &bit : sig)
 | |
| 			apply(bit);
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigBit operator()(RTLIL::SigBit bit) const
 | |
| 	{
 | |
| 		apply(bit);
 | |
| 		return bit;
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec operator()(RTLIL::SigSpec sig) const
 | |
| 	{
 | |
| 		apply(sig);
 | |
| 		return sig;
 | |
| 	}
 | |
| 
 | |
| 	RTLIL::SigSpec operator()(RTLIL::Wire *wire) const
 | |
| 	{
 | |
| 		SigSpec sig(wire);
 | |
| 		apply(sig);
 | |
| 		return sig;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * SigMap wraps a union-find "database"
 | |
|  * to map SigBits of a module to canonical representative SigBits.
 | |
|  * SigBits that are connected share a set in the underlying database.
 | |
|  * If a SigBit has a const state (impl: bit.wire is nullptr),
 | |
|  * it's promoted to a representative.
 | |
|  */
 | |
| struct SigMap final : public SigMapView
 | |
| {
 | |
| 	SigMap(RTLIL::Module *module = NULL)
 | |
| 	{
 | |
| 		if (module != NULL)
 | |
| 			set(module);
 | |
| 	}
 | |
| 
 | |
| 	void swap(SigMap &other)
 | |
| 	{
 | |
| 		database.swap(other.database);
 | |
| 	}
 | |
| 
 | |
| 	void clear()
 | |
| 	{
 | |
| 		database.clear();
 | |
| 	}
 | |
| 
 | |
| 	// Rebuild SigMap for all connections in module
 | |
| 	void set(RTLIL::Module *module)
 | |
| 	{
 | |
| 		int bitcount = 0;
 | |
| 		for (auto &it : module->connections())
 | |
| 			bitcount += it.first.size();
 | |
| 
 | |
| 		database.clear();
 | |
| 		database.reserve(bitcount);
 | |
| 
 | |
| 		for (auto &it : module->connections())
 | |
| 			add(it.first, it.second);
 | |
| 	}
 | |
| 
 | |
| 	// Add connections from "from" to "to", bit-by-bit
 | |
| 	void add(const RTLIL::SigSpec& from, const RTLIL::SigSpec& to)
 | |
| 	{
 | |
| 		log_assert(GetSize(from) == GetSize(to));
 | |
| 
 | |
| 		for (int i = 0; i < GetSize(from); i++)
 | |
| 		{
 | |
| 			int bfi = database.lookup(from[i]);
 | |
| 			int bti = database.lookup(to[i]);
 | |
| 
 | |
| 			const RTLIL::SigBit &bf = database[bfi];
 | |
| 			const RTLIL::SigBit &bt = database[bti];
 | |
| 
 | |
| 			if (bf.wire || bt.wire)
 | |
| 			{
 | |
| 				database.imerge(bfi, bti);
 | |
| 
 | |
| 				if (bf.wire == nullptr)
 | |
| 					database.ipromote(bfi);
 | |
| 
 | |
| 				if (bt.wire == nullptr)
 | |
| 					database.ipromote(bti);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Add sig as disconnected from anything
 | |
| 	void add(const RTLIL::SigBit &bit)
 | |
| 	{
 | |
| 		const auto &b = database.find(bit);
 | |
| 		if (b.wire != nullptr)
 | |
| 			database.promote(bit);
 | |
| 	}
 | |
| 
 | |
| 	void add(const RTLIL::SigSpec &sig)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			add(bit);
 | |
| 	}
 | |
| 
 | |
| 	inline void add(Wire *wire) { return add(RTLIL::SigSpec(wire)); }
 | |
| 
 | |
| 	// All non-const bits
 | |
| 	RTLIL::SigSpec allbits() const
 | |
| 	{
 | |
| 		RTLIL::SigSpec sig;
 | |
| 		for (const auto &bit : database)
 | |
| 			if (bit.wire != nullptr)
 | |
| 				sig.append(bit);
 | |
| 		return sig;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * SiValgMap wraps a union-find "database" to map SigBits of a module to
 | |
|  * canonical representative SigBits plus some optional Val value associated with the bits.
 | |
|  * Val has a commutative, associative, idempotent operator|=, a default constructor
 | |
|  * which constructs an identity element, and a copy constructor.
 | |
|  * SigBits that are connected share a set in the underlying database;
 | |
|  * the associated value is the "sum" of all the values associated with the contributing bits.
 | |
|  * If any of the SigBits in a set are a constant, the canonical SigBit is a constant.
 | |
|  */
 | |
| template <class Val>
 | |
| struct SigValMap final : public SigMapView
 | |
| {
 | |
| 	dict<SigBit, Val> values;
 | |
| 
 | |
| 	void swap(SigValMap<Val> &other)
 | |
| 	{
 | |
| 		database.swap(other.database);
 | |
| 		values.swap(other.values);
 | |
| 	}
 | |
| 
 | |
| 	void clear()
 | |
| 	{
 | |
| 		database.clear();
 | |
| 		values.clear();
 | |
| 	}
 | |
| 
 | |
| 	// Rebuild SigMap for all connections in module
 | |
| 	void set(RTLIL::Module *module)
 | |
| 	{
 | |
| 		int bitcount = 0;
 | |
| 		for (auto &it : module->connections())
 | |
| 			bitcount += it.first.size();
 | |
| 
 | |
| 		database.clear();
 | |
| 		values.clear();
 | |
| 		database.reserve(bitcount);
 | |
| 
 | |
| 		for (auto &it : module->connections())
 | |
| 			add(it.first, it.second);
 | |
| 	}
 | |
| 
 | |
| 	// Add connections from "from" to "to", bit-by-bit.
 | |
| 	void add(const RTLIL::SigSpec& from, const RTLIL::SigSpec& to)
 | |
| 	{
 | |
| 		log_assert(GetSize(from) == GetSize(to));
 | |
| 
 | |
| 		for (int i = 0; i < GetSize(from); i++)
 | |
| 		{
 | |
| 			int bfi = database.lookup(from[i]);
 | |
| 			int bti = database.lookup(to[i]);
 | |
| 			if (bfi == bti) {
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			const RTLIL::SigBit &bf = database[bfi];
 | |
| 			const RTLIL::SigBit &bt = database[bti];
 | |
| 			if (bf.wire == nullptr) {
 | |
| 				// bf is constant so make it the canonical representative.
 | |
| 				database.imerge(bti, bfi);
 | |
| 				merge_value(bt, bf);
 | |
| 			} else {
 | |
| 				// Make bt the canonical representative.
 | |
| 				database.imerge(bfi, bti);
 | |
| 				merge_value(bf, bt);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void addVal(const RTLIL::SigBit &bit, const Val &val)
 | |
| 	{
 | |
| 		values[database.find(bit)] |= val;
 | |
| 	}
 | |
| 
 | |
| 	void addVal(const RTLIL::SigSpec &sig, const Val &val)
 | |
| 	{
 | |
| 		for (const auto &bit : sig)
 | |
| 			addVal(bit, val);
 | |
| 	}
 | |
| 
 | |
| 	Val apply_and_get_value(RTLIL::SigBit &bit) const
 | |
| 	{
 | |
| 		bit = database.find(bit);
 | |
| 		auto it = values.find(bit);
 | |
| 		return it == values.end() ? Val() : it->second;
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	void merge_value(const RTLIL::SigBit &from, const RTLIL::SigBit &to)
 | |
| 	{
 | |
| 		auto it = values.find(from);
 | |
| 		if (it == values.end()) {
 | |
| 			return;
 | |
| 		}
 | |
| 		// values[to] could resize the underlying `entries` so
 | |
| 		// finish using `it` first.
 | |
| 		Val v = it->second;
 | |
| 		values.erase(it);
 | |
| 		values[to] |= v;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| YOSYS_NAMESPACE_END
 | |
| 
 | |
| #endif /* SIGTOOLS_H */
 |