forked from github/verilator
Better optimize large always block splitting, bug1244.
Signed-off-by: Wilson Snyder <wsnyder@wsnyder.org>
This commit is contained in:
parent
0a887c29f1
commit
ef3c7bb6a2
2
Changes
2
Changes
@ -8,6 +8,8 @@ The contributors that suggested a given feature are shown in []. Thanks!
|
||||
|
||||
** Fix internals to make null-pointer-check clean.
|
||||
|
||||
*** Better optimize large always block splitting, bug1244. [John Coiner]
|
||||
|
||||
**** Fix internals to avoid 'using namespace std'.
|
||||
|
||||
|
||||
|
@ -5614,6 +5614,14 @@ public:
|
||||
AstNode* bodysp() const { return op1p(); } // op1= expressions to print
|
||||
};
|
||||
|
||||
class AstSplitPlaceholder : public AstNode {
|
||||
public:
|
||||
// Dummy node used within V3Split; never exists outside of V3Split.
|
||||
AstSplitPlaceholder(FileLine* filelinep)
|
||||
: AstNode(filelinep) {}
|
||||
ASTNODE_NODE_FUNCS(SplitPlaceholder)
|
||||
};
|
||||
|
||||
//######################################################################
|
||||
// Right below top
|
||||
|
||||
|
830
src/V3Split.cpp
830
src/V3Split.cpp
File diff suppressed because it is too large
Load Diff
29
test_regress/t/t_alw_noreorder.pl
Normal file
29
test_regress/t/t_alw_noreorder.pl
Normal file
@ -0,0 +1,29 @@
|
||||
#!/usr/bin/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.
|
||||
|
||||
$Self->{vlt} or $Self->skip("Verilator only test");
|
||||
|
||||
top_filename("t/t_alw_reorder.v");
|
||||
compile (
|
||||
verilator_flags2 => ["--stats -Or"],
|
||||
);
|
||||
|
||||
file_grep ($Self->{stats}, qr/Optimizations, Split always\s+(\d+)/i, 0);
|
||||
# Here we should see some dly vars since reorder is disabled.
|
||||
# (Whereas our twin test, t_alw_reorder, should see no dly vars
|
||||
# since it enables the reorder step.)
|
||||
file_grep ("$Self->{obj_dir}/$Self->{VM_PREFIX}.cpp", qr/dly__t__DOT__v1/i);
|
||||
file_grep ("$Self->{obj_dir}/$Self->{VM_PREFIX}.cpp", qr/dly__t__DOT__v2/i);
|
||||
|
||||
execute (
|
||||
check_finished=>1,
|
||||
);
|
||||
|
||||
ok(1);
|
||||
1;
|
23
test_regress/t/t_alw_nosplit.pl
Normal file
23
test_regress/t/t_alw_nosplit.pl
Normal file
@ -0,0 +1,23 @@
|
||||
#!/usr/bin/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.
|
||||
|
||||
compile (
|
||||
verilator_flags2 => ["--stats"],
|
||||
);
|
||||
|
||||
if ($Self->{vlt}) {
|
||||
file_grep ($Self->{stats}, qr/Optimizations, Split always\s+(\d+)/i, 0);
|
||||
}
|
||||
|
||||
execute (
|
||||
check_finished=>1,
|
||||
);
|
||||
|
||||
ok(1);
|
||||
1;
|
131
test_regress/t/t_alw_nosplit.v
Normal file
131
test_regress/t/t_alw_nosplit.v
Normal file
@ -0,0 +1,131 @@
|
||||
// DESCRIPTION: Verilator: Verilog Test module
|
||||
//
|
||||
// This file ONLY is placed into the Public Domain, for any use,
|
||||
// without warranty, 2018 by Wilson Snyder.
|
||||
|
||||
module t (/*AUTOARG*/
|
||||
// Inputs
|
||||
clk
|
||||
);
|
||||
|
||||
input clk;
|
||||
integer cyc; initial cyc=1;
|
||||
|
||||
reg [15:0] m_din;
|
||||
|
||||
// We expect none of these blocks to split.
|
||||
// Blocks that can split should go in t_alw_split.v instead.
|
||||
|
||||
reg [15:0] b_split_1, b_split_2;
|
||||
always @ (/*AS*/m_din) begin
|
||||
b_split_1 = m_din;
|
||||
b_split_2 = b_split_1;
|
||||
end
|
||||
|
||||
reg [15:0] c_split_1, c_split_2;
|
||||
always @ (/*AS*/m_din) begin
|
||||
c_split_1 = m_din;
|
||||
c_split_2 = c_split_1;
|
||||
c_split_1 = ~m_din;
|
||||
end
|
||||
|
||||
always @ (posedge clk) begin
|
||||
$write(" foo %x", m_din);
|
||||
$write(" bar %x\n", m_din);
|
||||
end
|
||||
|
||||
reg [15:0] e_split_1, e_split_2;
|
||||
always @ (posedge clk) begin
|
||||
e_split_1 = m_din;
|
||||
e_split_2 = e_split_1;
|
||||
end
|
||||
|
||||
reg [15:0] f_split_1, f_split_2;
|
||||
always @ (posedge clk) begin
|
||||
f_split_2 = f_split_1;
|
||||
f_split_1 = m_din;
|
||||
end
|
||||
|
||||
reg [15:0] l_split_1, l_split_2;
|
||||
always @ (posedge clk) begin
|
||||
l_split_2 <= l_split_1;
|
||||
l_split_1 <= l_split_2 | m_din;
|
||||
end
|
||||
|
||||
reg [15:0] z_split_1, z_split_2;
|
||||
always @ (posedge clk) begin
|
||||
z_split_1 <= 0;
|
||||
z_split_1 <= ~m_din;
|
||||
end
|
||||
always @ (posedge clk) begin
|
||||
z_split_2 <= 0;
|
||||
z_split_2 <= z_split_1;
|
||||
end
|
||||
|
||||
reg [15:0] h_split_1;
|
||||
reg [15:0] h_split_2;
|
||||
reg [15:0] h_foo;
|
||||
always @ (posedge clk) begin
|
||||
// $write(" cyc = %x m_din = %x\n", cyc, m_din);
|
||||
h_foo = m_din;
|
||||
if (cyc > 2) begin
|
||||
// This conditional depends on non-primary-input foo.
|
||||
// Its dependency on foo should not be pruned. As a result,
|
||||
// the dependencies of h_split_1 and h_split_2 on this
|
||||
// conditional will also not be pruned, making them all
|
||||
// weakly connected such that they'll end up in the same graph
|
||||
// and we can't split.
|
||||
if (h_foo == 16'h0) begin
|
||||
h_split_1 <= 16'h0;
|
||||
h_split_2 <= 16'h0;
|
||||
end
|
||||
else begin
|
||||
h_split_1 <= m_din;
|
||||
h_split_2 <= ~m_din;
|
||||
end
|
||||
end
|
||||
else begin
|
||||
h_split_1 <= 16'h0;
|
||||
h_split_2 <= 16'h0;
|
||||
end
|
||||
end // always @ (posedge clk)
|
||||
|
||||
always @ (posedge clk) begin
|
||||
if (cyc!=0) begin
|
||||
cyc<=cyc+1;
|
||||
end
|
||||
if (cyc==1) begin
|
||||
m_din <= 16'hfeed;
|
||||
end
|
||||
if (cyc==4) begin
|
||||
m_din <= 16'he11e;
|
||||
if (!(b_split_1==16'hfeed && b_split_2==16'hfeed)) $stop;
|
||||
if (!(c_split_1==16'h0112 && c_split_2==16'hfeed)) $stop;
|
||||
if (!(e_split_1==16'hfeed && e_split_2==16'hfeed)) $stop;
|
||||
if (!(f_split_1==16'hfeed && f_split_2==16'hfeed)) $stop;
|
||||
if (!(z_split_1==16'h0112 && z_split_2==16'h0112)) $stop;
|
||||
end
|
||||
if (cyc==5) begin
|
||||
m_din <= 16'he22e;
|
||||
if (!(b_split_1==16'he11e && b_split_2==16'he11e)) $stop;
|
||||
if (!(c_split_1==16'h1ee1 && c_split_2==16'he11e)) $stop;
|
||||
// Two valid orderings, as we don't know which posedge clk gets evaled first
|
||||
if (!(e_split_1==16'hfeed && e_split_2==16'hfeed) && !(e_split_1==16'he11e && e_split_2==16'he11e)) $stop;
|
||||
if (!(f_split_1==16'hfeed && f_split_2==16'hfeed) && !(f_split_1==16'he11e && f_split_2==16'hfeed)) $stop;
|
||||
if (!(z_split_1==16'h0112 && z_split_2==16'h0112)) $stop;
|
||||
end
|
||||
if (cyc==6) begin
|
||||
m_din <= 16'he33e;
|
||||
if (!(b_split_1==16'he22e && b_split_2==16'he22e)) $stop;
|
||||
if (!(c_split_1==16'h1dd1 && c_split_2==16'he22e)) $stop;
|
||||
// Two valid orderings, as we don't know which posedge clk gets evaled first
|
||||
if (!(e_split_1==16'he11e && e_split_2==16'he11e) && !(e_split_1==16'he22e && e_split_2==16'he22e)) $stop;
|
||||
if (!(f_split_1==16'he11e && f_split_2==16'hfeed) && !(f_split_1==16'he22e && f_split_2==16'he11e)) $stop;
|
||||
if (!(z_split_1==16'h1ee1 && z_split_2==16'h0112)) $stop;
|
||||
end
|
||||
if (cyc==7) begin
|
||||
$write("*-* All Finished *-*\n");
|
||||
$finish;
|
||||
end
|
||||
end
|
||||
endmodule
|
32
test_regress/t/t_alw_reorder.pl
Normal file
32
test_regress/t/t_alw_reorder.pl
Normal file
@ -0,0 +1,32 @@
|
||||
#!/usr/bin/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.
|
||||
|
||||
$Self->{vlt} or $Self->skip("Verilator only test");
|
||||
|
||||
compile (
|
||||
verilator_flags2 => ["--stats"],
|
||||
);
|
||||
|
||||
file_grep ($Self->{stats}, qr/Optimizations, Split always\s+(\d+)/i, 0);
|
||||
# Important: if reorder succeeded, we should see no dly vars.
|
||||
# Equally important: twin test t_alw_noreorder should see dly vars,
|
||||
# is identical to this test except for disabling the reorder step.
|
||||
foreach my $file ("$Self->{obj_dir}/$Self->{VM_PREFIX}.cpp",
|
||||
"$Self->{obj_dir}/$Self->{VM_PREFIX}.h") {
|
||||
file_grep_not($file, qr/dly__t__DOT__v1/i);
|
||||
file_grep_not($file, qr/dly__t__DOT__v2/i);
|
||||
file_grep_not($file, qr/dly__t__DOT__v3/i);
|
||||
}
|
||||
|
||||
execute (
|
||||
check_finished=>1,
|
||||
);
|
||||
|
||||
ok(1);
|
||||
1;
|
55
test_regress/t/t_alw_reorder.v
Normal file
55
test_regress/t/t_alw_reorder.v
Normal file
@ -0,0 +1,55 @@
|
||||
// DESCRIPTION: Verilator: Verilog Test module
|
||||
//
|
||||
// This file ONLY is placed into the Public Domain, for any use,
|
||||
// without warranty, 2018 by Wilson Snyder.
|
||||
|
||||
module t (/*AUTOARG*/
|
||||
// Inputs
|
||||
clk
|
||||
);
|
||||
|
||||
input clk;
|
||||
integer cyc; initial cyc=1;
|
||||
|
||||
reg [15:0] m_din;
|
||||
|
||||
reg [15:0] v1;
|
||||
reg [15:0] v2;
|
||||
reg [15:0] v3;
|
||||
integer nosplit;
|
||||
|
||||
always @ (posedge clk) begin
|
||||
// write needed so that V3Dead doesn't kill v0..v3
|
||||
$write(" values %x %x %x\n", v1, v2, v3);
|
||||
|
||||
// Locally-set 'nosplit' will prevent the if from splitting
|
||||
// in splitAlwaysAll(). This whole always block should still be
|
||||
// intact when we call splitReorderAll() which is the subject
|
||||
// of this test.
|
||||
nosplit = cyc;
|
||||
if (nosplit > 2) begin
|
||||
/* S1 */ v1 <= 16'h0;
|
||||
/* S2 */ v1 <= m_din;
|
||||
/* S3 */ if (m_din == 16'h0) begin
|
||||
/* X1 */ v2 <= v1;
|
||||
/* X2 */ v3 <= v2;
|
||||
end
|
||||
end
|
||||
|
||||
// We expect to swap S2 and S3, and to swap X1 and X2.
|
||||
// We can check that this worked by the absense of dly vars
|
||||
// in the generated output; if the reorder fails (or is disabled)
|
||||
// we should see dly vars for v1 and v2.
|
||||
end
|
||||
|
||||
always @ (posedge clk) begin
|
||||
if (cyc!=0) begin
|
||||
cyc<=cyc+1;
|
||||
if (cyc==7) begin
|
||||
$write("*-* All Finished *-*\n");
|
||||
$finish;
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
endmodule
|
@ -12,7 +12,7 @@ compile (
|
||||
);
|
||||
|
||||
if ($Self->{vlt}) {
|
||||
file_grep ($Self->{stats}, qr/Optimizations, Split always\s+(\d+)/i, 6);
|
||||
file_grep ($Self->{stats}, qr/Optimizations, Split always\s+(\d+)/i, 3);
|
||||
}
|
||||
|
||||
execute (
|
||||
|
@ -13,29 +13,15 @@ module t (/*AUTOARG*/
|
||||
|
||||
reg [15:0] m_din;
|
||||
|
||||
// OK
|
||||
// We expect all these blocks should split;
|
||||
// blocks that don't split should go in t_alw_nosplit.v
|
||||
|
||||
reg [15:0] a_split_1, a_split_2;
|
||||
always @ (/*AS*/m_din) begin
|
||||
a_split_1 = m_din;
|
||||
a_split_2 = m_din;
|
||||
end
|
||||
|
||||
// OK
|
||||
reg [15:0] b_split_1, b_split_2;
|
||||
always @ (/*AS*/m_din) begin
|
||||
b_split_1 = m_din;
|
||||
b_split_2 = b_split_1;
|
||||
end
|
||||
|
||||
// Not OK
|
||||
reg [15:0] c_split_1, c_split_2;
|
||||
always @ (/*AS*/m_din) begin
|
||||
c_split_1 = m_din;
|
||||
c_split_2 = c_split_1;
|
||||
c_split_1 = ~m_din;
|
||||
end
|
||||
|
||||
// OK
|
||||
reg [15:0] d_split_1, d_split_2;
|
||||
always @ (posedge clk) begin
|
||||
d_split_1 <= m_din;
|
||||
@ -43,44 +29,27 @@ module t (/*AUTOARG*/
|
||||
d_split_1 <= ~m_din;
|
||||
end
|
||||
|
||||
// Not OK
|
||||
reg [15:0] h_split_1;
|
||||
reg [15:0] h_split_2;
|
||||
always @ (posedge clk) begin
|
||||
$write(" foo %x", m_din);
|
||||
$write(" bar %x\n", m_din);
|
||||
end
|
||||
|
||||
// Not OK
|
||||
reg [15:0] e_split_1, e_split_2;
|
||||
always @ (posedge clk) begin
|
||||
e_split_1 = m_din;
|
||||
e_split_2 = e_split_1;
|
||||
end
|
||||
|
||||
// Not OK
|
||||
reg [15:0] f_split_1, f_split_2;
|
||||
always @ (posedge clk) begin
|
||||
f_split_2 = f_split_1;
|
||||
f_split_1 = m_din;
|
||||
end
|
||||
|
||||
// Not Ok
|
||||
reg [15:0] l_split_1, l_split_2;
|
||||
always @ (posedge clk) begin
|
||||
l_split_2 <= l_split_1;
|
||||
l_split_1 <= l_split_2 | m_din;
|
||||
end
|
||||
|
||||
// OK
|
||||
reg [15:0] z_split_1, z_split_2;
|
||||
always @ (posedge clk) begin
|
||||
z_split_1 <= 0;
|
||||
z_split_1 <= ~m_din;
|
||||
end
|
||||
always @ (posedge clk) begin
|
||||
z_split_2 <= 0;
|
||||
z_split_2 <= z_split_1;
|
||||
// $write(" cyc = %x m_din = %x\n", cyc, m_din);
|
||||
if (cyc > 2) begin
|
||||
if (m_din == 16'h0) begin
|
||||
h_split_1 <= 16'h0;
|
||||
h_split_2 <= 16'h0;
|
||||
end
|
||||
else begin
|
||||
h_split_1 <= m_din;
|
||||
h_split_2 <= ~m_din;
|
||||
end
|
||||
end
|
||||
else begin
|
||||
h_split_1 <= 16'h0;
|
||||
h_split_2 <= 16'h0;
|
||||
end
|
||||
end
|
||||
|
||||
// (The checker block is an exception, it won't split.)
|
||||
always @ (posedge clk) begin
|
||||
if (cyc!=0) begin
|
||||
cyc<=cyc+1;
|
||||
@ -93,39 +62,26 @@ module t (/*AUTOARG*/
|
||||
m_din <= 16'he11e;
|
||||
//$write(" A %x %x\n", a_split_1, a_split_2);
|
||||
if (!(a_split_1==16'hfeed && a_split_2==16'hfeed)) $stop;
|
||||
if (!(b_split_1==16'hfeed && b_split_2==16'hfeed)) $stop;
|
||||
if (!(c_split_1==16'h0112 && c_split_2==16'hfeed)) $stop;
|
||||
if (!(d_split_1==16'h0112 && d_split_2==16'h0112)) $stop;
|
||||
if (!(e_split_1==16'hfeed && e_split_2==16'hfeed)) $stop;
|
||||
if (!(f_split_1==16'hfeed && f_split_2==16'hfeed)) $stop;
|
||||
if (!(z_split_1==16'h0112 && z_split_2==16'h0112)) $stop;
|
||||
if (!(h_split_1==16'hfeed && h_split_2==16'h0112)) $stop;
|
||||
end
|
||||
if (cyc==5) begin
|
||||
m_din <= 16'he22e;
|
||||
if (!(a_split_1==16'he11e && a_split_2==16'he11e)) $stop;
|
||||
if (!(b_split_1==16'he11e && b_split_2==16'he11e)) $stop;
|
||||
if (!(c_split_1==16'h1ee1 && c_split_2==16'he11e)) $stop;
|
||||
if (!(d_split_1==16'h0112 && d_split_2==16'h0112)) $stop;
|
||||
if (!(z_split_1==16'h0112 && z_split_2==16'h0112)) $stop;
|
||||
// Two valid orderings, as we don't know which posedge clk gets evaled first
|
||||
if (!(e_split_1==16'hfeed && e_split_2==16'hfeed) && !(e_split_1==16'he11e && e_split_2==16'he11e)) $stop;
|
||||
if (!(f_split_1==16'hfeed && f_split_2==16'hfeed) && !(f_split_1==16'he11e && f_split_2==16'hfeed)) $stop;
|
||||
if (!(h_split_1==16'hfeed && h_split_2==16'h0112)) $stop;
|
||||
end
|
||||
if (cyc==6) begin
|
||||
m_din <= 16'he33e;
|
||||
if (!(a_split_1==16'he22e && a_split_2==16'he22e)) $stop;
|
||||
if (!(b_split_1==16'he22e && b_split_2==16'he22e)) $stop;
|
||||
if (!(c_split_1==16'h1dd1 && c_split_2==16'he22e)) $stop;
|
||||
if (!(d_split_1==16'h1ee1 && d_split_2==16'h0112)) $stop;
|
||||
if (!(z_split_1==16'h1ee1 && d_split_2==16'h0112)) $stop;
|
||||
// Two valid orderings, as we don't know which posedge clk gets evaled first
|
||||
if (!(e_split_1==16'he11e && e_split_2==16'he11e) && !(e_split_1==16'he22e && e_split_2==16'he22e)) $stop;
|
||||
if (!(f_split_1==16'he11e && f_split_2==16'hfeed) && !(f_split_1==16'he22e && f_split_2==16'he11e)) $stop;
|
||||
if (!(h_split_1==16'he11e && h_split_2==16'h1ee1)) $stop;
|
||||
end
|
||||
if (cyc==7) begin
|
||||
$write("*-* All Finished *-*\n");
|
||||
$finish;
|
||||
end
|
||||
end
|
||||
end
|
||||
end // always @ (posedge clk)
|
||||
|
||||
endmodule
|
||||
|
Loading…
Reference in New Issue
Block a user