From 325b764341091732f54d1135b57a8d75ce7b7c2a Mon Sep 17 00:00:00 2001 From: Clifford Wolf Date: Sat, 7 Dec 2013 11:58:22 +0100 Subject: [PATCH] Added eval -set-undef and eval -table --- passes/sat/eval.cc | 151 +++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 140 insertions(+), 11 deletions(-) diff --git a/passes/sat/eval.cc b/passes/sat/eval.cc index 85a136cba..2ae74063f 100644 --- a/passes/sat/eval.cc +++ b/passes/sat/eval.cc @@ -381,6 +381,12 @@ struct EvalPass : public Pass { log(" -set \n"); log(" set the specified signal to the specified value.\n"); log("\n"); + log(" -set-undef\n"); + log(" set all unspecified source signals to undef (x)\n"); + log("\n"); + log(" -table \n"); + log(" create a truth table using the specified input signals\n"); + log("\n"); log(" -show \n"); log(" show the value for the specified signal. if no -show option is passed\n"); log(" then all output ports of the current module are used.\n"); @@ -389,7 +395,8 @@ struct EvalPass : public Pass { virtual void execute(std::vector args, RTLIL::Design *design) { std::vector> sets; - std::vector shows; + std::vector shows, tables; + bool set_undef = false; log_header("Executing EVAL pass (evaluate the circuit given an input).\n"); @@ -401,10 +408,18 @@ struct EvalPass : public Pass { sets.push_back(std::pair(lhs, rhs)); continue; } + if (args[argidx] == "-set-undef") { + set_undef = true; + continue; + } if (args[argidx] == "-show" && argidx+1 < args.size()) { shows.push_back(args[++argidx]); continue; } + if (args[argidx] == "-table" && argidx+1 < args.size()) { + tables.push_back(args[++argidx]); + continue; + } if ((args[argidx] == "-brute_force_equiv_checker" || args[argidx] == "-brute_force_equiv_checker_x") && argidx+3 == args.size()) { /* this should only be used for regression testing of ConstEval -- see vloghammer */ std::string mod1_name = RTLIL::escape_id(args[++argidx]); @@ -467,16 +482,130 @@ struct EvalPass : public Pass { shows.push_back(it.second->name); } - for (auto &it : shows) { - RTLIL::SigSpec signal, value, undef; - if (!RTLIL::SigSpec::parse(signal, module, it)) - log_cmd_error("Failed to parse lhs set expression `%s'.\n", it.c_str()); - signal.optimize(); - value = signal; - if (!ce.eval(value, undef)) - log("Failed to evaluate signal %s: Missing value for %s.\n", log_signal(signal), log_signal(undef)); - else - log("Eval result: %s = %s.\n", log_signal(signal), log_signal(value)); + if (tables.empty()) + { + for (auto &it : shows) { + RTLIL::SigSpec signal, value, undef; + if (!RTLIL::SigSpec::parse(signal, module, it)) + log_cmd_error("Failed to parse show expression `%s'.\n", it.c_str()); + signal.optimize(); + value = signal; + if (set_undef) { + while (!ce.eval(value, undef)) { + log("Failed to evaluate signal %s: Missing value for %s. -> setting to undef\n", log_signal(signal), log_signal(undef)); + ce.set(undef, RTLIL::Const(RTLIL::State::Sx, undef.width)); + undef = RTLIL::SigSpec(); + } + log("Eval result: %s = %s.\n", log_signal(signal), log_signal(value)); + } else { + if (!ce.eval(value, undef)) + log("Failed to evaluate signal %s: Missing value for %s.\n", log_signal(signal), log_signal(undef)); + else + log("Eval result: %s = %s.\n", log_signal(signal), log_signal(value)); + } + } + } + else + { + RTLIL::SigSpec tabsigs, signal, value, undef; + std::vector> tab; + int tab_sep_colidx = 0; + + for (auto &it : shows) { + RTLIL::SigSpec sig; + if (!RTLIL::SigSpec::parse(sig, module, it)) + log_cmd_error("Failed to parse show expression `%s'.\n", it.c_str()); + signal.append(sig); + } + + for (auto &it : tables) { + RTLIL::SigSpec sig; + if (!RTLIL::SigSpec::parse(sig, module, it)) + log_cmd_error("Failed to parse table expression `%s'.\n", it.c_str()); + tabsigs.append(sig); + } + + std::vector tab_line; + for (auto &c : tabsigs.chunks) + tab_line.push_back(log_signal(c)); + tab_sep_colidx = tab_line.size(); + for (auto &c : signal.chunks) + tab_line.push_back(log_signal(c)); + tab.push_back(tab_line); + tab_line.clear(); + + RTLIL::Const tabvals(0, tabsigs.width); + do + { + ce.push(); + ce.set(tabsigs, tabvals); + value = signal; + + RTLIL::SigSpec this_undef; + while (!ce.eval(value, this_undef)) { + if (!set_undef) { + log("Failed to evaluate signal %s at %s = %s: Missing value for %s.\n", log_signal(signal), + log_signal(tabsigs), log_signal(tabvals), log_signal(this_undef)); + return; + } + ce.set(this_undef, RTLIL::Const(RTLIL::State::Sx, this_undef.width)); + undef.append(this_undef); + this_undef = RTLIL::SigSpec(); + } + + int pos = 0; + for (auto &c : tabsigs.chunks) { + tab_line.push_back(log_signal(RTLIL::SigSpec(tabvals).extract(pos, c.width))); + pos += c.width; + } + + pos = 0; + for (auto &c : signal.chunks) { + tab_line.push_back(log_signal(value.extract(pos, c.width))); + pos += c.width; + } + + tab.push_back(tab_line); + tab_line.clear(); + ce.pop(); + + tabvals = RTLIL::const_add(tabvals, RTLIL::Const(1), false, false, tabvals.bits.size()); + } + while (tabvals.as_bool()); + + std::vector tab_column_width; + for (auto &row : tab) { + if (tab_column_width.size() < row.size()) + tab_column_width.resize(row.size()); + for (size_t i = 0; i < row.size(); i++) + tab_column_width[i] = std::max(tab_column_width[i], int(row[i].size())); + } + + log("\n"); + bool first = true; + for (auto &row : tab) { + for (size_t i = 0; i < row.size(); i++) { + int k = i < tab_sep_colidx ? tab_sep_colidx - i - 1 : i; + log(" %s%*s", k == tab_sep_colidx ? "| " : "", tab_column_width[k], row[k].c_str()); + } + log("\n"); + if (first) { + for (size_t i = 0; i < row.size(); i++) { + int k = i < tab_sep_colidx ? tab_sep_colidx - i - 1 : i; + log(" %s", k == tab_sep_colidx ? "| " : ""); + for (int j = 0; j < tab_column_width[k]; j++) + log("-"); + } + log("\n"); + first = false; + } + } + + log("\n"); + if (undef.width > 0) { + undef.sort_and_unify(); + log("Assumend undef (x) value for the following singals: %s\n\n", log_signal(undef)); + } } } } EvalPass;