| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
# Verilog - Verilog Perl Interface |
|
2
|
|
|
|
|
|
|
# See copyright, etc in below POD section. |
|
3
|
|
|
|
|
|
|
###################################################################### |
|
4
|
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
package Verilog::Netlist::Net; |
|
6
|
|
|
|
|
|
|
|
|
7
|
8
|
|
|
8
|
|
61
|
use Verilog::Netlist; |
|
|
8
|
|
|
|
|
16
|
|
|
|
8
|
|
|
|
|
229
|
|
|
8
|
8
|
|
|
8
|
|
41
|
use Verilog::Netlist::Subclass; |
|
|
8
|
|
|
|
|
15
|
|
|
|
8
|
|
|
|
|
480
|
|
|
9
|
8
|
|
|
8
|
|
54
|
use vars qw($VERSION @ISA); |
|
|
8
|
|
|
|
|
14
|
|
|
|
8
|
|
|
|
|
446
|
|
|
10
|
8
|
|
|
8
|
|
59
|
use strict; |
|
|
8
|
|
|
|
|
17
|
|
|
|
8
|
|
|
|
|
15327
|
|
|
11
|
|
|
|
|
|
|
@ISA = qw(Verilog::Netlist::Net::Struct |
|
12
|
|
|
|
|
|
|
Verilog::Netlist::Subclass); |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
$VERSION = '3.478'; |
|
15
|
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
my %_Type_Widths = ( |
|
17
|
|
|
|
|
|
|
'bit' => 1, |
|
18
|
|
|
|
|
|
|
'byte' => 8, |
|
19
|
|
|
|
|
|
|
'genvar' => 32, |
|
20
|
|
|
|
|
|
|
'integer' => 32, |
|
21
|
|
|
|
|
|
|
'localparam'=> 32, |
|
22
|
|
|
|
|
|
|
'logic' => 1, |
|
23
|
|
|
|
|
|
|
'longint' => 64, |
|
24
|
|
|
|
|
|
|
'parameter' => 32, |
|
25
|
|
|
|
|
|
|
'reg' => 1, |
|
26
|
|
|
|
|
|
|
'shortint' => 16, |
|
27
|
|
|
|
|
|
|
'supply0' => 1, |
|
28
|
|
|
|
|
|
|
'supply1' => 1, |
|
29
|
|
|
|
|
|
|
'tri' => 1, |
|
30
|
|
|
|
|
|
|
'tri0' => 1, |
|
31
|
|
|
|
|
|
|
'tri1' => 1, |
|
32
|
|
|
|
|
|
|
'triand' => 1, |
|
33
|
|
|
|
|
|
|
'trior' => 1, |
|
34
|
|
|
|
|
|
|
'trireg' => 1, |
|
35
|
|
|
|
|
|
|
'wand' => 1, |
|
36
|
|
|
|
|
|
|
'wire' => 1, |
|
37
|
|
|
|
|
|
|
'wor' => 1, |
|
38
|
|
|
|
|
|
|
); |
|
39
|
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
my %_Type_Accessors = ( |
|
41
|
|
|
|
|
|
|
'genvar' => 'decl_type', |
|
42
|
|
|
|
|
|
|
'localparam'=> 'decl_type', |
|
43
|
|
|
|
|
|
|
'parameter' => 'decl_type', |
|
44
|
|
|
|
|
|
|
'var' => 'decl_type', # Not in old version, but for completeness |
|
45
|
|
|
|
|
|
|
#'port' => 'decl_type', # Internals - Look at Port (input/output/inout/ref) |
|
46
|
|
|
|
|
|
|
#'net' => 'decl_type', # Internals - Look at net_type (wire/tri/...) |
|
47
|
|
|
|
|
|
|
# |
|
48
|
|
|
|
|
|
|
'supply0' => 'net_type', |
|
49
|
|
|
|
|
|
|
'supply1' => 'net_type', |
|
50
|
|
|
|
|
|
|
'tri' => 'net_type', |
|
51
|
|
|
|
|
|
|
'tri0' => 'net_type', |
|
52
|
|
|
|
|
|
|
'tri1' => 'net_type', |
|
53
|
|
|
|
|
|
|
'triand' => 'net_type', |
|
54
|
|
|
|
|
|
|
'trior' => 'net_type', |
|
55
|
|
|
|
|
|
|
'trireg' => 'net_type', |
|
56
|
|
|
|
|
|
|
'wand' => 'net_type', |
|
57
|
|
|
|
|
|
|
'wire' => 'net_type', |
|
58
|
|
|
|
|
|
|
'wor' => 'net_type', |
|
59
|
|
|
|
|
|
|
# |
|
60
|
|
|
|
|
|
|
'bit' => 'data_type', |
|
61
|
|
|
|
|
|
|
'byte' => 'data_type', |
|
62
|
|
|
|
|
|
|
'chandle' => 'data_type', |
|
63
|
|
|
|
|
|
|
'event' => 'data_type', |
|
64
|
|
|
|
|
|
|
'int' => 'data_type', |
|
65
|
|
|
|
|
|
|
'integer' => 'data_type', |
|
66
|
|
|
|
|
|
|
'logic' => 'data_type', |
|
67
|
|
|
|
|
|
|
'longint' => 'data_type', |
|
68
|
|
|
|
|
|
|
'real' => 'data_type', |
|
69
|
|
|
|
|
|
|
'realtime' => 'data_type', |
|
70
|
|
|
|
|
|
|
'reg' => 'data_type', |
|
71
|
|
|
|
|
|
|
'shortint' => 'data_type', |
|
72
|
|
|
|
|
|
|
'shortreal' => 'data_type', |
|
73
|
|
|
|
|
|
|
'string' => 'data_type', |
|
74
|
|
|
|
|
|
|
'time' => 'data_type', |
|
75
|
|
|
|
|
|
|
); |
|
76
|
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
###################################################################### |
|
78
|
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
structs('_new_base', |
|
80
|
|
|
|
|
|
|
'Verilog::Netlist::Net::Struct' |
|
81
|
|
|
|
|
|
|
=>[name => '$', #' # Name of the net |
|
82
|
|
|
|
|
|
|
filename => '$', #' # Filename this came from |
|
83
|
|
|
|
|
|
|
lineno => '$', #' # Linenumber this came from |
|
84
|
|
|
|
|
|
|
userdata => '%', # User information |
|
85
|
|
|
|
|
|
|
attributes => '%', #' # Misc attributes for systemperl or other processors |
|
86
|
|
|
|
|
|
|
# |
|
87
|
|
|
|
|
|
|
data_type => '$', #' # SystemVerilog Type (logic/integer/reg [3:0] etc) |
|
88
|
|
|
|
|
|
|
decl_type => '$', #' # Declaration type (parameter/genvar/port/net etc) |
|
89
|
|
|
|
|
|
|
net_type => '$', #' # Net type (wire/tri/supply0 etc) |
|
90
|
|
|
|
|
|
|
comment => '$', #' # Comment provided by user |
|
91
|
|
|
|
|
|
|
array => '$', #' # Vector |
|
92
|
|
|
|
|
|
|
module => '$', #' # Module, Program or Interface entity belongs to |
|
93
|
|
|
|
|
|
|
signed => '$', #' # True if signed |
|
94
|
|
|
|
|
|
|
value => '$', #' # For parameters, the value of the parameter |
|
95
|
|
|
|
|
|
|
# below only after links() |
|
96
|
|
|
|
|
|
|
port => '$', #' # Reference to port connected to |
|
97
|
|
|
|
|
|
|
msb => '$', #' # MSB of signal (if known) |
|
98
|
|
|
|
|
|
|
lsb => '$', #' # LSB of signal (if known) |
|
99
|
|
|
|
|
|
|
stored_lsb => '$', #' # Bit number of signal stored in bit 0 (generally lsb) |
|
100
|
|
|
|
|
|
|
_used_in => '$', #' # Driver count onto signal |
|
101
|
|
|
|
|
|
|
_used_out => '$', #' # Receiver count on signal |
|
102
|
|
|
|
|
|
|
_used_inout => '$', #' # Bidirect count on signal |
|
103
|
|
|
|
|
|
|
# SystemPerl only: below only after autos() |
|
104
|
|
|
|
|
|
|
simple_type => '$', #' # True if is uint (as opposed to sc_signal) |
|
105
|
|
|
|
|
|
|
sp_traced => '$', #' # Created by SP_TRACED |
|
106
|
|
|
|
|
|
|
sp_autocreated => '$', #' # Created by /*AUTOSIGNAL*/ |
|
107
|
|
|
|
|
|
|
]); |
|
108
|
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
sub new { |
|
110
|
12582
|
|
|
12582
|
0
|
16500
|
my $class = shift; |
|
111
|
12582
|
|
|
|
|
61138
|
my %params = @_; |
|
112
|
12582
|
|
|
|
|
215432
|
my $self = $class->_new_base(%params); |
|
113
|
12582
|
50
|
|
|
|
33540
|
$self->type($params{type}) if $params{type}; # Backward compatibility |
|
114
|
12582
|
|
|
|
|
40963
|
return $self; |
|
115
|
|
|
|
|
|
|
} |
|
116
|
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
sub delete { |
|
118
|
12407
|
|
|
12407
|
0
|
13978
|
my $self = shift; |
|
119
|
12407
|
|
|
|
|
157283
|
my $h = $self->module->_nets; |
|
120
|
12407
|
|
|
|
|
155019
|
delete $h->{$self->name}; |
|
121
|
12407
|
|
|
|
|
21862
|
return undef; |
|
122
|
|
|
|
|
|
|
} |
|
123
|
|
|
|
|
|
|
|
|
124
|
|
|
|
|
|
|
###################################################################### |
|
125
|
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
sub logger { |
|
127
|
0
|
|
|
0
|
1
|
0
|
return $_[0]->netlist->logger; |
|
128
|
|
|
|
|
|
|
} |
|
129
|
|
|
|
|
|
|
sub netlist { |
|
130
|
0
|
|
|
0
|
0
|
0
|
return $_[0]->module->netlist; |
|
131
|
|
|
|
|
|
|
} |
|
132
|
|
|
|
|
|
|
|
|
133
|
46
|
|
100
|
46
|
|
661
|
sub _used_in_inc { $_[0]->_used_in(1+($_[0]->_used_in()||0)); } |
|
134
|
51
|
|
100
|
51
|
|
700
|
sub _used_out_inc { $_[0]->_used_out(1+($_[0]->_used_out()||0)); } |
|
135
|
7
|
|
50
|
7
|
|
179
|
sub _used_inout_inc { $_[0]->_used_inout(1+($_[0]->_used_inout()||0)); } |
|
136
|
|
|
|
|
|
|
sub _used_in_dec { |
|
137
|
5
|
50
|
|
5
|
|
70
|
return if !$_[0]->_used_in(); |
|
138
|
5
|
|
|
|
|
72
|
$_[0]->_used_in(-1+$_[0]->_used_in()); |
|
139
|
|
|
|
|
|
|
} |
|
140
|
|
|
|
|
|
|
sub _used_out_dec { |
|
141
|
0
|
0
|
|
0
|
|
0
|
return if !$_[0]->_used_out(); |
|
142
|
0
|
|
|
|
|
0
|
$_[0]->_used_out(-1+$_[0]->_used_out()); |
|
143
|
|
|
|
|
|
|
} |
|
144
|
|
|
|
|
|
|
sub _used_inout_dec { |
|
145
|
0
|
0
|
|
0
|
|
0
|
return if !$_[0]->_used_inout(); |
|
146
|
0
|
|
|
|
|
0
|
$_[0]->_used_inout(-1+$_[0]->_used_inout()); |
|
147
|
|
|
|
|
|
|
} |
|
148
|
|
|
|
|
|
|
|
|
149
|
0
|
0
|
|
0
|
0
|
0
|
sub stored_lsb { defined $_[0]->SUPER::stored_lsb ? $_[0]->SUPER::stored_lsb : $_[0]->lsb; } |
|
150
|
|
|
|
|
|
|
|
|
151
|
|
|
|
|
|
|
sub width { |
|
152
|
9
|
|
|
9
|
1
|
46
|
my $self = shift; |
|
153
|
|
|
|
|
|
|
# Return bit width (if known) |
|
154
|
9
|
50
|
|
|
|
125
|
my $dt = $self->data_type; $dt="" if $dt eq "signed"; |
|
|
9
|
|
|
|
|
22
|
|
|
155
|
9
|
100
|
66
|
|
|
119
|
if (defined $self->msb && defined $self->lsb) { |
|
|
|
50
|
100
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
156
|
6
|
|
|
|
|
75
|
return (abs($self->msb - $self->lsb) + 1); |
|
157
|
|
|
|
|
|
|
} elsif (my $width = $_Type_Widths{$dt || $self->net_type || $self->decl_type}) { |
|
158
|
3
|
|
|
|
|
15
|
return $width; |
|
159
|
|
|
|
|
|
|
} |
|
160
|
0
|
|
|
|
|
0
|
return undef; |
|
161
|
|
|
|
|
|
|
} |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
sub type { |
|
164
|
0
|
|
|
0
|
1
|
0
|
my $self = shift; |
|
165
|
0
|
|
|
|
|
0
|
my $flag = shift; |
|
166
|
0
|
0
|
|
|
|
0
|
if (defined $flag) { |
|
167
|
0
|
0
|
|
|
|
0
|
if (my $acc = $_Type_Accessors{$flag}) { |
|
168
|
0
|
0
|
|
|
|
0
|
if ($acc eq 'decl_type') { $self->decl_type($flag); } |
|
|
0
|
0
|
|
|
|
0
|
|
|
169
|
0
|
|
|
|
|
0
|
elsif ($acc eq 'net_type') { $self->net_type($flag); } |
|
170
|
0
|
|
|
|
|
0
|
else { $self->data_type($flag); } |
|
171
|
|
|
|
|
|
|
} else { |
|
172
|
0
|
|
|
|
|
0
|
$self->data_type($flag); |
|
173
|
|
|
|
|
|
|
} |
|
174
|
|
|
|
|
|
|
} |
|
175
|
0
|
0
|
0
|
|
|
0
|
my $dt = $self->data_type; $dt="" if $dt && $dt eq "signed"; |
|
|
0
|
|
|
|
|
0
|
|
|
176
|
0
|
|
0
|
|
|
0
|
return $dt || $self->net_type || $self->decl_type; |
|
177
|
|
|
|
|
|
|
} |
|
178
|
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
###################################################################### |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
350
|
|
|
sub _link {} |
|
182
|
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
sub lint { |
|
184
|
137
|
|
|
137
|
1
|
189
|
my $self = shift; |
|
185
|
|
|
|
|
|
|
# Sequential logic may gen/use a signal, so we have to be a little sloppy |
|
186
|
137
|
100
|
|
|
|
1701
|
if (0&&$self->_used_inout() && $self->_used_out() |
|
187
|
|
|
|
|
|
|
&& !$self->array()) { # if an array, different outputs might hit different bits |
|
188
|
|
|
|
|
|
|
$self->warn("Signal is used as both a inout and output: ",$self->name(), "\n"); |
|
189
|
|
|
|
|
|
|
$self->dump_drivers(8); |
|
190
|
|
|
|
|
|
|
} |
|
191
|
0
|
|
|
|
|
0
|
elsif ($self->_used_out()) { |
|
192
|
40
|
50
|
66
|
|
|
497
|
if ($self->_used_out()>1 |
|
|
|
|
66
|
|
|
|
|
|
193
|
|
|
|
|
|
|
# if an array, different outputs might hit different bits |
|
194
|
|
|
|
|
|
|
&& !$self->array() |
|
195
|
|
|
|
|
|
|
# if vector, warn only if # of usages is higher than # of bits in vector |
|
196
|
|
|
|
|
|
|
&& (abs($self->msb() - $self->lsb()) + 1) < $self->_used_out()) { |
|
197
|
0
|
|
|
|
|
0
|
$self->warn("Signal has multiple drivers (", |
|
198
|
|
|
|
|
|
|
$self->_used_out(),"): ",$self->name(), "\n"); |
|
199
|
0
|
|
|
|
|
0
|
$self->dump_drivers(8); |
|
200
|
|
|
|
|
|
|
} |
|
201
|
|
|
|
|
|
|
} |
|
202
|
137
|
|
|
|
|
160
|
if (0&&$self->_used_in() && !$self->_used_out()) { |
|
203
|
|
|
|
|
|
|
$self->warn("Signal has no drivers: ",$self->name(), "\n"); |
|
204
|
|
|
|
|
|
|
} |
|
205
|
137
|
|
|
|
|
207
|
if (0&&$self->_used_out() && !$self->_used_in() |
|
206
|
|
|
|
|
|
|
&& $self->name() !~ /unused/) { |
|
207
|
|
|
|
|
|
|
$self->dump(5); |
|
208
|
|
|
|
|
|
|
$self->port->dump(10) if $self->port; |
|
209
|
|
|
|
|
|
|
$self->warn("Signal is not used (or needs signal declaration): ",$self->name(), "\n"); |
|
210
|
|
|
|
|
|
|
flush STDOUT; |
|
211
|
|
|
|
|
|
|
flush STDERR; |
|
212
|
|
|
|
|
|
|
} |
|
213
|
|
|
|
|
|
|
} |
|
214
|
|
|
|
|
|
|
|
|
215
|
|
|
|
|
|
|
###################################################################### |
|
216
|
|
|
|
|
|
|
## Outputters |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
sub _decls { |
|
219
|
120
|
|
|
120
|
|
140
|
my $self = shift; |
|
220
|
120
|
|
66
|
|
|
1531
|
my $out = $self->net_type || $self->decl_type; |
|
221
|
120
|
100
|
|
|
|
1531
|
if ($self->port) { |
|
222
|
69
|
100
|
|
|
|
843
|
$out = "input" if $self->port->direction eq "in"; |
|
223
|
69
|
100
|
|
|
|
864
|
$out = "output" if $self->port->direction eq "out"; |
|
224
|
69
|
100
|
|
|
|
870
|
$out = "inout" if $self->port->direction eq "inout"; |
|
225
|
|
|
|
|
|
|
} |
|
226
|
120
|
|
|
|
|
239
|
return $out; |
|
227
|
|
|
|
|
|
|
} |
|
228
|
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
sub verilog_text { |
|
230
|
120
|
|
|
120
|
0
|
167
|
my $self = shift; |
|
231
|
120
|
|
|
|
|
130
|
my @out; |
|
232
|
120
|
|
|
|
|
176
|
foreach my $decl ($self->_decls) { |
|
233
|
120
|
|
|
|
|
197
|
push @out, $decl; |
|
234
|
120
|
100
|
|
|
|
1524
|
push @out, " ".$self->data_type if $self->data_type; |
|
235
|
120
|
|
|
|
|
1586
|
push @out, " ".$self->name; |
|
236
|
120
|
50
|
|
|
|
1610
|
push @out, " ".$self->array if $self->array; |
|
237
|
120
|
100
|
100
|
|
|
1543
|
push @out, " = ".$self->value if defined $self->value && $self->value ne ''; |
|
238
|
120
|
|
|
|
|
205
|
push @out, ";"; |
|
239
|
120
|
100
|
66
|
|
|
1534
|
push @out, " ".$self->comment if defined $self->comment && $self->comment ne ''; |
|
240
|
|
|
|
|
|
|
} |
|
241
|
120
|
50
|
|
|
|
537
|
return (wantarray ? @out : join('',@out)); |
|
242
|
|
|
|
|
|
|
} |
|
243
|
|
|
|
|
|
|
|
|
244
|
|
|
|
|
|
|
sub dump { |
|
245
|
156
|
|
|
156
|
1
|
254
|
my $self = shift; |
|
246
|
156
|
|
50
|
|
|
258
|
my $indent = shift||0; |
|
247
|
156
|
100
|
50
|
|
|
2633
|
print " "x$indent,"Net:",$self->name() |
|
|
|
100
|
100
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
248
|
|
|
|
|
|
|
," ",($self->_used_in() ? "I":""),($self->_used_out() ? "O":""), |
|
249
|
|
|
|
|
|
|
," DeclT:",$self->decl_type||'' |
|
250
|
|
|
|
|
|
|
," NetT:",$self->net_type||'' |
|
251
|
|
|
|
|
|
|
," DataT:",$self->data_type||'' |
|
252
|
|
|
|
|
|
|
," Array:",$self->array()||''; |
|
253
|
156
|
100
|
|
|
|
3282
|
print " ",($self->msb).":".($self->lsb) if defined $self->msb; |
|
254
|
156
|
100
|
66
|
|
|
2507
|
print " Value:",$self->value if defined $self->value && $self->value ne ''; |
|
255
|
156
|
|
|
|
|
2515
|
print "\n"; |
|
256
|
|
|
|
|
|
|
} |
|
257
|
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
sub dump_drivers { |
|
259
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
|
260
|
0
|
|
0
|
|
|
|
my $indent = shift||0; |
|
261
|
0
|
|
|
|
|
|
print " "x$indent,"Net:",$self->name,"\n"; |
|
262
|
0
|
0
|
|
|
|
|
if (my $port = $self->port) { |
|
263
|
0
|
|
|
|
|
|
print " "x$indent," Port: ",$port->name," ",$port->direction,"\n"; |
|
264
|
|
|
|
|
|
|
} |
|
265
|
0
|
|
|
|
|
|
foreach my $cell ($self->module->cells_sorted) { |
|
266
|
0
|
|
|
|
|
|
foreach my $pin ($cell->pins_sorted) { |
|
267
|
0
|
|
|
|
|
|
foreach my $net ($pin->nets) { |
|
268
|
0
|
0
|
|
|
|
|
next unless defined $net->{net}; |
|
269
|
0
|
0
|
0
|
|
|
|
if ($pin->port && $net->{net} == $self) { |
|
|
|
0
|
|
|
|
|
|
|
270
|
0
|
|
|
|
|
|
print " "x$indent," Pin: ",$cell->name,".",$pin->name |
|
271
|
|
|
|
|
|
|
," ",$pin->port->direction,"\n"; |
|
272
|
|
|
|
|
|
|
} |
|
273
|
|
|
|
|
|
|
elsif ($self->name eq $net->{net}->name) { |
|
274
|
|
|
|
|
|
|
warn "%Warning: Internal net name duplicate: ".$cell->name." ".$self->name."\n" |
|
275
|
|
|
|
|
|
|
.$self->comment." ".$net->{net}->comment."\n" |
|
276
|
0
|
|
|
|
|
|
."$self ".$net->{net}->name."\n"; |
|
277
|
|
|
|
|
|
|
} |
|
278
|
|
|
|
|
|
|
} |
|
279
|
|
|
|
|
|
|
} |
|
280
|
|
|
|
|
|
|
} |
|
281
|
0
|
|
|
|
|
|
flush STDERR; |
|
282
|
0
|
|
|
|
|
|
flush STDOUT; |
|
283
|
|
|
|
|
|
|
} |
|
284
|
|
|
|
|
|
|
|
|
285
|
|
|
|
|
|
|
###################################################################### |
|
286
|
|
|
|
|
|
|
#### Package return |
|
287
|
|
|
|
|
|
|
1; |
|
288
|
|
|
|
|
|
|
__END__ |