mirror of https://github.com/YosysHQ/yosys.git
Merge branch 'master' of github.com:cliffordwolf/yosys
This commit is contained in:
commit
21e1bac084
|
@ -49,6 +49,7 @@ struct Smt2Worker
|
||||||
regsmode(regsmode), wiresmode(wiresmode), verbose(verbose), idcounter(0)
|
regsmode(regsmode), wiresmode(wiresmode), verbose(verbose), idcounter(0)
|
||||||
{
|
{
|
||||||
decls.push_back(stringf("(declare-sort |%s_s| 0)\n", log_id(module)));
|
decls.push_back(stringf("(declare-sort |%s_s| 0)\n", log_id(module)));
|
||||||
|
decls.push_back(stringf("(declare-fun |%s_is| (|%s_s|) Bool)\n", log_id(module), log_id(module)));
|
||||||
|
|
||||||
for (auto cell : module->cells())
|
for (auto cell : module->cells())
|
||||||
for (auto &conn : cell->connections()) {
|
for (auto &conn : cell->connections()) {
|
||||||
|
@ -324,6 +325,16 @@ struct Smt2Worker
|
||||||
exported_cells.insert(cell);
|
exported_cells.insert(cell);
|
||||||
recursive_cells.insert(cell);
|
recursive_cells.insert(cell);
|
||||||
|
|
||||||
|
if (cell->type == "$initstate")
|
||||||
|
{
|
||||||
|
SigBit bit = sigmap(cell->getPort("\\Y").as_bit());
|
||||||
|
decls.push_back(stringf("(define-fun |%s#%d| ((state |%s_s|)) Bool (|%s_is| state)) ; %s\n",
|
||||||
|
log_id(module), idcounter, log_id(module), log_id(module), log_signal(bit)));
|
||||||
|
register_bool(bit, idcounter++);
|
||||||
|
recursive_cells.erase(cell);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (cell->type == "$_DFF_P_" || cell->type == "$_DFF_N_")
|
if (cell->type == "$_DFF_P_" || cell->type == "$_DFF_N_")
|
||||||
{
|
{
|
||||||
registers.insert(cell);
|
registers.insert(cell);
|
||||||
|
@ -755,7 +766,9 @@ struct Smt2Backend : public Backend {
|
||||||
log("the assumptions in the module.\n");
|
log("the assumptions in the module.\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
log("The '<mod>_i' function evaluates to 'true' when the given state conforms\n");
|
log("The '<mod>_i' function evaluates to 'true' when the given state conforms\n");
|
||||||
log("to the initial state.\n");
|
log("to the initial state. Furthermore the '<mod>_is' function should be asserted\n");
|
||||||
|
log("to be true for initial states in addition to '<mod>_i', and should be\n");
|
||||||
|
log("asserted to be false for non-initial states.\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
log("For hierarchical designs, the '<mod>_h' function must be asserted for each\n");
|
log("For hierarchical designs, the '<mod>_h' function must be asserted for each\n");
|
||||||
log("state to establish the design hierarchy. The '<mod>_h <cellname>' function\n");
|
log("state to establish the design hierarchy. The '<mod>_h <cellname>' function\n");
|
||||||
|
|
|
@ -130,6 +130,7 @@ if tempind:
|
||||||
smt.write("(declare-fun s%d () %s_s)" % (step, topmod))
|
smt.write("(declare-fun s%d () %s_s)" % (step, topmod))
|
||||||
smt.write("(assert (%s_u s%d))" % (topmod, step))
|
smt.write("(assert (%s_u s%d))" % (topmod, step))
|
||||||
smt.write("(assert (%s_h s%d))" % (topmod, step))
|
smt.write("(assert (%s_h s%d))" % (topmod, step))
|
||||||
|
smt.write("(assert (not (%s_is s%d)))" % (topmod, step))
|
||||||
|
|
||||||
if step == num_steps:
|
if step == num_steps:
|
||||||
smt.write("(assert (not (%s_a s%d)))" % (topmod, step))
|
smt.write("(assert (not (%s_a s%d)))" % (topmod, step))
|
||||||
|
@ -172,9 +173,11 @@ else: # not tempind
|
||||||
|
|
||||||
if step == 0:
|
if step == 0:
|
||||||
smt.write("(assert (%s_i s0))" % (topmod))
|
smt.write("(assert (%s_i s0))" % (topmod))
|
||||||
|
smt.write("(assert (%s_is s0))" % (topmod))
|
||||||
|
|
||||||
else:
|
else:
|
||||||
smt.write("(assert (%s_t s%d s%d))" % (topmod, step-1, step))
|
smt.write("(assert (%s_t s%d s%d))" % (topmod, step-1, step))
|
||||||
|
smt.write("(assert (not (%s_is s%d)))" % (topmod, step))
|
||||||
|
|
||||||
if step < skip_steps:
|
if step < skip_steps:
|
||||||
if assume_skipped is not None and step >= assume_skipped:
|
if assume_skipped is not None and step >= assume_skipped:
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
module demo1(input clk, input addtwo, output iseven);
|
module demo1(input clk, input addtwo, output iseven);
|
||||||
reg [3:0] cnt = 0;
|
reg [3:0] cnt;
|
||||||
wire [3:0] next_cnt;
|
wire [3:0] next_cnt;
|
||||||
|
|
||||||
inc inc_inst (addtwo, iseven, cnt, next_cnt);
|
inc inc_inst (addtwo, iseven, cnt, next_cnt);
|
||||||
|
@ -8,6 +8,7 @@ module demo1(input clk, input addtwo, output iseven);
|
||||||
cnt = (iseven ? cnt == 10 : cnt == 11) ? 0 : next_cnt;
|
cnt = (iseven ? cnt == 10 : cnt == 11) ? 0 : next_cnt;
|
||||||
|
|
||||||
assert property (cnt != 15);
|
assert property (cnt != 15);
|
||||||
|
initial assume (!cnt[3] && !cnt[0]);
|
||||||
// initial predict ((iseven && addtwo) || cnt == 9);
|
// initial predict ((iseven && addtwo) || cnt == 9);
|
||||||
endmodule
|
endmodule
|
||||||
|
|
||||||
|
|
|
@ -44,7 +44,8 @@ namespace AST {
|
||||||
|
|
||||||
// instanciate global variables (private API)
|
// instanciate global variables (private API)
|
||||||
namespace AST_INTERNAL {
|
namespace AST_INTERNAL {
|
||||||
bool flag_dump_ast1, flag_dump_ast2, flag_dump_vlog, flag_nolatches, flag_nomeminit, flag_nomem2reg, flag_mem2reg, flag_lib, flag_noopt, flag_icells, flag_autowire;
|
bool flag_dump_ast1, flag_dump_ast2, flag_dump_vlog, flag_dump_rtlil, flag_nolatches, flag_nomeminit;
|
||||||
|
bool flag_nomem2reg, flag_mem2reg, flag_lib, flag_noopt, flag_icells, flag_autowire;
|
||||||
AstNode *current_ast, *current_ast_mod;
|
AstNode *current_ast, *current_ast_mod;
|
||||||
std::map<std::string, AstNode*> current_scope;
|
std::map<std::string, AstNode*> current_scope;
|
||||||
const dict<RTLIL::SigBit, RTLIL::SigBit> *genRTLIL_subst_ptr = NULL;
|
const dict<RTLIL::SigBit, RTLIL::SigBit> *genRTLIL_subst_ptr = NULL;
|
||||||
|
@ -175,7 +176,7 @@ bool AstNode::get_bool_attribute(RTLIL::IdString id)
|
||||||
|
|
||||||
// create new node (AstNode constructor)
|
// create new node (AstNode constructor)
|
||||||
// (the optional child arguments make it easier to create AST trees)
|
// (the optional child arguments make it easier to create AST trees)
|
||||||
AstNode::AstNode(AstNodeType type, AstNode *child1, AstNode *child2)
|
AstNode::AstNode(AstNodeType type, AstNode *child1, AstNode *child2, AstNode *child3)
|
||||||
{
|
{
|
||||||
static unsigned int hashidx_count = 123456789;
|
static unsigned int hashidx_count = 123456789;
|
||||||
hashidx_count = mkhash_xorshift(hashidx_count);
|
hashidx_count = mkhash_xorshift(hashidx_count);
|
||||||
|
@ -203,6 +204,8 @@ AstNode::AstNode(AstNodeType type, AstNode *child1, AstNode *child2)
|
||||||
children.push_back(child1);
|
children.push_back(child1);
|
||||||
if (child2)
|
if (child2)
|
||||||
children.push_back(child2);
|
children.push_back(child2);
|
||||||
|
if (child3)
|
||||||
|
children.push_back(child3);
|
||||||
}
|
}
|
||||||
|
|
||||||
// create a (deep recursive) copy of a node
|
// create a (deep recursive) copy of a node
|
||||||
|
@ -969,16 +972,25 @@ static AstModule* process_module(AstNode *ast, bool defer)
|
||||||
current_module->icells = flag_icells;
|
current_module->icells = flag_icells;
|
||||||
current_module->autowire = flag_autowire;
|
current_module->autowire = flag_autowire;
|
||||||
current_module->fixup_ports();
|
current_module->fixup_ports();
|
||||||
|
|
||||||
|
if (flag_dump_rtlil) {
|
||||||
|
log("Dumping generated RTLIL:\n");
|
||||||
|
log_module(current_module);
|
||||||
|
log("--- END OF RTLIL DUMP ---\n");
|
||||||
|
}
|
||||||
|
|
||||||
return current_module;
|
return current_module;
|
||||||
}
|
}
|
||||||
|
|
||||||
// create AstModule instances for all modules in the AST tree and add them to 'design'
|
// create AstModule instances for all modules in the AST tree and add them to 'design'
|
||||||
void AST::process(RTLIL::Design *design, AstNode *ast, bool dump_ast1, bool dump_ast2, bool dump_vlog, bool nolatches, bool nomeminit, bool nomem2reg, bool mem2reg, bool lib, bool noopt, bool icells, bool ignore_redef, bool defer, bool autowire)
|
void AST::process(RTLIL::Design *design, AstNode *ast, bool dump_ast1, bool dump_ast2, bool dump_vlog, bool dump_rtlil,
|
||||||
|
bool nolatches, bool nomeminit, bool nomem2reg, bool mem2reg, bool lib, bool noopt, bool icells, bool ignore_redef, bool defer, bool autowire)
|
||||||
{
|
{
|
||||||
current_ast = ast;
|
current_ast = ast;
|
||||||
flag_dump_ast1 = dump_ast1;
|
flag_dump_ast1 = dump_ast1;
|
||||||
flag_dump_ast2 = dump_ast2;
|
flag_dump_ast2 = dump_ast2;
|
||||||
flag_dump_vlog = dump_vlog;
|
flag_dump_vlog = dump_vlog;
|
||||||
|
flag_dump_rtlil = dump_rtlil;
|
||||||
flag_nolatches = nolatches;
|
flag_nolatches = nolatches;
|
||||||
flag_nomeminit = nomeminit;
|
flag_nomeminit = nomeminit;
|
||||||
flag_nomem2reg = nomem2reg;
|
flag_nomem2reg = nomem2reg;
|
||||||
|
@ -1023,9 +1035,8 @@ void AST::process(RTLIL::Design *design, AstNode *ast, bool dump_ast1, bool dump
|
||||||
|
|
||||||
design->add(process_module(*it, defer));
|
design->add(process_module(*it, defer));
|
||||||
}
|
}
|
||||||
else if ((*it)->type == AST_PACKAGE){
|
else if ((*it)->type == AST_PACKAGE)
|
||||||
design->verilog_packages.push_back((*it)->clone());
|
design->verilog_packages.push_back((*it)->clone());
|
||||||
}
|
|
||||||
else
|
else
|
||||||
global_decls.push_back(*it);
|
global_decls.push_back(*it);
|
||||||
}
|
}
|
||||||
|
|
|
@ -187,7 +187,7 @@ namespace AST
|
||||||
int linenum;
|
int linenum;
|
||||||
|
|
||||||
// creating and deleting nodes
|
// creating and deleting nodes
|
||||||
AstNode(AstNodeType type = AST_NONE, AstNode *child1 = NULL, AstNode *child2 = NULL);
|
AstNode(AstNodeType type = AST_NONE, AstNode *child1 = NULL, AstNode *child2 = NULL, AstNode *child3 = NULL);
|
||||||
AstNode *clone();
|
AstNode *clone();
|
||||||
void cloneInto(AstNode *other);
|
void cloneInto(AstNode *other);
|
||||||
void delete_children();
|
void delete_children();
|
||||||
|
@ -272,7 +272,8 @@ namespace AST
|
||||||
};
|
};
|
||||||
|
|
||||||
// process an AST tree (ast must point to an AST_DESIGN node) and generate RTLIL code
|
// process an AST tree (ast must point to an AST_DESIGN node) and generate RTLIL code
|
||||||
void process(RTLIL::Design *design, AstNode *ast, bool dump_ast1, bool dump_ast2, bool dump_vlog, bool nolatches, bool nomeminit, bool nomem2reg, bool mem2reg, bool lib, bool noopt, bool icells, bool ignore_redef, bool defer, bool autowire);
|
void process(RTLIL::Design *design, AstNode *ast, bool dump_ast1, bool dump_ast2, bool dump_vlog, bool dump_rtlil, bool nolatches, bool nomeminit,
|
||||||
|
bool nomem2reg, bool mem2reg, bool lib, bool noopt, bool icells, bool ignore_redef, bool defer, bool autowire);
|
||||||
|
|
||||||
// parametric modules are supported directly by the AST library
|
// parametric modules are supported directly by the AST library
|
||||||
// therefore we need our own derivate of RTLIL::Module with overloaded virtual functions
|
// therefore we need our own derivate of RTLIL::Module with overloaded virtual functions
|
||||||
|
@ -302,7 +303,8 @@ namespace AST
|
||||||
namespace AST_INTERNAL
|
namespace AST_INTERNAL
|
||||||
{
|
{
|
||||||
// internal state variables
|
// internal state variables
|
||||||
extern bool flag_dump_ast1, flag_dump_ast2, flag_nolatches, flag_nomeminit, flag_nomem2reg, flag_mem2reg, flag_lib, flag_noopt, flag_icells, flag_autowire;
|
extern bool flag_dump_ast1, flag_dump_ast2, flag_dump_rtlil, flag_nolatches, flag_nomeminit;
|
||||||
|
extern bool flag_nomem2reg, flag_mem2reg, flag_lib, flag_noopt, flag_icells, flag_autowire;
|
||||||
extern AST::AstNode *current_ast, *current_ast_mod;
|
extern AST::AstNode *current_ast, *current_ast_mod;
|
||||||
extern std::map<std::string, AST::AstNode*> current_scope;
|
extern std::map<std::string, AST::AstNode*> current_scope;
|
||||||
extern const dict<RTLIL::SigBit, RTLIL::SigBit> *genRTLIL_subst_ptr;
|
extern const dict<RTLIL::SigBit, RTLIL::SigBit> *genRTLIL_subst_ptr;
|
||||||
|
|
|
@ -750,6 +750,19 @@ void AstNode::detectSignWidthWorker(int &width_hint, bool &sign_hint, bool *foun
|
||||||
width_hint = max(width_hint, this_width);
|
width_hint = max(width_hint, this_width);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case AST_FCALL:
|
||||||
|
if (str == "\\$anyconst" || str == "\\$aconst") {
|
||||||
|
if (GetSize(children) == 1) {
|
||||||
|
while (children[0]->simplify(true, false, false, 1, -1, false, true) == true) { }
|
||||||
|
if (children[0]->type != AST_CONSTANT)
|
||||||
|
log_error("System function %s called with non-const argument at %s:%d!\n",
|
||||||
|
RTLIL::unescape_id(str).c_str(), filename.c_str(), linenum);
|
||||||
|
width_hint = max(width_hint, int(children[0]->asInt(true)));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* fall through */
|
||||||
|
|
||||||
// everything should have been handled above -> print error if not.
|
// everything should have been handled above -> print error if not.
|
||||||
default:
|
default:
|
||||||
for (auto f : log_files)
|
for (auto f : log_files)
|
||||||
|
@ -1427,6 +1440,38 @@ RTLIL::SigSpec AstNode::genRTLIL(int width_hint, bool sign_hint)
|
||||||
delete always;
|
delete always;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
|
case AST_FCALL: {
|
||||||
|
if (str == "\\$anyconst" || str == "\\$aconst")
|
||||||
|
{
|
||||||
|
string myid = stringf("%s$%d", RTLIL::unescape_id(str).c_str(), autoidx++);
|
||||||
|
int width = width_hint;
|
||||||
|
|
||||||
|
if (GetSize(children) > 1)
|
||||||
|
log_error("System function %s got %d arguments, expected 1 or 0 at %s:%d.\n",
|
||||||
|
RTLIL::unescape_id(str).c_str(), GetSize(children), filename.c_str(), linenum);
|
||||||
|
|
||||||
|
if (GetSize(children) == 1) {
|
||||||
|
if (children[0]->type != AST_CONSTANT)
|
||||||
|
log_error("System function %s called with non-const argument at %s:%d!\n",
|
||||||
|
RTLIL::unescape_id(str).c_str(), filename.c_str(), linenum);
|
||||||
|
width = children[0]->asInt(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (width <= 0)
|
||||||
|
log_error("Failed to detect width of %s at %s:%d!\n",
|
||||||
|
RTLIL::unescape_id(str).c_str(), filename.c_str(), linenum);
|
||||||
|
|
||||||
|
Cell *cell = current_module->addCell(myid, str.substr(1));
|
||||||
|
cell->parameters["\\WIDTH"] = width;
|
||||||
|
|
||||||
|
Wire *wire = current_module->addWire(myid + "_wire", width);
|
||||||
|
cell->setPort("\\Y", wire);
|
||||||
|
|
||||||
|
is_signed = sign_hint;
|
||||||
|
return SigSpec(wire);
|
||||||
|
}
|
||||||
|
} /* fall through */
|
||||||
|
|
||||||
// everything should have been handled above -> print error if not.
|
// everything should have been handled above -> print error if not.
|
||||||
default:
|
default:
|
||||||
for (auto f : log_files)
|
for (auto f : log_files)
|
||||||
|
|
|
@ -1655,6 +1655,10 @@ skip_dynamic_range_lvalue_expansion:;
|
||||||
goto apply_newNode;
|
goto apply_newNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// $anyconst and $aconst are mapped in AstNode::genRTLIL()
|
||||||
|
if (str == "\\$anyconst" || str == "\\$aconst")
|
||||||
|
return false;
|
||||||
|
|
||||||
if (str == "\\$clog2")
|
if (str == "\\$clog2")
|
||||||
{
|
{
|
||||||
if (children.size() != 1)
|
if (children.size() != 1)
|
||||||
|
|
|
@ -75,6 +75,9 @@ struct VerilogFrontend : public Frontend {
|
||||||
log(" -dump_vlog\n");
|
log(" -dump_vlog\n");
|
||||||
log(" dump ast as Verilog code (after simplification)\n");
|
log(" dump ast as Verilog code (after simplification)\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
|
log(" -dump_rtlil\n");
|
||||||
|
log(" dump generated RTLIL netlist\n");
|
||||||
|
log("\n");
|
||||||
log(" -yydebug\n");
|
log(" -yydebug\n");
|
||||||
log(" enable parser debug output\n");
|
log(" enable parser debug output\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
|
@ -168,6 +171,7 @@ struct VerilogFrontend : public Frontend {
|
||||||
bool flag_dump_ast1 = false;
|
bool flag_dump_ast1 = false;
|
||||||
bool flag_dump_ast2 = false;
|
bool flag_dump_ast2 = false;
|
||||||
bool flag_dump_vlog = false;
|
bool flag_dump_vlog = false;
|
||||||
|
bool flag_dump_rtlil = false;
|
||||||
bool flag_nolatches = false;
|
bool flag_nolatches = false;
|
||||||
bool flag_nomeminit = false;
|
bool flag_nomeminit = false;
|
||||||
bool flag_nomem2reg = false;
|
bool flag_nomem2reg = false;
|
||||||
|
@ -216,6 +220,10 @@ struct VerilogFrontend : public Frontend {
|
||||||
flag_dump_vlog = true;
|
flag_dump_vlog = true;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if (arg == "-dump_rtlil") {
|
||||||
|
flag_dump_rtlil = true;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
if (arg == "-yydebug") {
|
if (arg == "-yydebug") {
|
||||||
frontend_verilog_yydebug = true;
|
frontend_verilog_yydebug = true;
|
||||||
continue;
|
continue;
|
||||||
|
@ -342,7 +350,7 @@ struct VerilogFrontend : public Frontend {
|
||||||
if (flag_nodpi)
|
if (flag_nodpi)
|
||||||
error_on_dpi_function(current_ast);
|
error_on_dpi_function(current_ast);
|
||||||
|
|
||||||
AST::process(design, current_ast, flag_dump_ast1, flag_dump_ast2, flag_dump_vlog, flag_nolatches, flag_nomeminit, flag_nomem2reg, flag_mem2reg, lib_mode, flag_noopt, flag_icells, flag_ignore_redef, flag_defer, default_nettype_wire);
|
AST::process(design, current_ast, flag_dump_ast1, flag_dump_ast2, flag_dump_vlog, flag_dump_rtlil, flag_nolatches, flag_nomeminit, flag_nomem2reg, flag_mem2reg, lib_mode, flag_noopt, flag_icells, flag_ignore_redef, flag_defer, default_nettype_wire);
|
||||||
|
|
||||||
if (!flag_nopp)
|
if (!flag_nopp)
|
||||||
delete lexin;
|
delete lexin;
|
||||||
|
|
|
@ -1219,7 +1219,7 @@ rvalue:
|
||||||
$$ = new AstNode(AST_IDENTIFIER, $2);
|
$$ = new AstNode(AST_IDENTIFIER, $2);
|
||||||
$$->str = *$1;
|
$$->str = *$1;
|
||||||
delete $1;
|
delete $1;
|
||||||
if ($2 == nullptr && $$->str == "\\$initstate")
|
if ($2 == nullptr && formal_mode && ($$->str == "\\$initstate" || $$->str == "\\$anyconst" || $$->str == "\\$aconst"))
|
||||||
$$->type = AST_FCALL;
|
$$->type = AST_FCALL;
|
||||||
} |
|
} |
|
||||||
hierarchical_id non_opt_multirange {
|
hierarchical_id non_opt_multirange {
|
||||||
|
|
|
@ -118,6 +118,8 @@ struct CellTypes
|
||||||
setup_type("$assume", {A, EN}, pool<RTLIL::IdString>(), true);
|
setup_type("$assume", {A, EN}, pool<RTLIL::IdString>(), true);
|
||||||
setup_type("$predict", {A, EN}, pool<RTLIL::IdString>(), true);
|
setup_type("$predict", {A, EN}, pool<RTLIL::IdString>(), true);
|
||||||
setup_type("$initstate", pool<RTLIL::IdString>(), {Y}, true);
|
setup_type("$initstate", pool<RTLIL::IdString>(), {Y}, true);
|
||||||
|
setup_type("$anyconst", pool<RTLIL::IdString>(), {Y}, true);
|
||||||
|
setup_type("$aconst", pool<RTLIL::IdString>(), {Y}, true);
|
||||||
setup_type("$equiv", {A, B}, {Y}, true);
|
setup_type("$equiv", {A, B}, {Y}, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -427,6 +427,13 @@ const char *log_id(RTLIL::IdString str)
|
||||||
return p+1;
|
return p+1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void log_module(RTLIL::Module *module, std::string indent)
|
||||||
|
{
|
||||||
|
std::stringstream buf;
|
||||||
|
ILANG_BACKEND::dump_module(buf, indent, module, module->design, false);
|
||||||
|
log("%s", buf.str().c_str());
|
||||||
|
}
|
||||||
|
|
||||||
void log_cell(RTLIL::Cell *cell, std::string indent)
|
void log_cell(RTLIL::Cell *cell, std::string indent)
|
||||||
{
|
{
|
||||||
std::stringstream buf;
|
std::stringstream buf;
|
||||||
|
|
|
@ -85,6 +85,7 @@ template<typename T> static inline const char *log_id(T *obj) {
|
||||||
return log_id(obj->name);
|
return log_id(obj->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void log_module(RTLIL::Module *module, std::string indent = "");
|
||||||
void log_cell(RTLIL::Cell *cell, std::string indent = "");
|
void log_cell(RTLIL::Cell *cell, std::string indent = "");
|
||||||
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
|
|
|
@ -1030,6 +1030,12 @@ namespace {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cell->type.in("$aconst", "$anyconst")) {
|
||||||
|
port("\\Y", param("\\WIDTH"));
|
||||||
|
check_expected();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (cell->type == "$equiv") {
|
if (cell->type == "$equiv") {
|
||||||
port("\\A", 1);
|
port("\\A", 1);
|
||||||
port("\\B", 1);
|
port("\\B", 1);
|
||||||
|
|
|
@ -69,7 +69,6 @@ struct SatGen
|
||||||
SigPool initial_state;
|
SigPool initial_state;
|
||||||
std::map<std::string, RTLIL::SigSpec> asserts_a, asserts_en;
|
std::map<std::string, RTLIL::SigSpec> asserts_a, asserts_en;
|
||||||
std::map<std::string, RTLIL::SigSpec> assumes_a, assumes_en;
|
std::map<std::string, RTLIL::SigSpec> assumes_a, assumes_en;
|
||||||
std::map<std::string, RTLIL::SigSpec> predict_a, predict_en;
|
|
||||||
std::map<std::string, std::map<RTLIL::SigBit, int>> imported_signals;
|
std::map<std::string, std::map<RTLIL::SigBit, int>> imported_signals;
|
||||||
std::map<std::pair<std::string, int>, bool> initstates;
|
std::map<std::pair<std::string, int>, bool> initstates;
|
||||||
bool ignore_div_by_zero;
|
bool ignore_div_by_zero;
|
||||||
|
@ -1320,6 +1319,28 @@ struct SatGen
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cell->type == "$anyconst")
|
||||||
|
{
|
||||||
|
if (timestep < 2)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
std::vector<int> d = importDefSigSpec(cell->getPort("\\Y"), timestep-1);
|
||||||
|
std::vector<int> q = importDefSigSpec(cell->getPort("\\Y"), timestep);
|
||||||
|
|
||||||
|
std::vector<int> qq = model_undef ? ez->vec_var(q.size()) : q;
|
||||||
|
ez->assume(ez->vec_eq(d, qq));
|
||||||
|
|
||||||
|
if (model_undef)
|
||||||
|
{
|
||||||
|
std::vector<int> undef_d = importUndefSigSpec(cell->getPort("\\D"), timestep-1);
|
||||||
|
std::vector<int> undef_q = importUndefSigSpec(cell->getPort("\\Q"), timestep);
|
||||||
|
|
||||||
|
ez->assume(ez->vec_eq(undef_d, undef_q));
|
||||||
|
undefGating(q, qq, undef_q);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (cell->type == "$_BUF_" || cell->type == "$equiv")
|
if (cell->type == "$_BUF_" || cell->type == "$equiv")
|
||||||
{
|
{
|
||||||
std::vector<int> a = importDefSigSpec(cell->getPort("\\A"), timestep);
|
std::vector<int> a = importDefSigSpec(cell->getPort("\\A"), timestep);
|
||||||
|
@ -1374,14 +1395,6 @@ struct SatGen
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cell->type == "$predict")
|
|
||||||
{
|
|
||||||
std::string pf = prefix + (timestep == -1 ? "" : stringf("@%d:", timestep));
|
|
||||||
predict_a[pf].append((*sigmap)(cell->getPort("\\A")));
|
|
||||||
predict_en[pf].append((*sigmap)(cell->getPort("\\EN")));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unsupported internal cell types: $pow $lut
|
// Unsupported internal cell types: $pow $lut
|
||||||
// .. and all sequential cells except $dff and $_DFF_[NP]_
|
// .. and all sequential cells except $dff and $_DFF_[NP]_
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -421,7 +421,7 @@ pass. The combinatorial logic cells can be mapped to physical cells from a Liber
|
||||||
using the {\tt abc} pass.
|
using the {\tt abc} pass.
|
||||||
|
|
||||||
\begin{fixme}
|
\begin{fixme}
|
||||||
Add information about {\tt \$assert}, {\tt \$assume}, {\tt \$predict}, {\tt \$equiv}, and {\tt \$initstate} cells.
|
Add information about {\tt \$assert}, {\tt \$assume}, {\tt \$predict}, {\tt \$equiv}, {\tt \$initstate}, {\tt \$aconst}, and {\tt \$anyconst} cells.
|
||||||
\end{fixme}
|
\end{fixme}
|
||||||
|
|
||||||
\begin{fixme}
|
\begin{fixme}
|
||||||
|
|
|
@ -1330,6 +1330,30 @@ endmodule
|
||||||
|
|
||||||
// --------------------------------------------------------
|
// --------------------------------------------------------
|
||||||
|
|
||||||
|
module \$aconst (Y);
|
||||||
|
|
||||||
|
parameter WIDTH = 0;
|
||||||
|
|
||||||
|
output [WIDTH-1:0] Y;
|
||||||
|
|
||||||
|
assign Y = 'bx;
|
||||||
|
|
||||||
|
endmodule
|
||||||
|
|
||||||
|
// --------------------------------------------------------
|
||||||
|
|
||||||
|
module \$anyconst (Y);
|
||||||
|
|
||||||
|
parameter WIDTH = 0;
|
||||||
|
|
||||||
|
output [WIDTH-1:0] Y;
|
||||||
|
|
||||||
|
assign Y = 'bx;
|
||||||
|
|
||||||
|
endmodule
|
||||||
|
|
||||||
|
// --------------------------------------------------------
|
||||||
|
|
||||||
module \$equiv (A, B, Y);
|
module \$equiv (A, B, Y);
|
||||||
|
|
||||||
input A, B;
|
input A, B;
|
||||||
|
|
Loading…
Reference in New Issue