Added hashlib::mfp and new SigMap
authorClifford Wolf <clifford@clifford.at>
Tue, 27 Oct 2015 14:04:47 +0000 (15:04 +0100)
committerClifford Wolf <clifford@clifford.at>
Tue, 27 Oct 2015 14:04:47 +0000 (15:04 +0100)
kernel/hashlib.h
kernel/sigtools.h
kernel/yosys.h

index 4f5a353c5ec713d2b1331343ec493eafc93714c5..83c112f3cc6d69c05e0262444f5e605fc6483d8b 100644 (file)
@@ -195,6 +195,7 @@ inline int hashtable_size(int min_size)
 template<typename K, typename T, typename OPS = hash_ops<K>> class dict;
 template<typename K, int offset = 0, typename OPS = hash_ops<K>> class idict;
 template<typename K, typename OPS = hash_ops<K>> class pool;
+template<typename K, typename OPS = hash_ops<K>> class mfp;
 
 template<typename K, typename T, typename OPS>
 class dict
@@ -914,10 +915,102 @@ public:
                return database.entries.at(index - offset).udata;
        }
 
+       void swap(idict &other)
+       {
+               database.swap(other.database);
+       }
+
+       size_t size() const { return database.size(); }
+       bool empty() const { return database.empty(); }
+       void clear() { database.clear(); }
+
        const_iterator begin() const { return database.begin(); }
        const_iterator end() const { return database.end(); }
 };
 
+template<typename K, typename OPS>
+class mfp
+{
+       mutable idict<K, 0, OPS> database;
+       mutable std::vector<int> parents;
+
+public:
+       int operator()(const K &key) const
+       {
+               int i = database(key);
+               parents.resize(database.size(), -1);
+               return i;
+       }
+
+       const K &operator[](int index) const
+       {
+               return database[index];
+       }
+
+       int ifind(int i) const
+       {
+               int p = i, k = i;
+
+               while (parents[p] != -1)
+                       p = parents[p];
+
+               while (k != p) {
+                       int next_k = parents[k];
+                       parents[k] = p;
+                       k = next_k;
+               }
+
+               return p;
+       }
+
+       void imerge(int i, int j)
+       {
+               i = ifind(i);
+               j = ifind(j);
+
+               if (i != j)
+                       parents[i] = j;
+       }
+
+       void ipromote(int i)
+       {
+               int k = i;
+
+               while (k != -1) {
+                       int next_k = parents[k];
+                       parents[k] = i;
+                       k = next_k;
+               }
+
+               parents[i] = -1;
+       }
+
+       const K &find(const K &a) const
+       {
+               return (*this)[ifind((*this)(a))];
+       }
+
+       void merge(const K &a, const K &b)
+       {
+               imerge((*this)(a), (*this)(b));
+       }
+
+       void promote(const K &a)
+       {
+               ipromote((*this)(a));
+       }
+
+       void swap(mfp &other)
+       {
+               database.swap(other.database);
+               parents.swap(other.parents);
+       }
+
+       size_t size() const { return database.size(); }
+       bool empty() const { return database.empty(); }
+       void clear() { database.clear(); parents.clear(); }
+};
+
 } /* namespace hashlib */
 
 #endif
index a9419f876038874125239badb5b46e7026a04d7c..b382cdeb27c0064ed8dd4d62f831fb579b4d57c4 100644 (file)
@@ -220,6 +220,96 @@ struct SigSet
        }
 };
 
+#if 1
+struct SigMap
+{
+       mfp<SigBit> database;
+
+       SigMap(RTLIL::Module *module = NULL)
+       {
+               if (module != NULL)
+                       set(module);
+       }
+
+       // SigMap(const SigMap &other)
+       // {
+       //      copy(other);
+       // }
+
+       // const SigMap &operator=(const SigMap &other)
+       // {
+       //      copy(other);
+       //      return *this;
+       // }
+
+       void swap(SigMap &other)
+       {
+               database.swap(other.database);
+       }
+
+       void clear()
+       {
+               database.clear();
+       }
+
+       void set(RTLIL::Module *module)
+       {
+               clear();
+               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++)
+               {
+                       RTLIL::SigBit &bf = from[i];
+                       RTLIL::SigBit &bt = to[i];
+
+                       if (bf.wire != nullptr)
+                               database.merge(bf, bt);
+               }
+       }
+
+       void add(RTLIL::SigSpec sig)
+       {
+               for (auto &bit : sig)
+                       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;
+       }
+};
+#else
 struct SigMap
 {
        struct bitDef_t : public std::pair<RTLIL::Wire*, int> {
@@ -430,6 +520,7 @@ struct SigMap
                return sig;
        }
 };
+#endif
 
 YOSYS_NAMESPACE_END
 
index 89a4f6084642796166a27662445e0c312b0b1969..92fa6ac1928dafdee58cbb690d8a71eaf86a19a2 100644 (file)
@@ -171,6 +171,7 @@ using hashlib::hash_obj_ops;
 using hashlib::dict;
 using hashlib::idict;
 using hashlib::pool;
+using hashlib::mfp;
 
 namespace RTLIL {
        struct IdString;