Support parameter access from arrays of interfaces, #2155.

This commit is contained in:
Todd Strader 2020-02-05 16:51:33 -05:00
parent 609a5dc26d
commit 77f1b3eef4
4 changed files with 44 additions and 8 deletions

View File

@ -41,6 +41,8 @@ The contributors that suggested a given feature are shown in []. Thanks!
**** Fix WIDTH warning on </<= of narrower value, #2141. [agrobman] **** Fix WIDTH warning on </<= of narrower value, #2141. [agrobman]
**** Support parameter access from arrays of interfaces, #2155. [Todd Strader]
* Verilator 4.026 2020-01-11 * Verilator 4.026 2020-01-11

View File

@ -2070,11 +2070,21 @@ private:
refp->varp(varp); refp->varp(varp);
m_ds.m_dotText = ""; m_ds.m_dotText = "";
if (m_ds.m_unresolved && m_ds.m_unlinkedScope) { if (m_ds.m_unresolved && m_ds.m_unlinkedScope) {
newp = new AstUnlinkedRef(nodep->fileline(), VN_CAST(refp, VarXRef), string dotted = refp->dotted();
size_t pos = dotted.find("__BRA__??__KET__");
// Arrays of interfaces all have the same parameters
if (pos != string::npos && varp->isParam()
&& VN_IS(m_ds.m_unlinkedScope, CellArrayRef)) {
refp->dotted(dotted.substr(0, pos));
newp = refp;
} else {
newp = new AstUnlinkedRef(nodep->fileline(),
VN_CAST(refp, VarXRef),
refp->name(), refp->name(),
m_ds.m_unlinkedScope->unlinkFrBack()); m_ds.m_unlinkedScope->unlinkFrBack());
m_ds.m_unlinkedScope = NULL; m_ds.m_unlinkedScope = NULL;
m_ds.m_unresolved = false; m_ds.m_unresolved = false;
}
} else { } else {
newp = refp; newp = refp;
} }

View File

@ -355,9 +355,16 @@ private:
if (VN_IS(backp, Var) if (VN_IS(backp, Var)
&& VN_CAST(backp, Var)->isIfaceRef() && VN_CAST(backp, Var)->isIfaceRef()
&& VN_CAST(backp, Var)->childDTypep() && VN_CAST(backp, Var)->childDTypep()
&& VN_CAST(VN_CAST(backp, Var)->childDTypep(), IfaceRefDType)) { && (VN_CAST(VN_CAST(backp, Var)->childDTypep(), IfaceRefDType)
|| (VN_CAST(VN_CAST(backp, Var)->childDTypep(), UnpackArrayDType)
&& VN_CAST(VN_CAST(backp, Var)->childDTypep()->getChildDTypep(),
IfaceRefDType)))) {
AstIfaceRefDType* ifacerefp AstIfaceRefDType* ifacerefp
= VN_CAST(VN_CAST(backp, Var)->childDTypep(), IfaceRefDType); = VN_CAST(VN_CAST(backp, Var)->childDTypep(), IfaceRefDType);
if (!ifacerefp) {
ifacerefp = VN_CAST(VN_CAST(backp, Var)->childDTypep()->getChildDTypep(),
IfaceRefDType);
}
// Interfaces passed in on the port map have ifaces // Interfaces passed in on the port map have ifaces
if (AstIface* ifacep = ifacerefp->ifacep()) { if (AstIface* ifacep = ifacerefp->ifacep()) {
if (dotted == backp->name()) { if (dotted == backp->name()) {

View File

@ -34,14 +34,17 @@ module t (/*AUTOARG*/
input clk; input clk;
test_if #( .FOO (identity(5)) ) the_interface (); test_if #( .FOO (identity(5)) ) the_interface ();
test_if #( .FOO (identity(7)) ) array_interface [1:0] ();
testmod testmod_i (.clk (clk), testmod testmod_i (.clk (clk),
.intf (the_interface), .intf (the_interface),
.intf_no_mp (the_interface) .intf_no_mp (the_interface),
.intf_array (array_interface)
); );
localparam THE_TOP_FOO = the_interface.FOO; localparam THE_TOP_FOO = the_interface.FOO;
localparam THE_TOP_FOO_BITS = $bits({the_interface.FOO, the_interface.FOO}); localparam THE_TOP_FOO_BITS = $bits({the_interface.FOO, the_interface.FOO});
localparam THE_ARRAY_FOO = array_interface[0].FOO;
initial begin initial begin
if (THE_TOP_FOO != 5) begin if (THE_TOP_FOO != 5) begin
@ -52,6 +55,10 @@ module t (/*AUTOARG*/
$display("%%Error: THE_TOP_FOO_BITS = %0d", THE_TOP_FOO_BITS); $display("%%Error: THE_TOP_FOO_BITS = %0d", THE_TOP_FOO_BITS);
$stop; $stop;
end end
if (THE_ARRAY_FOO != 7) begin
$display("%%Error: THE_ARRAY_FOO = %0d", THE_ARRAY_FOO);
$stop;
end
end end
endmodule endmodule
@ -61,11 +68,13 @@ module testmod
( (
input clk, input clk,
test_if.mp intf, test_if.mp intf,
test_if intf_no_mp test_if intf_no_mp,
test_if.mp intf_array [1:0]
); );
localparam THE_FOO = intf.FOO; localparam THE_FOO = intf.FOO;
localparam THE_OTHER_FOO = intf_no_mp.FOO; localparam THE_OTHER_FOO = intf_no_mp.FOO;
localparam THE_ARRAY_FOO = intf_array[0].FOO;
always @(posedge clk) begin always @(posedge clk) begin
if (THE_FOO != 5) begin if (THE_FOO != 5) begin
@ -76,6 +85,10 @@ module testmod
$display("%%Error: THE_OTHER_FOO = %0d", THE_OTHER_FOO); $display("%%Error: THE_OTHER_FOO = %0d", THE_OTHER_FOO);
$stop; $stop;
end end
if (THE_ARRAY_FOO != 7) begin
$display("%%Error: THE_ARRAY_FOO = %0d", THE_ARRAY_FOO);
$stop;
end
if (intf.FOO != 5) begin if (intf.FOO != 5) begin
$display("%%Error: intf.FOO = %0d", intf.FOO); $display("%%Error: intf.FOO = %0d", intf.FOO);
$stop; $stop;
@ -84,6 +97,10 @@ module testmod
$display("%%Error: intf_no_mp.FOO = %0d", intf_no_mp.FOO); $display("%%Error: intf_no_mp.FOO = %0d", intf_no_mp.FOO);
$stop; $stop;
end end
if (intf_array[0].FOO != 7) begin
$display("%%Error: intf_array[0].FOO = %0d", intf_array[0].FOO);
$stop;
end
// if (i.getFoo() != 5) begin // if (i.getFoo() != 5) begin
// $display("%%Error: i.getFoo() = %0d", i.getFoo()); // $display("%%Error: i.getFoo() = %0d", i.getFoo());
// $stop; // $stop;