mirror of
https://github.com/verilator/verilator.git
synced 2025-01-01 04:07:34 +00:00
Add --pins-sc-uint-bool
to force SystemC uint type (#5192)
This commit is contained in:
parent
4c584c17e6
commit
9e2c8aefc8
@ -410,6 +410,7 @@ detailed descriptions of these arguments.
|
|||||||
--pins-inout-enables Specify that __en and __out signals be created for inouts
|
--pins-inout-enables Specify that __en and __out signals be created for inouts
|
||||||
--pins-sc-biguint Specify types for top-level ports
|
--pins-sc-biguint Specify types for top-level ports
|
||||||
--pins-sc-uint Specify types for top-level ports
|
--pins-sc-uint Specify types for top-level ports
|
||||||
|
--pins-sc-uint-bool Specify types for top-level ports
|
||||||
--pins-uint8 Specify types for top-level ports
|
--pins-uint8 Specify types for top-level ports
|
||||||
--no-pins64 Don't use uint64_t's for 33-64 bit sigs
|
--no-pins64 Don't use uint64_t's for 33-64 bit sigs
|
||||||
--pipe-filter <command> Filter all input through a script
|
--pipe-filter <command> Filter all input through a script
|
||||||
|
@ -1004,6 +1004,10 @@ Summary:
|
|||||||
:vlopt:`--pins-sc-biguint` combination, it results in sc_uint being used
|
:vlopt:`--pins-sc-biguint` combination, it results in sc_uint being used
|
||||||
between 2 and 64 and sc_biguint being used between 65 and 512.
|
between 2 and 64 and sc_biguint being used between 65 and 512.
|
||||||
|
|
||||||
|
.. option:: --pins-sc-uint-bool
|
||||||
|
|
||||||
|
Specifies SystemC inputs/outputs one bit wide should use sc_uint<1>.
|
||||||
|
|
||||||
.. option:: --pins-sc-biguint
|
.. option:: --pins-sc-biguint
|
||||||
|
|
||||||
Specifies SystemC inputs/outputs greater than 65 bits wide should use
|
Specifies SystemC inputs/outputs greater than 65 bits wide should use
|
||||||
|
@ -2067,6 +2067,7 @@ public:
|
|||||||
bool isScQuad() const;
|
bool isScQuad() const;
|
||||||
bool isScBv() const;
|
bool isScBv() const;
|
||||||
bool isScUint() const;
|
bool isScUint() const;
|
||||||
|
bool isScUintBool() const;
|
||||||
bool isScBigUint() const;
|
bool isScBigUint() const;
|
||||||
bool isScSensitive() const { return m_scSensitive; }
|
bool isScSensitive() const { return m_scSensitive; }
|
||||||
bool isSigPublic() const;
|
bool isSigPublic() const;
|
||||||
|
@ -380,6 +380,9 @@ bool AstVar::isScBv() const {
|
|||||||
bool AstVar::isScUint() const {
|
bool AstVar::isScUint() const {
|
||||||
return ((isSc() && v3Global.opt.pinsScUint() && width() >= 2 && width() <= 64) && !isScBv());
|
return ((isSc() && v3Global.opt.pinsScUint() && width() >= 2 && width() <= 64) && !isScBv());
|
||||||
}
|
}
|
||||||
|
bool AstVar::isScUintBool() const {
|
||||||
|
return (isSc() && v3Global.opt.pinsScUintBool() && width() == 1);
|
||||||
|
}
|
||||||
bool AstVar::isScBigUint() const {
|
bool AstVar::isScBigUint() const {
|
||||||
return ((isSc() && v3Global.opt.pinsScBigUint() && width() >= 65 && width() <= 512)
|
return ((isSc() && v3Global.opt.pinsScBigUint() && width() >= 65 && width() <= 512)
|
||||||
&& !isScBv());
|
&& !isScBv());
|
||||||
@ -671,7 +674,7 @@ string AstVar::scType() const {
|
|||||||
if (isScBigUint()) {
|
if (isScBigUint()) {
|
||||||
return (string{"sc_dt::sc_biguint<"} + cvtToStr(widthMin())
|
return (string{"sc_dt::sc_biguint<"} + cvtToStr(widthMin())
|
||||||
+ "> "); // Keep the space so don't get >>
|
+ "> "); // Keep the space so don't get >>
|
||||||
} else if (isScUint()) {
|
} else if (isScUint() || isScUintBool()) {
|
||||||
return (string{"sc_dt::sc_uint<"} + cvtToStr(widthMin())
|
return (string{"sc_dt::sc_uint<"} + cvtToStr(widthMin())
|
||||||
+ "> "); // Keep the space so don't get >>
|
+ "> "); // Keep the space so don't get >>
|
||||||
} else if (isScBv()) {
|
} else if (isScBv()) {
|
||||||
|
@ -192,10 +192,10 @@ public:
|
|||||||
}
|
}
|
||||||
void emitScIQW(AstVar* nodep) {
|
void emitScIQW(AstVar* nodep) {
|
||||||
UASSERT_OBJ(nodep->isSc(), nodep, "emitting SystemC operator on non-SC variable");
|
UASSERT_OBJ(nodep->isSc(), nodep, "emitting SystemC operator on non-SC variable");
|
||||||
puts(nodep->isScBigUint() ? "SB"
|
puts(nodep->isScBigUint() ? "SB"
|
||||||
: nodep->isScUint() ? "SU"
|
: (nodep->isScUint() || nodep->isScUintBool()) ? "SU"
|
||||||
: nodep->isScBv() ? "SW"
|
: nodep->isScBv() ? "SW"
|
||||||
: (nodep->isScQuad() ? "SQ" : "SI"));
|
: (nodep->isScQuad() ? "SQ" : "SI"));
|
||||||
}
|
}
|
||||||
void emitDatap(AstNode* nodep) {
|
void emitDatap(AstNode* nodep) {
|
||||||
// When passing to a function with va_args the compiler doesn't
|
// When passing to a function with va_args the compiler doesn't
|
||||||
|
@ -694,7 +694,7 @@ class EmitCTrace final : EmitCFunc {
|
|||||||
const AstVarRef* const varrefp = VN_CAST(nodep->declp()->valuep(), VarRef);
|
const AstVarRef* const varrefp = VN_CAST(nodep->declp()->valuep(), VarRef);
|
||||||
if (!varrefp) return false;
|
if (!varrefp) return false;
|
||||||
AstVar* const varp = varrefp->varp();
|
AstVar* const varp = varrefp->varp();
|
||||||
return varp->isSc() && varp->isScUint();
|
return varp->isSc() && (varp->isScUint() || varp->isScUintBool());
|
||||||
}
|
}
|
||||||
|
|
||||||
void emitTraceInitOne(AstTraceDecl* nodep, int enumNum) {
|
void emitTraceInitOne(AstTraceDecl* nodep, int enumNum) {
|
||||||
|
@ -1406,6 +1406,7 @@ void V3Options::parseOptsList(FileLine* fl, const string& optdir, int argc,
|
|||||||
m_pinsScUint = flag;
|
m_pinsScUint = flag;
|
||||||
if (!m_pinsScBigUint) m_pinsBv = 65;
|
if (!m_pinsScBigUint) m_pinsBv = 65;
|
||||||
});
|
});
|
||||||
|
DECL_OPTION("-pins-sc-uint-bool", CbOnOff, [this](bool flag) { m_pinsScUintBool = flag; });
|
||||||
DECL_OPTION("-pins-sc-biguint", CbOnOff, [this](bool flag) {
|
DECL_OPTION("-pins-sc-biguint", CbOnOff, [this](bool flag) {
|
||||||
m_pinsScBigUint = flag;
|
m_pinsScBigUint = flag;
|
||||||
m_pinsBv = 513;
|
m_pinsBv = 513;
|
||||||
|
@ -260,6 +260,7 @@ private:
|
|||||||
bool m_pedantic = false; // main switch: --Wpedantic
|
bool m_pedantic = false; // main switch: --Wpedantic
|
||||||
bool m_pinsInoutEnables = false;// main switch: --pins-inout-enables
|
bool m_pinsInoutEnables = false;// main switch: --pins-inout-enables
|
||||||
bool m_pinsScUint = false; // main switch: --pins-sc-uint
|
bool m_pinsScUint = false; // main switch: --pins-sc-uint
|
||||||
|
bool m_pinsScUintBool = false; // main switch: --pins-sc-uint-bool
|
||||||
bool m_pinsScBigUint = false; // main switch: --pins-sc-biguint
|
bool m_pinsScBigUint = false; // main switch: --pins-sc-biguint
|
||||||
bool m_pinsUint8 = false; // main switch: --pins-uint8
|
bool m_pinsUint8 = false; // main switch: --pins-uint8
|
||||||
bool m_ppComments = false; // main switch: --pp-comments
|
bool m_ppComments = false; // main switch: --pp-comments
|
||||||
@ -514,6 +515,7 @@ public:
|
|||||||
bool pedantic() const { return m_pedantic; }
|
bool pedantic() const { return m_pedantic; }
|
||||||
bool pinsInoutEnables() const { return m_pinsInoutEnables; }
|
bool pinsInoutEnables() const { return m_pinsInoutEnables; }
|
||||||
bool pinsScUint() const { return m_pinsScUint; }
|
bool pinsScUint() const { return m_pinsScUint; }
|
||||||
|
bool pinsScUintBool() const { return m_pinsScUintBool; }
|
||||||
bool pinsScBigUint() const { return m_pinsScBigUint; }
|
bool pinsScBigUint() const { return m_pinsScBigUint; }
|
||||||
bool pinsUint8() const { return m_pinsUint8; }
|
bool pinsUint8() const { return m_pinsUint8; }
|
||||||
bool ppComments() const { return m_ppComments; }
|
bool ppComments() const { return m_ppComments; }
|
||||||
|
@ -967,6 +967,10 @@ sub _compile_vlt_flags {
|
|||||||
my $checkflags = $self->_checkflags(%param);
|
my $checkflags = $self->_checkflags(%param);
|
||||||
die "%Error: specify threads via 'threads =>' argument, not as a command line option" unless ($checkflags !~ /(^|\s)-?-threads\s/);
|
die "%Error: specify threads via 'threads =>' argument, not as a command line option" unless ($checkflags !~ /(^|\s)-?-threads\s/);
|
||||||
$self->{coverage} = 1 if ($checkflags =~ /-coverage\b/);
|
$self->{coverage} = 1 if ($checkflags =~ /-coverage\b/);
|
||||||
|
my $driver_verilator_flags = ' '.join(' ', driver_verilator_flags()).' ';
|
||||||
|
$self->{pins_sc_uint_bool} = 1 if (($checkflags =~ /-pins-sc-uint-bool\b/)
|
||||||
|
|| ($driver_verilator_flags =~ /-pins-sc-uint-bool\b/));
|
||||||
|
|
||||||
$self->{savable} = 1 if ($checkflags =~ /-savable\b/);
|
$self->{savable} = 1 if ($checkflags =~ /-savable\b/);
|
||||||
$self->{sc} = 1 if ($checkflags =~ /-sc\b/);
|
$self->{sc} = 1 if ($checkflags =~ /-sc\b/);
|
||||||
$self->{timing} = 1 if ($checkflags =~ / -?-timing\b/ || $checkflags =~ / -?-binary\b/ );
|
$self->{timing} = 1 if ($checkflags =~ / -?-timing\b/ || $checkflags =~ / -?-binary\b/ );
|
||||||
@ -1963,8 +1967,13 @@ sub _make_main {
|
|||||||
if ($self->sc) {
|
if ($self->sc) {
|
||||||
print $fh "extern int sc_main(int argc, char** argv);\n";
|
print $fh "extern int sc_main(int argc, char** argv);\n";
|
||||||
print $fh "int sc_main(int argc, char** argv) {\n";
|
print $fh "int sc_main(int argc, char** argv) {\n";
|
||||||
print $fh " sc_signal<bool> fastclk;\n" if $self->{inputs}{fastclk};
|
if ($self->{pins_sc_uint_bool}) {
|
||||||
print $fh " sc_signal<bool> clk;\n" if $self->{inputs}{clk};
|
print $fh " sc_signal<sc_dt::sc_uint<1>> fastclk;\n" if $self->{inputs}{fastclk};
|
||||||
|
print $fh " sc_signal<sc_dt::sc_uint<1>> clk;\n" if $self->{inputs}{clk};
|
||||||
|
} else {
|
||||||
|
print $fh " sc_signal<bool> fastclk;\n" if $self->{inputs}{fastclk};
|
||||||
|
print $fh " sc_signal<bool> clk;\n" if $self->{inputs}{clk};
|
||||||
|
}
|
||||||
print $fh " sc_set_time_resolution(1, $Self->{sc_time_resolution});\n";
|
print $fh " sc_set_time_resolution(1, $Self->{sc_time_resolution});\n";
|
||||||
print $fh " sc_time sim_time($self->{sim_time}, $Self->{sc_time_resolution});\n";
|
print $fh " sc_time sim_time($self->{sim_time}, $Self->{sc_time_resolution});\n";
|
||||||
} else {
|
} else {
|
||||||
@ -2059,7 +2068,11 @@ sub _make_main {
|
|||||||
print $fh " topp->nextTimeSlot() / MAIN_TIME_MULTIPLIER <= cycles) {\n";
|
print $fh " topp->nextTimeSlot() / MAIN_TIME_MULTIPLIER <= cycles) {\n";
|
||||||
print $fh " new_time = topp->nextTimeSlot();\n";
|
print $fh " new_time = topp->nextTimeSlot();\n";
|
||||||
print $fh " } else {\n";
|
print $fh " } else {\n";
|
||||||
print $fh " ${set}clk = !${set}clk;\n";
|
if ($self->{pins_sc_uint_bool}) {
|
||||||
|
print $fh " ${set}clk.write(!${set}clk.read());\n";
|
||||||
|
} else {
|
||||||
|
print $fh " ${set}clk = !${set}clk;\n";
|
||||||
|
}
|
||||||
print $fh " }\n";
|
print $fh " }\n";
|
||||||
print $fh " contextp->time(new_time);\n";
|
print $fh " contextp->time(new_time);\n";
|
||||||
} else {
|
} else {
|
||||||
@ -2070,11 +2083,19 @@ sub _make_main {
|
|||||||
for (my $i = 0; $i < 5; $i++) {
|
for (my $i = 0; $i < 5; $i++) {
|
||||||
my $action = 0;
|
my $action = 0;
|
||||||
if ($self->{inputs}{fastclk}) {
|
if ($self->{inputs}{fastclk}) {
|
||||||
print $fh " ${set}fastclk = !${set}fastclk;\n";
|
if ($self->{pins_sc_uint_bool}) {
|
||||||
|
print $fh " ${set}fastclk.write(!${set}fastclk.read());\n";
|
||||||
|
} else {
|
||||||
|
print $fh " ${set}fastclk = !${set}fastclk;\n";
|
||||||
|
}
|
||||||
$action = 1;
|
$action = 1;
|
||||||
}
|
}
|
||||||
if ($i == 0 && $self->{inputs}{clk}) {
|
if ($i == 0 && $self->{inputs}{clk}) {
|
||||||
print $fh " ${set}clk = !${set}clk;\n";
|
if ($self->{pins_sc_uint_bool}) {
|
||||||
|
print $fh " ${set}clk.write(!${set}clk.read());\n";
|
||||||
|
} else {
|
||||||
|
print $fh " ${set}clk = !${set}clk;\n";
|
||||||
|
}
|
||||||
$action = 1;
|
$action = 1;
|
||||||
}
|
}
|
||||||
if ($self->{savable}) {
|
if ($self->{savable}) {
|
||||||
@ -2206,10 +2227,18 @@ sub _make_top_v {
|
|||||||
print $fh " fastclk = 1;\n" if $self->{inputs}{fastclk};
|
print $fh " fastclk = 1;\n" if $self->{inputs}{fastclk};
|
||||||
print $fh " clk = 1;\n" if $self->{inputs}{clk};
|
print $fh " clk = 1;\n" if $self->{inputs}{clk};
|
||||||
print $fh " while (\$time < $self->{sim_time}) begin\n";
|
print $fh " while (\$time < $self->{sim_time}) begin\n";
|
||||||
for (my $i = 0; $i < 5; $i++) {
|
if ($self->{pins_sc_uint_bool}) {
|
||||||
print $fh " #1;\n";
|
for (my $i = 0; $i < 5; $i++) {
|
||||||
print $fh " fastclk = !fastclk;\n" if $self->{inputs}{fastclk};
|
print $fh " #1;\n";
|
||||||
print $fh " clk = !clk;\n" if $i == 4 && $self->{inputs}{clk};
|
print $fh " fastclk.write(!fastclk.read());\n" if $self->{inputs}{fastclk};
|
||||||
|
print $fh " clk.write(!clk.read());\n" if $i == 4 && $self->{inputs}{clk};
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (my $i = 0; $i < 5; $i++) {
|
||||||
|
print $fh " #1;\n";
|
||||||
|
print $fh " fastclk = !fastclk;\n" if $self->{inputs}{fastclk};
|
||||||
|
print $fh " clk = !clk;\n" if $i == 4 && $self->{inputs}{clk};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
print $fh " end\n";
|
print $fh " end\n";
|
||||||
print $fh " end\n";
|
print $fh " end\n";
|
||||||
|
43
test_regress/t/t_var_pins_sc_uint_bool.pl
Executable file
43
test_regress/t/t_var_pins_sc_uint_bool.pl
Executable file
@ -0,0 +1,43 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2024 by Wilson Snyder. This program is free software; you
|
||||||
|
# can redistribute it and/or modify it under the terms of either the GNU
|
||||||
|
# Lesser General Public License Version 3 or the Perl Artistic License
|
||||||
|
# Version 2.0.
|
||||||
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
scenarios(vlt => 1);
|
||||||
|
|
||||||
|
top_filename("t/t_var_pinsizes.v");
|
||||||
|
|
||||||
|
compile(
|
||||||
|
verilator_flags2 => ["-sc --pins-sc-uint-bool --trace --exe $Self->{t_dir}/t_var_pinsizes.cpp"],
|
||||||
|
make_main => 0,
|
||||||
|
);
|
||||||
|
|
||||||
|
{
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<sc_dt::sc_uint<1>\s> \s+ &i1;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<uint32_t> \s+ &i8;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<uint32_t> \s+ &i16;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<uint32_t> \s+ &i32;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<uint64_t> \s+ &i64;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<sc_dt::sc_bv<65>\s> \s+ &i65;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<sc_dt::sc_bv<128>\s> \s+ &i128;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_in<sc_dt::sc_bv<513>\s> \s+ &i513;/x);
|
||||||
|
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<sc_dt::sc_uint<1>\s> \s+ &o1;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<uint32_t> \s+ &o8;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<uint32_t> \s+ &o16;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<uint32_t> \s+ &o32;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<uint64_t> \s+ &o64;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<sc_dt::sc_bv<65>\s> \s+ &o65;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<sc_dt::sc_bv<128>\s> \s+ &o128;/x);
|
||||||
|
file_grep("$Self->{obj_dir}/$Self->{vm_prefix}.h", qr/sc_core::sc_out<sc_dt::sc_bv<513>\s> \s+ &o513;/x);
|
||||||
|
}
|
||||||
|
|
||||||
|
execute();
|
||||||
|
|
||||||
|
ok(1);
|
||||||
|
1;
|
30
test_regress/t/t_var_pins_sc_uint_bool_nomain.pl
Executable file
30
test_regress/t/t_var_pins_sc_uint_bool_nomain.pl
Executable file
@ -0,0 +1,30 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2024 by Wilson Snyder. This program is free software; you
|
||||||
|
# can redistribute it and/or modify it under the terms of either the GNU
|
||||||
|
# Lesser General Public License Version 3 or the Perl Artistic License
|
||||||
|
# Version 2.0.
|
||||||
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
# This test runs the very first time we've executed Verilator --sc
|
||||||
|
# after building so we make sure to run with --gdbbt, so if it dumps we'll
|
||||||
|
# get a trace.
|
||||||
|
|
||||||
|
scenarios(simulator => 1);
|
||||||
|
|
||||||
|
top_filename("t/t_a1_first_cc.v");
|
||||||
|
|
||||||
|
$DEBUG_QUIET = "--debug --debugi 0 --gdbbt --no-dump-tree";
|
||||||
|
|
||||||
|
compile(
|
||||||
|
verilator_flags2 => [$DEBUG_QUIET, "-sc --trace --pins-sc-uint-bool"],
|
||||||
|
);
|
||||||
|
|
||||||
|
execute(
|
||||||
|
check_finished => 1,
|
||||||
|
);
|
||||||
|
|
||||||
|
ok(1);
|
||||||
|
1;
|
Loading…
Reference in New Issue
Block a user