summaryrefslogtreecommitdiff
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/rtlil.cc89
-rw-r--r--kernel/rtlil.h24
2 files changed, 101 insertions, 12 deletions
diff --git a/kernel/rtlil.cc b/kernel/rtlil.cc
index a1091286..6271aeef 100644
--- a/kernel/rtlil.cc
+++ b/kernel/rtlil.cc
@@ -341,6 +341,47 @@ void RTLIL::Module::optimize()
}
}
+void RTLIL::Module::cloneInto(RTLIL::Module *new_mod) const
+{
+ new_mod->name = name;
+ new_mod->connections = connections;
+ new_mod->attributes = attributes;
+
+ for (auto &it : wires)
+ new_mod->wires[it.first] = new RTLIL::Wire(*it.second);
+
+ for (auto &it : memories)
+ new_mod->memories[it.first] = new RTLIL::Memory(*it.second);
+
+ for (auto &it : cells)
+ new_mod->cells[it.first] = new RTLIL::Cell(*it.second);
+
+ for (auto &it : processes)
+ new_mod->processes[it.first] = it.second->clone();
+
+ struct RewriteSigSpecWorker
+ {
+ RTLIL::Module *mod;
+ void operator()(RTLIL::SigSpec &sig)
+ {
+ for (auto &c : sig.chunks)
+ if (c.wire != NULL)
+ c.wire = mod->wires.at(c.wire->name);
+ }
+ };
+
+ RewriteSigSpecWorker rewriteSigSpecWorker;
+ rewriteSigSpecWorker.mod = new_mod;
+ new_mod->rewrite_sigspecs(rewriteSigSpecWorker);
+}
+
+RTLIL::Module *RTLIL::Module::clone() const
+{
+ RTLIL::Module *new_mod = new RTLIL::Module;
+ cloneInto(new_mod);
+ return new_mod;
+}
+
void RTLIL::Module::add(RTLIL::Wire *wire)
{
assert(!wire->name.empty());
@@ -1165,6 +1206,16 @@ void RTLIL::CaseRule::optimize()
}
}
+RTLIL::CaseRule *RTLIL::CaseRule::clone() const
+{
+ RTLIL::CaseRule *new_caserule = new RTLIL::CaseRule;
+ new_caserule->compare = compare;
+ new_caserule->actions = actions;
+ for (auto &it : switches)
+ new_caserule->switches.push_back(it->clone());
+ return new_caserule;
+}
+
RTLIL::SwitchRule::~SwitchRule()
{
for (auto it = cases.begin(); it != cases.end(); it++)
@@ -1178,6 +1229,17 @@ void RTLIL::SwitchRule::optimize()
it->optimize();
}
+RTLIL::SwitchRule *RTLIL::SwitchRule::clone() const
+{
+ RTLIL::SwitchRule *new_switchrule = new RTLIL::SwitchRule;
+ new_switchrule->signal = signal;
+ new_switchrule->attributes = attributes;
+ for (auto &it : cases)
+ new_switchrule->cases.push_back(it->clone());
+ return new_switchrule;
+
+}
+
void RTLIL::SyncRule::optimize()
{
signal.optimize();
@@ -1187,6 +1249,15 @@ void RTLIL::SyncRule::optimize()
}
}
+RTLIL::SyncRule *RTLIL::SyncRule::clone() const
+{
+ RTLIL::SyncRule *new_syncrule = new RTLIL::SyncRule;
+ new_syncrule->type = type;
+ new_syncrule->signal = signal;
+ new_syncrule->actions = actions;
+ return new_syncrule;
+}
+
RTLIL::Process::~Process()
{
for (auto it = syncs.begin(); it != syncs.end(); it++)
@@ -1200,3 +1271,21 @@ void RTLIL::Process::optimize()
it->optimize();
}
+RTLIL::Process *RTLIL::Process::clone() const
+{
+ RTLIL::Process *new_proc = new RTLIL::Process;
+
+ new_proc->name = name;
+ new_proc->attributes = attributes;
+
+ RTLIL::CaseRule *rc_ptr = root_case.clone();
+ new_proc->root_case = *rc_ptr;
+ rc_ptr->switches.clear();
+ delete rc_ptr;
+
+ for (auto &it : syncs)
+ new_proc->syncs.push_back(it->clone());
+
+ return new_proc;
+}
+
diff --git a/kernel/rtlil.h b/kernel/rtlil.h
index 6dde475d..f5e8ae47 100644
--- a/kernel/rtlil.h
+++ b/kernel/rtlil.h
@@ -239,8 +239,9 @@ struct RTLIL::Module {
void add(RTLIL::Cell *cell);
void fixup_ports();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
+ void cloneInto(RTLIL::Module *new_mod) const;
+ virtual RTLIL::Module *clone() const;
};
struct RTLIL::Wire {
@@ -266,8 +267,7 @@ struct RTLIL::Cell {
std::map<RTLIL::IdString, RTLIL::Const> parameters;
void optimize();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
};
struct RTLIL::SigChunk {
@@ -337,8 +337,8 @@ struct RTLIL::CaseRule {
~CaseRule();
void optimize();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
+ RTLIL::CaseRule *clone() const;
};
struct RTLIL::SwitchRule {
@@ -348,8 +348,8 @@ struct RTLIL::SwitchRule {
~SwitchRule();
void optimize();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
+ RTLIL::SwitchRule *clone() const;
};
struct RTLIL::SyncRule {
@@ -358,8 +358,8 @@ struct RTLIL::SyncRule {
std::vector<RTLIL::SigSig> actions;
void optimize();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
+ RTLIL::SyncRule *clone() const;
};
struct RTLIL::Process {
@@ -370,8 +370,8 @@ struct RTLIL::Process {
~Process();
void optimize();
- template<typename T>
- void rewrite_sigspecs(T functor);
+ template<typename T> void rewrite_sigspecs(T functor);
+ RTLIL::Process *clone() const;
};
template<typename T>