From 66e4656a8ebb1d001e3a801e701c61599d917efa Mon Sep 17 00:00:00 2001 From: Kamil Rakoczy Date: Thu, 16 Mar 2023 12:12:54 +0100 Subject: [PATCH] Fix event controls reusing same variable (#4014) --- src/V3SenExprBuilder.h | 8 +- .../t/t_event_control_prev_name_collision.pl | 22 +++++ .../t/t_event_control_prev_name_collision.v | 98 +++++++++++++++++++ 3 files changed, 126 insertions(+), 2 deletions(-) create mode 100755 test_regress/t/t_event_control_prev_name_collision.pl create mode 100644 test_regress/t/t_event_control_prev_name_collision.v diff --git a/src/V3SenExprBuilder.h b/src/V3SenExprBuilder.h index 7188b22fd..6b70e7d23 100644 --- a/src/V3SenExprBuilder.h +++ b/src/V3SenExprBuilder.h @@ -93,8 +93,12 @@ class SenExprBuilder final { FileLine* const flp = exprp->fileline(); const auto rdCurr = [=]() { return getCurr(exprp); }; + AstNode* scopeExprp = exprp; + if (AstVarRef* const refp = VN_CAST(exprp, VarRef)) { + scopeExprp = refp->varScopep()->varp(); + } // Create the 'previous value' variable - auto it = m_prev.find(*exprp); + auto it = m_prev.find(*scopeExprp); if (it == m_prev.end()) { // For readability, use the scoped signal name if the trigger is a simple AstVarRef string name; @@ -117,7 +121,7 @@ class SenExprBuilder final { prevp = new AstVarScope{flp, m_scopep, varp}; m_scopep->addVarsp(prevp); } - it = m_prev.emplace(*exprp, prevp).first; + it = m_prev.emplace(*scopeExprp, prevp).first; // Add the initializer init AstAssign* const initp = new AstAssign{flp, new AstVarRef{flp, prevp, VAccess::WRITE}, diff --git a/test_regress/t/t_event_control_prev_name_collision.pl b/test_regress/t/t_event_control_prev_name_collision.pl new file mode 100755 index 000000000..0fa78eb28 --- /dev/null +++ b/test_regress/t/t_event_control_prev_name_collision.pl @@ -0,0 +1,22 @@ +#!/usr/bin/env perl +if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; } +# DESCRIPTION: Verilator: Verilog Test driver/expect definition +# +# Copyright 2003 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(simulator => 1); + +compile( + verilator_flags2 => ["-fno-inline"], + ); + +execute( + check_finished => 1, + ); + +ok(1); +1; diff --git a/test_regress/t/t_event_control_prev_name_collision.v b/test_regress/t/t_event_control_prev_name_collision.v new file mode 100644 index 000000000..2b95b86c5 --- /dev/null +++ b/test_regress/t/t_event_control_prev_name_collision.v @@ -0,0 +1,98 @@ +// DESCRIPTION: Verilator: Verilog Test module +// +// This file ONLY is placed under the Creative Commons Public Domain, for +// any use, without warranty, 2023 by Antmicro Ltd. +// SPDX-License-Identifier: CC0-1.0 +module S( + input reset, + io_i, + output io_o +); + reg s; + always @(posedge reset) begin + if (reset) begin + s <= 1'h0; + end + else begin + s <= io_i; + end + end + assign io_o = s; +endmodule + +module Q( + input reset_e, + input reset_d, + output ready_e +); + + wire reset_n; + wire io_v; + wire io_e; + S e ( + .io_i (), + .reset (reset_e | ~reset_n), + .io_o (io_e) + ); + S v ( + .io_i (io_e), + .reset (reset_e), + .io_o (io_v) + ); + assign reset_n = ~reset_d; + assign ready_e = io_v; +endmodule + +module Test( + input reset, + output valid +); + wire ready_e; + + Q q ( + .reset_e (reset), + .reset_d (reset), + .ready_e (ready_e) + ); + + assign valid = ready_e; +endmodule + +module Test2( + input reset, + input valid +); + always begin + if (~reset & valid) begin + $fatal; + end + end +endmodule + +module Dut( + input reset +); + wire valid_g; + + Test t ( + .reset (reset), + .valid (valid_g) + ); + Test2 t2 ( + .reset (reset), + .valid (valid_g) + ); +endmodule + +module t (/*AUTOARG*/ + ); + reg [$bits(dut.reset)-1:0] reset; + + Dut dut ( + .reset(reset) + ); + initial begin + $write("*-* All Finished *-*\n"); + $finish; + end +endmodule