Fix --x-assign to be independent from +verilator+rand+reset (#5214)

This commit is contained in:
Andrew Nolte 2024-07-14 17:04:00 -04:00 committed by GitHub
parent dc037db1cd
commit 60f9e21d8c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
6 changed files with 50 additions and 8 deletions

View File

@ -412,6 +412,8 @@ IData VL_RAND_RESET_I(int obits) VL_MT_SAFE {
data &= VL_MASK_I(obits); data &= VL_MASK_I(obits);
return data; return data;
} }
IData VL_RAND_RESET_ASSIGN_I(int obits) VL_MT_SAFE { return VL_RANDOM_I() & VL_MASK_I(obits); }
QData VL_RAND_RESET_Q(int obits) VL_MT_SAFE { QData VL_RAND_RESET_Q(int obits) VL_MT_SAFE {
if (Verilated::threadContextp()->randReset() == 0) return 0; if (Verilated::threadContextp()->randReset() == 0) return 0;
QData data = ~0ULL; QData data = ~0ULL;
@ -421,11 +423,19 @@ QData VL_RAND_RESET_Q(int obits) VL_MT_SAFE {
data &= VL_MASK_Q(obits); data &= VL_MASK_Q(obits);
return data; return data;
} }
QData VL_RAND_RESET_ASSIGN_Q(int obits) VL_MT_SAFE { return VL_RANDOM_Q() & VL_MASK_Q(obits); }
WDataOutP VL_RAND_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE { WDataOutP VL_RAND_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE {
for (int i = 0; i < VL_WORDS_I(obits) - 1; ++i) outwp[i] = VL_RAND_RESET_I(32); for (int i = 0; i < VL_WORDS_I(obits) - 1; ++i) outwp[i] = VL_RAND_RESET_I(32);
outwp[VL_WORDS_I(obits) - 1] = VL_RAND_RESET_I(32) & VL_MASK_E(obits); outwp[VL_WORDS_I(obits) - 1] = VL_RAND_RESET_I(32) & VL_MASK_E(obits);
return outwp; return outwp;
} }
WDataOutP VL_RAND_RESET_ASSIGN_W(int obits, WDataOutP outwp) VL_MT_SAFE {
for (int i = 0; i < VL_WORDS_I(obits) - 1; ++i) outwp[i] = VL_RAND_RESET_ASSIGN_I(32);
outwp[VL_WORDS_I(obits) - 1] = VL_RAND_RESET_ASSIGN_I(32) & VL_MASK_E(obits);
return outwp;
}
WDataOutP VL_ZERO_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE { WDataOutP VL_ZERO_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE {
// Not inlined to speed up compilation of slowpath code // Not inlined to speed up compilation of slowpath code
return VL_ZERO_W(obits, outwp); return VL_ZERO_W(obits, outwp);

View File

@ -106,6 +106,14 @@ extern IData VL_RAND_RESET_I(int obits) VL_MT_SAFE;
extern QData VL_RAND_RESET_Q(int obits) VL_MT_SAFE; extern QData VL_RAND_RESET_Q(int obits) VL_MT_SAFE;
/// Random reset a signal of given width (init time only) /// Random reset a signal of given width (init time only)
extern WDataOutP VL_RAND_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE; extern WDataOutP VL_RAND_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE;
/// Random reset a signal of given width (assign time only)
extern IData VL_RAND_RESET_ASSIGN_I(int obits) VL_MT_SAFE;
/// Random reset a signal of given width (assign time only)
extern QData VL_RAND_RESET_ASSIGN_Q(int obits) VL_MT_SAFE;
/// Random reset a signal of given width (assign time only)
extern WDataOutP VL_RAND_RESET_ASSIGN_W(int obits, WDataOutP outwp) VL_MT_SAFE;
/// Zero reset a signal (slow - else use VL_ZERO_W) /// Zero reset a signal (slow - else use VL_ZERO_W)
extern WDataOutP VL_ZERO_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE; extern WDataOutP VL_ZERO_RESET_W(int obits, WDataOutP outwp) VL_MT_SAFE;

View File

@ -1740,13 +1740,27 @@ public:
: (m_urandom ? "%f$urandom()" : "%f$random()"); : (m_urandom ? "%f$urandom()" : "%f$random()");
} }
string emitC() override { string emitC() override {
return m_reset ? "VL_RAND_RESET_%nq(%nw, %P)" if (m_reset) {
: seedp() if (v3Global.opt.xAssign() == "unique") {
// cppcheck-has-bug-suppress knownConditionTrueFalse return "VL_RAND_RESET_ASSIGN_%nq(%nw, %P)";
? (urandom() ? "VL_URANDOM_SEEDED_%nq%lq(%li)" // } else {
: "VL_RANDOM_SEEDED_%nq%lq(%li)") // This follows xInitial randomization
: (isWide() ? "VL_RANDOM_%nq(%nw, %P)" // return "VL_RAND_RESET_%nq(%nw, %P)";
: "VL_RANDOM_%nq()"); }
}
if (seedp()) {
if (urandom()) {
return "VL_URANDOM_SEEDED_%nq%lq(%li)";
} else {
return "VL_RANDOM_SEEDED_%nq%lq(%li)";
}
}
if (isWide()) {
return "VL_RANDOM_%nq(%nw, %P)";
} else {
return "VL_RANDOM_%nq()";
}
} }
bool cleanOut() const override { return false; } bool cleanOut() const override { return false; }
bool isGateOptimizable() const override { return false; } bool isGateOptimizable() const override { return false; }

View File

@ -13,6 +13,7 @@ scenarios(simulator => 1);
$Self->{verilated_randReset} = 1; # allow checking if we initialize vars to zero only when needed $Self->{verilated_randReset} = 1; # allow checking if we initialize vars to zero only when needed
compile( compile(
verilator_flags2 => ["--x-assign 1"],
); );
execute( execute(

View File

@ -45,10 +45,17 @@ int main(int argc, const char** argv) {
top->clk = 1; top->clk = 1;
top->eval(); top->eval();
#if defined(T_X_ASSIGN_UNIQUE_0) || defined(T_X_ASSIGN_UNIQUE_1)
if (top->o_int == 0 || top->o_int == -1) {
vl_fatal(__FILE__, __LINE__, "TOP.t", "x assign was not unique");
exit(1);
}
#else
if (top->o != EXPECTED) { if (top->o != EXPECTED) {
vl_fatal(__FILE__, __LINE__, "TOP.t", "incorrect module output"); vl_fatal(__FILE__, __LINE__, "TOP.t", "incorrect module output");
exit(1); exit(1);
} }
#endif
VL_DO_DANGLING(delete top, top); VL_DO_DANGLING(delete top, top);
std::cout << "*-* All Finished *-*" << std::endl; std::cout << "*-* All Finished *-*" << std::endl;

View File

@ -8,9 +8,11 @@
module t_x_assign( module t_x_assign(
input wire clk, input wire clk,
output reg o output reg o,
output reg[31:0] o_int
); );
always @(posedge clk) begin always @(posedge clk) begin
if (1'bx) o <= 1'd1; else o <= 1'd0; if (1'bx) o <= 1'd1; else o <= 1'd0;
o_int <= 'x;
end end
endmodule endmodule