Fix for Returning an object of the wrong type from a static function of a parameterized class (#5479) (#7387)

This commit is contained in:
em2machine 2026-04-11 07:49:45 -04:00 committed by GitHub
parent 529a6b7f93
commit dc33e8bb18
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 334 additions and 11 deletions

View File

@ -5588,7 +5588,12 @@ class LinkDotResolveVisitor final : public VNVisitor {
AstNodeDType* const unwrappedp = typedefp->subDTypep()->skipRefp();
if (AstClassRefDType* const classRefp = VN_CAST(unwrappedp, ClassRefDType)) {
AstPin* paramsp = cpackagerefp->paramsp();
if (paramsp) {
if (!paramsp && classRefp->paramsp()) {
// No explicit #(...) on extends clause; carry over the
// typedef's type-parameter pins so V3Param sees them.
paramsp = classRefp->paramsp()->cloneTree(true);
nodep->parameterized(true);
} else if (paramsp) {
paramsp = paramsp->cloneTree(true);
nodep->parameterized(true);
}

View File

@ -1295,8 +1295,8 @@ class ParamProcessor final {
return isEq.isNeqZero();
}
void cellPinCleanup(AstNode* nodep, AstPin* pinp, AstNodeModule* srcModp, string& longnamer,
bool& any_overridesr) {
void cellPinCleanup(AstNode* nodep, AstPin* pinp, AstPin* paramsp, AstNodeModule* srcModp,
string& longnamer, bool& any_overridesr) {
if (!pinp->exprp()) return; // No-connect
if (AstVar* const modvarp = pinp->modVarp()) {
if (!modvarp->isGParam()) {
@ -1336,6 +1336,48 @@ class ParamProcessor final {
}
AstConst* const exprp = VN_CAST(pinp->exprp(), Const);
AstConst* const origp = VN_CAST(modvarp->valuep(), Const);
// Width the pin value to match the port type so that the same
// logical value always produces the same specialization name.
AstConst* normedNamep = nullptr;
if (exprp && !exprp->num().isDouble() && !exprp->num().isString()) {
AstVar* cloneVarp = modvarp->cloneTree(false);
if (AstNode* const oldValuep = cloneVarp->valuep()) {
oldValuep->unlinkFrBack();
VL_DO_DANGLING(oldValuep->deleteTree(), oldValuep);
}
cloneVarp->valuep(exprp->cloneTree(false));
// Clone the dtype and resolve VarRefs to other parameters
// with their already-constified pin values (e.g., N-1 in
// logic [N-1:0] becomes Const-1 which can be folded).
if (AstNodeDType* const origDTypep = modvarp->subDTypep()) {
AstNodeDType* const dtypeClonep = origDTypep->cloneTree(false);
dtypeClonep->foreach([&](AstVarRef* varrefp) {
for (AstPin* pp = paramsp; pp; pp = VN_AS(pp->nextp(), Pin)) {
if (pp->modVarp() == varrefp->varp()) {
if (AstConst* const constp = VN_CAST(pp->exprp(), Const)) {
varrefp->replaceWith(constp->cloneTree(false));
VL_DO_DANGLING(varrefp->deleteTree(), varrefp);
}
break;
}
}
});
if (cloneVarp->childDTypep())
cloneVarp->childDTypep()->unlinkFrBack()->deleteTree();
cloneVarp->childDTypep(dtypeClonep);
cloneVarp->dtypep(nullptr);
}
V3Const::constifyParamsEdit(cloneVarp);
if (AstConst* const widthedp = VN_CAST(cloneVarp->valuep(), Const)) {
// Set the constant's dtype to the port's widthed type
// so identical values hash the same in paramValueNumber.
if (cloneVarp->dtypep()) widthedp->dtypep(cloneVarp->dtypep());
widthedp->unlinkFrBack();
normedNamep = widthedp;
}
VL_DO_DANGLING(cloneVarp->deleteTree(), cloneVarp);
}
AstConst* const namingExprp = normedNamep ? normedNamep : exprp;
if (!exprp) {
// With DepGraph architecture, all expressions should be constants
// by the time V3Param runs. If not, it's an error.
@ -1356,16 +1398,18 @@ class ParamProcessor final {
// This prevents making additional modules, and makes coverage more
// obvious as it won't show up under a unique module page name.
UINFO(9, "cellPinCleanup: same as default " << pinp);
} else if (exprp->num().isDouble() || exprp->num().isString()
|| exprp->num().isFourState() || exprp->num().width() != 32) {
longnamer
+= ("_" + paramSmallName(srcModp, modvarp) + paramValueNumber(exprp));
} else if (namingExprp->num().isDouble() || namingExprp->num().isString()
|| namingExprp->num().isFourState()
|| namingExprp->num().width() != 32) {
longnamer += ("_" + paramSmallName(srcModp, modvarp)
+ paramValueNumber(namingExprp));
any_overridesr = true;
} else {
longnamer
+= ("_" + paramSmallName(srcModp, modvarp) + exprp->num().ascii(false));
longnamer += ("_" + paramSmallName(srcModp, modvarp)
+ namingExprp->num().ascii(false));
any_overridesr = true;
}
if (normedNamep) VL_DO_DANGLING(normedNamep->deleteTree(), normedNamep);
}
} else if (AstParamTypeDType* const modvarp = pinp->modPTypep()) {
// Handle DOT with ParseRef RHS (e.g., p_class#(8)::p_type)
@ -1681,7 +1725,8 @@ class ParamProcessor final {
any_overrides = longname != srcModp->name();
} else {
for (AstPin* pinp = paramsp; pinp; pinp = VN_AS(pinp->nextp(), Pin)) {
cellPinCleanup(nodep, pinp, srcModp, longname /*ref*/, any_overrides /*ref*/);
cellPinCleanup(nodep, pinp, paramsp, srcModp, longname /*ref*/,
any_overrides /*ref*/);
}
}
IfaceRefRefs ifaceRefRefs;

View File

@ -0,0 +1,18 @@
#!/usr/bin/env python3
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
#
# 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-FileCopyrightText: 2025 Wilson Snyder
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
import vltest_bootstrap
test.scenarios('simulator')
test.compile(verilator_flags2=["--binary"])
test.execute()
test.passes()

View File

@ -0,0 +1,72 @@
// Reproduction: type parameter lost through typedef extends
//
// When a non-parameterized class extends a typedef of a parameterized class,
// the inner member's type parameter falls back to the default instead of
// inheriting from the typedef's specialization.
//
// 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-FileCopyrightText: 2025 Wilson Snyder
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
module t;
class my_item;
int x;
endclass
// Inner parameterized classes
virtual class if_base #(type T1 = int, type T2 = T1);
endclass
virtual class port_base #(type IF = if_base#());
function void connect(port_base #(IF) other);
endfunction
endclass
class pull_port #(type REQ = int, type RSP = REQ)
extends port_base #(if_base #(REQ, RSP));
endclass
class pull_imp #(type REQ = int, type RSP = REQ, type IMP = int)
extends port_base #(if_base #(REQ, RSP));
endclass
// Outer parameterized class with member using the type param
class driver #(type REQ = int, type RSP = REQ);
pull_port #(REQ, RSP) seq_port;
function new;
seq_port = new;
endfunction
endclass
// Outer parameterized class on the other side
class sequencer #(type REQ = int, type RSP = REQ);
pull_imp #(REQ, RSP, sequencer #(REQ, RSP)) seq_export;
function new;
seq_export = new;
endfunction
endclass
// Typedef specialization + extends
typedef driver #(my_item) my_driver;
class low_driver extends my_driver;
function new;
super.new();
endfunction
endclass
initial begin
sequencer #(my_item) sqr;
low_driver drv;
sqr = new;
drv = new;
drv.seq_port.connect(sqr.seq_export);
$write("*-* All Finished *-*\n");
$finish;
end
endmodule

View File

@ -1,5 +1,4 @@
%Warning-WIDTHTRUNC: t/t_param_width_loc_bad.v:19:21: Operator VAR 'PARAM' expects 1 bits on the Initial value, but Initial value's CONST '32'h1' generates 32 bits.
: ... note: In instance 't.test_i'
19 | parameter logic PARAM = 1'b0
| ^~~~~
... For warning description see https://verilator.org/warn/WIDTHTRUNC?v=latest

View File

@ -0,0 +1,18 @@
#!/usr/bin/env python3
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
#
# 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-FileCopyrightText: 2026 Wilson Snyder
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
import vltest_bootstrap
test.scenarios('simulator')
test.compile(verilator_flags2=["--binary"])
test.execute()
test.passes()

View File

@ -0,0 +1,26 @@
// DESCRIPTION: Verilator: Verify that a parameterized class's static function
// return type resolves to the same specialization as the caller's context.
//
// 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-FileCopyrightText: 2026 Wilson Snyder
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
module t;
class cls #(bit T = 1);
static function cls#(T) f();
cls#(T) c = new();
return c;
endfunction
endclass
initial begin
static cls#(0) c = cls#(0)::f();
if (c == null) $stop;
$write("*-* All Finished *-*\n");
$finish;
end
endmodule

View File

@ -0,0 +1,18 @@
#!/usr/bin/env python3
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
#
# 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-FileCopyrightText: 2026 Wilson Snyder
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
import vltest_bootstrap
test.scenarios('simulator')
test.compile(verilator_flags2=["--binary"])
test.execute()
test.passes()

View File

@ -0,0 +1,30 @@
// DESCRIPTION: Verilator: Verify parameterized class self-referential return
// types work with explicit-range value parameters (e.g. logic [7:0]).
//
// This exposes a gap where the declared type has a Range child not yet folded
// by V3Width, so the targeted keyword-width normalization cannot determine the
// port width.
//
// 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-FileCopyrightText: 2026 Wilson Snyder
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
module t;
class cls #(logic [7:0] T = 8'd1);
static function cls#(T) f();
cls#(T) c = new();
return c;
endfunction
endclass
initial begin
static cls#(8'd0) c = cls#(8'd0)::f();
if (c == null) $stop;
$write("*-* All Finished *-*\n");
$finish;
end
endmodule

View File

@ -0,0 +1,18 @@
#!/usr/bin/env python3
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
#
# 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-FileCopyrightText: 2026 Wilson Snyder
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
import vltest_bootstrap
test.scenarios('simulator')
test.compile(verilator_flags2=["--binary"])
test.execute()
test.passes()

View File

@ -0,0 +1,74 @@
// DESCRIPTION: Verilator: Verify parameterized class self-referential return
// types across a variety of value-parameter types (keyword, ranged, signed).
//
// 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-FileCopyrightText: 2026 Wilson Snyder
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
module t;
// Keyword type: bit (1-bit unsigned)
class cls_bit #(bit P = 1);
static function cls_bit#(P) f();
cls_bit#(P) c = new();
return c;
endfunction
endclass
// Keyword type: byte (8-bit signed)
class cls_byte #(byte P = 8'd1);
static function cls_byte#(P) f();
cls_byte#(P) c = new();
return c;
endfunction
endclass
// Keyword type: shortint (16-bit signed)
class cls_shortint #(shortint P = 16'd1);
static function cls_shortint#(P) f();
cls_shortint#(P) c = new();
return c;
endfunction
endclass
// Keyword type: integer (32-bit signed)
class cls_integer #(integer P = 1);
static function cls_integer#(P) f();
cls_integer#(P) c = new();
return c;
endfunction
endclass
// Explicit range: logic [15:0] (16-bit unsigned)
class cls_logic16 #(logic [15:0] P = 16'd1);
static function cls_logic16#(P) f();
cls_logic16#(P) c = new();
return c;
endfunction
endclass
// Explicit range: logic [31:0] (32-bit unsigned)
class cls_logic32 #(logic [31:0] P = 1);
static function cls_logic32#(P) f();
cls_logic32#(P) c = new();
return c;
endfunction
endclass
initial begin
static cls_bit#(0) c1 = cls_bit#(0)::f();
static cls_byte#(8'd0) c2 = cls_byte#(8'd0)::f();
static cls_shortint#(16'd0) c3 = cls_shortint#(16'd0)::f();
static cls_integer#(0) c4 = cls_integer#(0)::f();
static cls_logic16#(16'd0) c5 = cls_logic16#(16'd0)::f();
static cls_logic32#(0) c6 = cls_logic32#(0)::f();
if (c1 == null || c2 == null || c3 == null
|| c4 == null || c5 == null || c6 == null) $stop;
$write("*-* All Finished *-*\n");
$finish;
end
endmodule