summaryrefslogtreecommitdiff
path: root/kernel/sigtools.h
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/sigtools.h')
-rw-r--r--kernel/sigtools.h332
1 files changed, 332 insertions, 0 deletions
diff --git a/kernel/sigtools.h b/kernel/sigtools.h
new file mode 100644
index 00000000..4e97bb77
--- /dev/null
+++ b/kernel/sigtools.h
@@ -0,0 +1,332 @@
+/*
+ * yosys -- Yosys Open SYnthesis Suite
+ *
+ * Copyright (C) 2012 Clifford Wolf <clifford@clifford.at>
+ *
+ * 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) { }
+ unsigned int hash() const { return first->name.hash() + second; }
+ };
+
+ pool<bitDef_t> bits;
+
+ void clear()
+ {
+ bits.clear();
+ }
+
+ void add(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(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(RTLIL::SigSpec from, 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(RTLIL::SigSpec sig)
+ {
+ RTLIL::SigSpec result;
+ for (auto &bit : sig)
+ if (bit.wire != NULL && bits.count(bit))
+ result.append_bit(bit);
+ return result;
+ }
+
+ RTLIL::SigSpec remove(RTLIL::SigSpec sig)
+ {
+ RTLIL::SigSpec result;
+ for (auto &bit : sig)
+ if (bit.wire != NULL && bits.count(bit) == 0)
+ result.append(bit);
+ return result;
+ }
+
+ bool check(RTLIL::SigBit bit)
+ {
+ return bit.wire != NULL && bits.count(bit);
+ }
+
+ bool check_any(RTLIL::SigSpec sig)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL && bits.count(bit))
+ return true;
+ return false;
+ }
+
+ bool check_all(RTLIL::SigSpec sig)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL && bits.count(bit) == 0)
+ return false;
+ return true;
+ }
+
+ RTLIL::SigSpec export_one()
+ {
+ for (auto &bit : bits)
+ return RTLIL::SigSpec(bit.first, bit.second);
+ return RTLIL::SigSpec();
+ }
+
+ RTLIL::SigSpec export_all()
+ {
+ 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 = std::less<T>>
+struct SigSet
+{
+ 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) { }
+ unsigned int hash() const { return first->name.hash() + second; }
+ };
+
+ dict<bitDef_t, std::set<T, Compare>> bits;
+
+ void clear()
+ {
+ bits.clear();
+ }
+
+ void insert(RTLIL::SigSpec sig, T data)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL)
+ bits[bit].insert(data);
+ }
+
+ void insert(RTLIL::SigSpec sig, const std::set<T> &data)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL)
+ bits[bit].insert(data.begin(), data.end());
+ }
+
+ void erase(RTLIL::SigSpec sig)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL)
+ bits[bit].clear();
+ }
+
+ void erase(RTLIL::SigSpec sig, T data)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL)
+ bits[bit].erase(data);
+ }
+
+ void erase(RTLIL::SigSpec sig, const std::set<T> &data)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL)
+ bits[bit].erase(data.begin(), data.end());
+ }
+
+ void find(RTLIL::SigSpec sig, std::set<T> &result)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL) {
+ auto &data = bits[bit];
+ result.insert(data.begin(), data.end());
+ }
+ }
+
+ void find(RTLIL::SigSpec sig, pool<T> &result)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL) {
+ auto &data = bits[bit];
+ result.insert(data.begin(), data.end());
+ }
+ }
+
+ std::set<T> find(RTLIL::SigSpec sig)
+ {
+ std::set<T> result;
+ find(sig, result);
+ return result;
+ }
+
+ bool has(RTLIL::SigSpec sig)
+ {
+ for (auto &bit : sig)
+ if (bit.wire != NULL && bits.count(bit))
+ return true;
+ return false;
+ }
+};
+
+struct SigMap
+{
+ mfp<SigBit> database;
+
+ SigMap(RTLIL::Module *module = NULL)
+ {
+ if (module != NULL)
+ set(module);
+ }
+
+ void swap(SigMap &other)
+ {
+ database.swap(other.database);
+ }
+
+ void clear()
+ {
+ database.clear();
+ }
+
+ 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);
+ }
+
+ void add(RTLIL::SigSpec from, 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);
+ }
+ }
+ }
+
+ void add(RTLIL::SigSpec sig)
+ {
+ for (auto &bit : sig) {
+ RTLIL::SigBit b = database.find(bit);
+ if (b.wire != nullptr)
+ database.promote(bit);
+ }
+ }
+
+ 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;
+ }
+
+ RTLIL::SigSpec allbits() const
+ {
+ RTLIL::SigSpec sig;
+ for (auto &bit : database)
+ if (bit.wire != nullptr)
+ sig.append(bit);
+ return sig;
+ }
+};
+
+YOSYS_NAMESPACE_END
+
+#endif /* SIGTOOLS_H */