diff --git a/ci/docker/buildenv/Dockerfile b/ci/docker/buildenv/Dockerfile index f3008bb97..6f9dd3539 100644 --- a/ci/docker/buildenv/Dockerfile +++ b/ci/docker/buildenv/Dockerfile @@ -40,7 +40,7 @@ RUN apt-get update \ WORKDIR /tmp -RUN cpan install -fi Unix::Processors Parallel::Forker Bit::Vector +RUN cpan install -fi Unix::Processors Parallel::Forker RUN git clone https://github.com/veripool/vcddiff.git && \ make -C vcddiff && \ diff --git a/ci/travis-install.bash b/ci/travis-install.bash index 27adf42ec..9178e66b5 100755 --- a/ci/travis-install.bash +++ b/ci/travis-install.bash @@ -94,7 +94,6 @@ elif [ "$TRAVIS_BUILD_STAGE_NAME" = "test" ]; then if [ "$TRAVIS_DIST" != "trusty" ]; then TRAVIS_CPAN_REPO=https://cpan.org fi - # Not listing Bit::Vector as slow to install, and only skips one test yes yes | sudo cpan -M $TRAVIS_CPAN_REPO -fi Unix::Processors Parallel::Forker install-vcddiff else diff --git a/docs/install.adoc b/docs/install.adoc index dfe5c0e75..65ae632f3 100644 --- a/docs/install.adoc +++ b/docs/install.adoc @@ -102,7 +102,6 @@ Those developing Verilator itself may also want these (see internals.adoc): cpan install Pod::Perldoc cpan install Unix::Processors cpan install Parallel::Forker - cpan install Bit::Vector ==== Install SystemC diff --git a/docs/internals.adoc b/docs/internals.adoc index 557d33b5e..aab4d8099 100644 --- a/docs/internals.adoc +++ b/docs/internals.adoc @@ -628,8 +628,6 @@ https://github.com/veripool/vcddiff * Cmake for build paths that use it. -* Bit::Vector to test vgen.pl - === Controlling the Test Driver Test drivers are written in PERL. All invoke the main test driver script, diff --git a/test_regress/t/t_vgen.pl b/test_regress/t/t_vgen.pl deleted file mode 100755 index a488b5a3f..000000000 --- a/test_regress/t/t_vgen.pl +++ /dev/null @@ -1,32 +0,0 @@ -#!/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); - -if (eval "use Bit::Vector; return 2;" != 2) { - skip("Vgen test requires Bit::Vector"); -} else { - - top_filename("$Self->{obj_dir}/vgen.v"); - - run(cmd => ["./vgen.pl", - "-o $Self->{top_filename}", - #"--seed 0", - ]); - - compile( - ); - - execute( - check_finished => 1, - ); -} -ok(1); -1; diff --git a/test_regress/vgen.pl b/test_regress/vgen.pl deleted file mode 100755 index 112e083ba..000000000 --- a/test_regress/vgen.pl +++ /dev/null @@ -1,1068 +0,0 @@ -#!/usr/bin/env perl -# See copyright, etc in below POD section. -###################################################################### - -require 5.006_001; -use warnings; - -use Getopt::Long; -use IO::File; -use Pod::Usage; -use Data::Dumper; $Data::Dumper::Indent = 1; -use Bit::Vector; -use strict; -use vars qw($Debug); - -our @Orig_ARGV = @ARGV; -our $Rerun_Args = $0." ".join(' ',@Orig_ARGV); -$Rerun_Args =~ s/\s+$//; - -use vars qw(@Blocks - %Vars - %VarAttrs - %VarsBlock - %Tree - @Commit - $Depth - %IdWidth - %Ops); - -#====================================================================== - -# width=> Number of bits the output size is, 0=you tell me. -# func=> What to put in output file -# signed=> 0=unsigned output, 1=signed output, '%1'=signed if op1 signed -# lsb=> LSB for variable declarations -# em=> How to calculate emulated return value -# %w Width of this output op ($treeref->{width}) -# %v Output value ($treeref->{val}) -# %1r First operand ($treeref->{op1}) -# %1v First operand value ($treeref->{op1}{val}) -# %1w First operand width ($treeref->{op1}{width}) - -our $Raise_Weight_Max = 50; -%Ops = -( - 'VCONST'=> {weight=>1&&20, width=>0, sc=>1, terminal=>1, v=>'%v', }, - 'VIDNEW'=> {weight=>1&&10, width=>0, sc=>1, terminal=>0, v=>'%i', }, - 'VIDOLD'=> {weight=>1&&20, width=>0, sc=>1, terminal=>0, v=>'%i', }, - 'VIDSAME'=> {weight=>1&&20, width=>0, sc=>1, terminal=>0, v=>'%i', }, - 'VRANGE'=> {weight=>1&&30, width=>0, signed=>0,sc=>0, terminal=>0, v=>'%i[%2:%3]', }, - 'VBITSEL'=> {weight=>1&&10, width=>1, signed=>0,sc=>0, terminal=>0, v=>'%i[%2]', }, - 'VBITSELP'=> {weight=>1&&10, width=>0, signed=>0,sc=>0, terminal=>0, v=>'%i[%2+:%3]', }, - 'VBITSELM'=> {weight=>1&&10, width=>0, signed=>0,sc=>0, terminal=>0, v=>'%i[%2-:%3]', }, - # Unary - 'VEXTEND'=> {weight=>1&&3, width=>-2, signed=>0,sc=>0, terminal=>0, v=>'{%xd\'h0,%1}', }, - 'VLOGNOT'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(! %1)', }, - 'VREDAND'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(& %1)', }, - 'VREDOR'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(| %1)', }, - 'VREDNAND'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(~& %1)', }, - 'VREDNOR'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(~| %1)', }, - 'VREDXNOR'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(^~ %1)', }, - 'VREDXOR'=> {weight=>1&&1, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(^ %1)', }, - 'VNOT'=> {weight=>1&&3, width=>0, sc=>1, terminal=>0, v=>'(~ %1)', }, - 'VNEGATE'=> {weight=>1&&2, width=>0, sc=>1, terminal=>0, v=>'(- %1)', }, - 'VCOUNTONES'=> {weight=>0&&2, width=>32, signed=>0, sc=>0, terminal=>0, v=>'\$countones(%1)', }, # No ncv support - 'VONEHOT'=> {weight=>0&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'\$onehot(%1)', }, # No ncv support - 'VONEHOT0'=> {weight=>0&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'\$onehot0(%1)', }, # No ncv support - # Binary - 'VAND'=> {weight=>1&&2, width=>0, sc=>1, terminal=>0, v=>'(%1 & %2)', }, - 'VOR'=> {weight=>1&&2, width=>0, sc=>1, terminal=>0, v=>'(%1 | %2)', }, - 'VNAND'=> {weight=>1&&0, width=>0, sc=>0, terminal=>0, v=>'(%1 ~& %2)', }, #FIX vcs bug! - 'VNOR'=> {weight=>1&&0, width=>0, sc=>0, terminal=>0, v=>'(%1 ~| %2)', }, #FIX vcs bug! - 'VXOR'=> {weight=>1&&2, width=>0, sc=>1, terminal=>0, v=>'(%1 ^ %2)', }, - 'VXNOR'=> {weight=>1&&0, width=>0, sc=>0, terminal=>0, v=>'(%1 ^~ %2)', }, #FIX vcs bug! - 'VEQ'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 == %2)', }, - 'VNEQ'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 != %2)', }, - 'VGT'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 > %2)', }, - 'VGTE'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 >= %2)', }, - 'VLT'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 < %2)', }, - 'VLTE'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 <= %2)', }, - 'VEQCASE'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 === %2)', }, # FIX just a = for now - 'VNEQCASE'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 !== %2)', }, # FIX just a != for now - 'VLOGOR'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 || %2)', }, - 'VLOGAND'=> {weight=>1&&2, width=>1, signed=>0, sc=>0, terminal=>0, v=>'(%1 && %2)', }, - 'VADD'=> {weight=>1&&10, width=>0, sc=>1, terminal=>0, v=>'(%1 + %2)', }, - 'VSUB'=> {weight=>1&&10, width=>0, sc=>1, terminal=>0, v=>'(%1 - %2)', }, - 'VMUL'=> {weight=>1&&15,width=>0, sc=>1, terminal=>0, v=>'(%1 * %2)', }, # High % as rarely applyable - # Unspecified behavior with == (a-signed / b) -- see t_math_signed5.v test - 'VDIV'=> {weight=>1&&8, width=>0, signed=>0, sc=>1, terminal=>0, v=>'((%2)==%xw\'h0 ? %xw\'%xsh0:(%1 / %2))', }, - 'VMODDIV'=> {weight=>1&&8, width=>0, signed=>0, sc=>1, terminal=>0, v=>'((%2)==%xw\'h0 ? %xw\'%xsh0:(%1 %% %2))', }, - #'VPOW'=> {weight=>2&&0,width=>-64, sc=>0, terminal=>0, v=>'(%1 ** %2)', }, - 'VSHIFTL'=> {weight=>1&&8, width=>0, signed=>0, sc=>0, terminal=>0, v=>'(%1 << %2)', }, - 'VSHIFTLS'=> {weight=>1&&8, width=>0, signed=>1, sc=>0, terminal=>0, v=>'(%1 <<< %2)', }, - 'VSHIFTR'=> {weight=>1&&8, width=>0, signed=>0, sc=>0, terminal=>0, v=>'(%1 >> %2)', }, - 'VSHIFTRS'=> {weight=>1&&15,width=>0, signed=>1, sc=>0, terminal=>0, v=>'(%1 >>> %2)', }, # ShiftR seems to sign extend differently for <=32 and >32 bits - 'VCONCAT'=> {weight=>1&&4, width=>-2,signed=>0, sc=>0, terminal=>0, v=>'{%1,%2}', }, - 'VREPLIC'=> {weight=>1&&2, width=>0, signed=>0, sc=>0, terminal=>0, v=>'{%1{%2}}', }, - 'VREPLIC1W'=> {weight=>1&&2, width=>0, signed=>0, sc=>0, terminal=>0, v=>'{%1{%2}}', }, - 'VSIGNED'=> {weight=>1&&2, width=>0, signed=>1, sc=>0, terminal=>0, v=>'\$signed(%1)', }, - 'VUNSIGNED'=> {weight=>1&&2, width=>0, signed=>0, sc=>0, terminal=>0, v=>'\$unsigned(%1)', }, - # Triops - 'VCOND'=> {weight=>1&&4, width=>0, sc=>0, terminal=>0, v=>'(%1 ? %2 : %3)', }, - # Control flow - #VIF - #VFOR - #VCASE - #VCASEX - #VCASEZ - ); - -my %ops2 = -( - 'VCONST'=> {pl=>'', rnd=>'rnd_const(%tr);'}, - 'VIDNEW'=> {pl=>'%tv=$Vars{%i}{val};', - rnd=>'%i=next_id(%tw);' - .' $Vars{%i}=gen_leaf(width=>%tw,trunc=>1,signed=>%tg);' - .' $VarAttrs{%i}{lsb} = rnd_lsb();' - .' id_commit(%tr,"%i");1;',}, - 'VIDOLD'=> {pl=>'%tv=$Vars{%i}{val};', rnd=>'%i=id_old(%tr);', ok_id_width=>1,}, - 'VIDSAME'=> {pl=>'%tv=$Vars{%i}{val};', rnd=>'%i=id_same(%tr);', ok_id_width=>1,}, - # These create IDs they then extract from - 'VRANGE'=> {pl=>'VRANGE(%tr,$Vars{%i}{val},%2v,%3v,$VarAttrs{%i}{lsb});', - rnd=>'%i=next_id(%tw); $VarAttrs{%i}{lsb} = 0&&rnd_lsb();' - .' my $lsb=rnd(128-%tw); my $msb=$lsb+%tw-1;' - .' %2r=val_leaf($msb); %3r=val_leaf($lsb);' - .' $Vars{%i}=gen_leaf(width=>($msb+1));'}, - 'VBITSEL'=> {pl=>'VRANGE(%tr,$Vars{%i}{val},%2v,%2v,$VarAttrs{%i}{lsb});', - rnd=>'%i=next_id(%tw); $VarAttrs{%i}{lsb} = 0&&rnd_lsb();' - .' my $wid=min(128,rnd_width()|3);' - .' %2r=gen_leaf(width=>(log2($wid)-1),signed=>0);' - .' $Vars{%i}=gen_leaf(width=>$wid);'}, - 'VBITSELP'=> {pl=>'VBITSELP(%tr,$Vars{%i}{val},%2v,%3v,$VarAttrs{%i}{lsb});', - rnd=>'%i=next_id(%tw); $VarAttrs{%i}{lsb} = 0&&rnd_lsb();' - .' my $wid=min(128,(%tw+rnd_width()|3)); %3r=val_leaf(%tw); my $maxval = $wid-%tw; %2r=(($maxval<4)?val_leaf($maxval):gen_leaf(width=>(log2($maxval)-1),signed=>0));' - .' $Vars{%i}=gen_leaf(width=>$wid);'}, - 'VBITSELM'=> {pl=>'VBITSELM(%tr,$Vars{%i}{val},%2v,%3v,$VarAttrs{%i}{lsb});', - rnd=>'%i=next_id(%tw); $VarAttrs{%i}{lsb} = 0&&rnd_lsb();' - .' my $wid=min(128,(%tw+rnd_width()|3)); %3r=val_leaf(%tw); my $maxval = $wid-1; my $minval=%tw-1; %2r=val_leaf(rnd($maxval-$minval)+$minval);' - .' $Vars{%i}=gen_leaf(width=>$wid);'}, # No easy way to make expr with specified minimum - # Unary - 'VEXTEND'=> {pl=>'VRESIZE (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>rnd_width(%tw-1));'}, - 'VLOGNOT'=> {pl=>'VLOGNOT (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDAND'=> {pl=>'VREDAND (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDOR'=> {pl=>'VREDOR (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDNAND'=> {pl=>'VREDNAND (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDNOR'=> {pl=>'VREDNOR (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDXOR'=> {pl=>'VREDXOR (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VREDXNOR'=> {pl=>'VREDXNOR (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VNOT'=> {pl=>'VNOT (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VNEGATE'=> {pl=>'VNEGATE (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VCOUNTONES'=> {pl=>'VCOUNTONES(%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VONEHOT'=> {pl=>'VONEHOT (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - 'VONEHOT0'=> {pl=>'VONEHOT0 (%tr,%1v);', rnd=>'%1r=gen_leaf(width=>0);'}, - # Binary - 'VAND'=> {pl=>'VAND (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VOR'=> {pl=>'VOR (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VNAND'=> {pl=>'VNAND (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VNOR'=> {pl=>'VNOR (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VXOR'=> {pl=>'VXOR (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VXNOR'=> {pl=>'VXNOR (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VEQ'=> {pl=>'VEQ (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VNEQ'=> {pl=>'VNE (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VGT'=> {pl=>'VGT (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VGTE'=> {pl=>'VGE (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VLT'=> {pl=>'VLT (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VLTE'=> {pl=>'VLE (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VEQCASE'=> {pl=>'VEQ (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VNEQCASE'=> {pl=>'VNE (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>%1w,signed=>%1g);'}, - 'VLOGOR'=> {pl=>'VLOGOR (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>0);'}, - 'VLOGAND'=> {pl=>'VLOGAND(%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>0); %2r=gen_leaf(width=>0);'}, - 'VADD'=> {pl=>'VADD (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);', trunc=>1,}, - 'VSUB'=> {pl=>'VSUB (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);', trunc=>1,}, - 'VMUL'=> {pl=>'VMUL (%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);', trunc=>1,}, # Multiply generates larger width, so need truncate for safety - 'VDIV'=> {pl=>'VDIV (%tr,%1r,%2r,0);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - 'VMODDIV'=> {pl=>'VDIV (%tr,%1r,%2r,1);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>%tw,signed=>%tg);'}, - #'VPOW'=> {pl=>'VPOW (%tr,%1r,%2r);', rnd=>'%1r=gen_leaf(width=>min(%tw,6),signed=>%tg); %2r=gen_leaf(width=>min(%tw,8),signed=>%tg);', trunc=>1,}, # Generates larger width, so need truncate for safety - 'VSHIFTL'=> {pl=>'VSHIFTL(%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>log2(%tw)+1,signed=>%tg);'}, - 'VSHIFTLS'=> {pl=>'VSHIFTL(%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>log2(%tw)+1,signed=>%tg);'}, - 'VSHIFTR'=> {pl=>'VSHIFTR(%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>log2(%tw)+1,signed=>%tg);'}, - 'VSHIFTRS'=> {pl=>'VSHIFTRS(%tr,%1v,%2v);', rnd=>'%1r=gen_leaf(width=>%tw,signed=>%tg); %2r=gen_leaf(width=>log2(%tw)+1,signed=>%tg);'}, - 'VCONCAT'=> {pl=>'VCONCAT(%tr,%1v,%2v);', rnd=>'my $d=(rnd(%tw-2)+1); %1r=gen_leaf(width=>$d,signed=>0); %2r=gen_leaf(width=>(%tw-$d),signed=>0);'}, - 'VREPLIC'=> {pl=>'VREPLIC(%tr,%1v,%2v);', rnd=>'my $d=rnd_rep_width(%tw); %1r=val_leaf($d); %2r=gen_leaf(width=>(%tw/$d),signed=>0);'}, - 'VREPLIC1W'=> {pl=>'VREPLIC(%tr,%1v,%2v);', rnd=>'%1r=val_leaf(%tw); %2r=gen_leaf(width=>1,signed=>0);'}, - 'VSIGNED'=> {pl=>'VCLONE (%tr,%1v,0);', rnd=>'%1r=gen_leaf(width=>%tw);'}, - 'VUNSIGNED'=> {pl=>'VCLONE (%tr,%1v,0);', rnd=>'%1r=gen_leaf(width=>%tw);'}, - # Triops - 'VCOND'=> {pl=>'VCOND(%tr,%1v,%2v,%3v);', rnd=>'%1r=gen_leaf(width=>1); %2r=gen_leaf(width=>%tw,signed=>%tg); %3r=gen_leaf(width=>%tw,signed=>%tg);'}, - ); - -foreach my $op (keys %ops2) { - while ((my $key,my $val) = each %{$ops2{$op}}) { - $Ops{$op}{$key} = $val; - } -} - -#====================================================================== -# main - -#Bit::Vector->Configuration("ops=arithmetic"); - - -my $opt_seed=5; -our $Opt_NumOps = 30; -our $Opt_Depth = 4; -our $Opt_Output; -our $Opt_Signed = 1; -our $Opt_Raise; -our $Opt_BlockStmts = 2; -our $Signed_Pct = 60; -$Debug = 0; -if (! GetOptions ( - "help" => \&usage, - "debug" => \&debug, - "depth=i" => \$Opt_Depth, - "blockstmts=i"=> \$Opt_BlockStmts, - "numops=i" => \$Opt_NumOps, - "o=s" => \$Opt_Output, - "raise=i" => \$Opt_Raise, - "seed=i" => \$opt_seed, - "signed!" => \$Opt_Signed, - "<>" => \¶meter, - )) { - usage(); -} - -if ($opt_seed==0) { - srand(); - $opt_seed = rnd(1<<20)+1; - $Rerun_Args =~ s/-seed[= ]+0/-seed=$opt_seed/; - print " $Rerun_Args\n"; -} -srand($opt_seed); -init(); -selftest(); -gentest(); -$Opt_Output or die "%Error: Need -o option,"; -write_output_v($Opt_Output); - -#---------------------------------------------------------------------- - -sub usage { - pod2usage(-verbose=>2, -exitval=>0, -output=>\*STDOUT); - exit(1); # Unreachable -} - -sub debug { - $Debug = 1; -} - -sub parameter { - my $param = shift; - die "%Error: Unknown parameter: $param\n"; -} - -####################################################################### -####################################################################### -####################################################################### -####################################################################### -# Global Functions - -sub init { - for my $op (keys %Ops) { - my $opref = $Ops{$op}; - $opref->{name} = $op; - gen_v($opref); - gen_pl($opref); - gen_rnd($opref); - } - raise(); -} - -sub raise { - for (my $i=0; $i<($Opt_Raise||0); $i++) { - my @ops = (values %Ops); - while (1) { - my $rndop = $ops[rnd($#ops + 1)]; - next if !$rndop->{weight}; # Don't turn on disabled ops - $rndop->{weight} += rnd($Raise_Weight_Max); - printf "\tWeight %-15s +%d\n",$rndop->{name},$rndop->{weight}; - last; - } - } -} - -sub gentest { - for (my $opn=0; $opn<$Opt_NumOps/$Opt_BlockStmts; $opn++) { - do_a_test(); - } -} - -####################################################################### -# Randomization - -sub _rnd_op_ok { - my $opref = shift; - my $paramref = shift; - return (($opref->{width} == 0 - || $opref->{width} == $paramref->{width} - # Note -2 means >, while -32 means {width}==-31 && $paramref->{width}<=31) # -31... must be <31 bits - || ($opref->{width}==-32 && $paramref->{width}<=32) # -32... must be <32 bits - || ($opref->{width}==-63 && $paramref->{width}<=63) # -63... must be <63 bits - || ($opref->{width}==-64 && $paramref->{width}<=64) # -64... must be <64 bits - || ($opref->{width}==-2 && $paramref->{width}>=2) # -2... must be >2 bits - ) - && (!$opref->{ok_id_width} || $IdWidth{$paramref->{width}}{$paramref->{signed}||0}) - && (!defined $opref->{signed} || ($opref->{signed} == ($paramref->{signed}||0))) - && (!$opref->{trunc} || $paramref->{trunc}) - && (!$opref->{opt_signed} || $Opt_Signed) - && (($Depth < $Opt_Depth && !$paramref->{need_terminal}) - || $opref->{terminal})); -} - -sub rnd_op { - my $paramref = shift; - - my $totweight = 0; - foreach my $opref (values %Ops) { - if (_rnd_op_ok($opref,$paramref)) { - $totweight += $opref->{weight}; - } - } - my $chooseweight = rnd($totweight); - $totweight = 0; - foreach my $opref (sort {$a->{name} cmp $b->{name}} values %Ops) { - if (_rnd_op_ok($opref,$paramref)) { - $totweight += $opref->{weight}; - if ($chooseweight < $totweight) { - return $opref; - } - } - } - die "%Error: No instructions match,"; -} - -sub rnd_width { - my $max = shift; - my $v = rnd(100); - my $n = (0 - || (($v<20) && 1) - || (($v<25) && 2) - || (($v<30) && 31) - || (($v<35) && 32) - || (($v<40) && 63) - || (($v<45) && 64) - || (($v<50) && 95) - || (($v<55) && 96) - || (rnd(128)+1)); - if ($max && $n>=$max) { $n = rnd($max-1)+1; } - return $n; -} - -sub rnd_rep_width { - my $out = shift; - return 1 if $out==1; - # We'd like to pick any divisor that works. - my @factors; - for (my $div=1; $div<$out; $div++) { - if (int($out/$div)==($out/$div)) { - push @factors, $div; - } - } - my $fac = $factors[rnd($#factors+1)]; - #print "RND REP $out -> $fac (@factors)\n" if $Debug; - return $fac; -} - -sub rnd_const { - my $treeref = shift; - my $width = $treeref->{width} or die; - my $v = rnd(100); - - my $val = Bit::Vector->new($width); - if ($v<25) { # zero - } elsif ($v<50) { # ones - for (my $w=0; $w<$val->Word_Size; ++$w) { - $val->Word_Store(0,~0); - } - } elsif ($v<60) { # one - $val->Word_Store(0,1); - } else { # random - for (my $w=0; $w<$val->Word_Size; ++$w) { - $val->Word_Store($w,rnd_int()); - } - } - $treeref->{val} = $val; -} - -sub rnd_int { - my $v = rnd(100); - return 0 if ($v<25); - return ~0 if ($v<50); - return 1 if ($v<60); - return rnd32(); -} - -sub rnd_lsb { - return 0; - #return rnd(8)-4; # Not working yet -} - -sub rnd { - return (int(rand($_[0]))) if ($_[0] < (1<<15)); - return (rnd32() % $_[0]); -} -sub rnd32 { - my $vp = int(rand(1<<16)); - $vp ^= (int(rand(1<<8)))<<16; # Single 1<<16 doesn't work - $vp ^= (int(rand(1<<8)))<<24; - return ($vp); -} - -####################################################################### - -our $Next_Id = 0; -sub next_id { - # Note width hasn't been determined yet - $Next_Id++; - my $id = sprintf("W%04d",$Next_Id); - return $id; -} -sub id_commit { - my $treeref = shift; - my $width = $treeref->{width}; - my $signed = $treeref->{signed}; - my $id = shift; - push @Commit, sub { - $IdWidth{$width}{$signed} = [] if !$IdWidth{$width}{$signed}; - push @{$IdWidth{$width}{$signed}}, $id; - $VarsBlock{$id}{set} = 1; - 1; - }; -} - -sub id_old { - my $treeref = shift; - my $width = $treeref->{width}; - my $signed = $treeref->{signed}; - - my $n = $#{$IdWidth{$width}{$signed}} + 1; - my $idn = rnd($n); - my $id = $IdWidth{$width}{$signed}[$idn]; - $VarsBlock{$id}{used} = 1; - return $id; -} - -sub id_same { - my $treeref = shift; - my $width = $treeref->{width}; - my $signed = $treeref->{signed}; - - my @possible; - foreach my $id (keys %VarsBlock) { - next if !$VarsBlock{$id}{used}; - my $varref = $Vars{$id}; - next if $varref->{signed} != $signed; - next if $varref->{width} != $width; - push @possible, $id; - } - my $n = $#possible + 1; - if ($n<1) { # Nothing, grab another! - return id_old($treeref,$width,$signed); - } - my $idn = rnd($n); - my $id = $possible[$idn]; - $VarsBlock{$id}{used} = 1; - return $id; -} - -sub write_output_v { - my $filename = shift; - - my $fh = IO::File->new($filename, "w") or die("%Error: $! $filename,\n"); - print $fh "// Created by: $Rerun_Args\n"; - - print $fh "module vgen (clk);\n"; - print $fh " input clk;\n"; - print $fh " reg check; initial check = '0;\n"; - print $fh ' initial $write("\n*** Vgen.v starting, seed = ',$opt_seed,'\n");',"\n"; - - print $fh " // verilator lint_off UNSIGNED\n"; - print $fh " // verilator lint_off CMPCONST\n"; - print $fh " // verilator lint_off WIDTH\n"; - print $fh "\n"; - - my $cycles = 2; - - foreach my $var (sort (keys %Vars)) { - print $fh "",decl_text ($var),"\n"; - } - - foreach my $block (@Blocks) { - print $fh "\t//".('='x60)."\n"; - my $style = rnd(100); - if ($style < 15) { - # This allows statements to get split up, and constants to propagate - print $fh " always @(", join(" or ", ('check', @{$block->{inputs}})); - print $fh ") begin : $block->{name}\n"; - print $fh @{$block->{preass}}; - print $fh " end\n"; - print $fh " always @(posedge clk) begin : $block->{name}Check\n"; - print $fh @{$block->{body}}; - print $fh " end\n"; - } - elsif ($style < 40) { - print $fh " always @(", join(" or ", ('check', @{$block->{inputs}})); - print $fh ") begin : $block->{name}\n"; - print $fh @{$block->{preass}}; - print $fh @{$block->{body}}; - print $fh " end\n"; - } - else { - foreach my $stmt (@{$block->{preass}}) { - $cycles++; - print $fh " always @(posedge clk) begin\n"; - $stmt =~ s/ = / <= /mg; - print $fh $stmt; - print $fh " end\n"; - } - print $fh " always @(posedge clk) begin\n"; - print $fh @{$block->{body}}; - print $fh " end\n"; - } - } - - print $fh "\n"; - print $fh " parameter [31:0] CYCLES /*verilator public*/ = $cycles;\n"; - print $fh "\n"; - print $fh " integer cyc; initial cyc = 0;\n"; - print $fh " always @(posedge clk) begin\n"; - print $fh "`ifdef TEST_VERBOSE\n"; - print $fh ' $write("[%0t] cyc=%0d check=%d\n", $time, cyc, check);',"\n"; - print $fh "`endif\n"; - print $fh " cyc <= cyc + 1;\n"; - print $fh " if (cyc < CYCLES) begin\n"; - print $fh " check <= 1'b0;\n"; - print $fh " end\n"; - print $fh " else if (cyc >= CYCLES) begin\n"; - print $fh " check <= 1'b1;\n"; - print $fh " if (cyc >= (CYCLES+10)) begin\n"; - print $fh ' $write("*-* All Finished *-*\n");',"\n"; - print $fh ' $finish;',"\n"; - print $fh " end\n"; - print $fh " end\n"; - print $fh " end\n"; - - print $fh "endmodule\n"; - - $fh->close(); -} - -###################################################################### - -sub callers { - for (my $i=0; ; $i++) { - my @c = caller($i); - last if !$c[0]; - print "Caller $i: ",join(' ',@c[0..3]),"\n"; - } -} - -####################################################################### -####################################################################### -####################################################################### -####################################################################### -# Code generation/emitting Functions - -sub do_a_test { - local $Depth = 0; - @Commit = (); - %VarsBlock = (); - - my $block = { - name=>"Block".($#Blocks+2), - body=>[], - preass=>[], - inputs=>[], - outputs=>[], - }; - - for (my $i=0; $i<$Opt_BlockStmts; $i++) { - my $treeref = gen_leaf(width=>0); - push @{$block->{body}}, - "\tif ($treeref->{text} != ".$treeref->val_to_text().") if (check) ".stop_text().";\n"; - } - - foreach my $var (keys %VarsBlock) { - push @{$block->{inputs}}, $var - if $VarsBlock{$var}{used} && !$VarsBlock{$var}{set}; - } - - foreach my $var (reverse (sort (keys %Vars))) { - my $varref = $Vars{$var}; - next if $varref->{printedit}; - $varref->{printedit} = 1; - push @{$block->{outputs}}, $var; - push @{$block->{preass}}, sprintf ("\t$var = %s;\n" - ,$varref->{text}); - } - - foreach my $com (@Commit) { - &{$com} or die "%Error: Can't eval:\n$com\n $@ "; - } - - push @Blocks, $block; -} - -sub gen_leaf { - my $inforef = {width=>0, # Anything - need_terminal=>0, - #trunc=>undef, # Allow multiply op - @_}; - - $inforef->{width} ||= rnd_width(); - $inforef->{signed} = ($Opt_Signed && $inforef->{width}>1 && (rnd(100)<$Signed_Pct))?1:0 - if !defined $inforef->{signed}; - print +((" "x$Depth)."Leaf of width $inforef->{width}\n") if $Debug; - my $op = rnd_op($inforef); - - my $treeref = new Vg::Base; - while ((my $key,my $val) = each %{$op}) { - $treeref->{$key} = $val; - } - while ((my $key,my $val) = each %{$inforef}) { - $treeref->{$key} = $val; - } - - local $Depth = $Depth+1; - print "RndSub $treeref->{rnd_sub_text}\n" if $Debug; - $treeref->{rnd_sub}($treeref); - $treeref->tree_dump() if $Debug; - - print "RndPl\n" if $Debug; - $treeref->{pl_sub}($treeref); - print "RndV\n" if $Debug; - $treeref->{text} = $treeref->{v_sub}($treeref); - print "Done\n" if $Debug; - print " Value ",$treeref->{val}," = ",$treeref->val_to_text(),"\n" if $Debug; - #$treeref->tree_dump() if $Debug; - - $treeref->{val_size} = $treeref->{val}->Size; # Debugging - $treeref->{val_text} = $treeref->{val}->to_Hex; # Debugging - - ($treeref->{val}->Size == $treeref->{width}) - or die "%Error: Size mismatch ",$treeref->{val}->Size,"!=",$treeref->{width},"\n",Dumper($treeref); - - return $treeref; -} - -sub gen_v { - my $opref = shift; - - my $fmt = $opref->{v}; - $fmt =~ s/%1/%s/g; - $fmt =~ s/%2/%s/g; - $fmt =~ s/%3/%s/g; - $fmt =~ s/%v/%s/g; - $fmt =~ s/%i/%s/g; - $fmt =~ s/%x[wds]/%s/g; - - my $argl = $opref->{v}; - my @args; - while ($argl =~ s/(%x.|%.)//) { - my $arg = $1; - push @args, '$treeref->{op1}{text}' if $arg =~ /%1/; - push @args, '$treeref->{op2}{text}' if $arg =~ /%2/; - push @args, '$treeref->{op3}{text}' if $arg =~ /%3/; - push @args, '$treeref->val_to_text' if $arg =~ /%v/; - push @args, '$treeref->{id}' if $arg =~ /%i/; - push @args, '$treeref->{signed}?"s":""' if $arg =~ /%xs/; - push @args, '$treeref->{width}' if $arg =~ /%xw/; - push @args, '$treeref->{width}-$treeref->{op1}{width}' if $arg =~ /%xd/; - } - - my $func = ("sub { " - ." my \$treeref = shift;" - ." sprintf(\"$fmt\",".join(',',@args).");" - ."}"); - my $set = ("\$opref->{v_sub} = $func; 1;"); - $opref->{v_sub_text} = $func; # For seeing it in debugging dumps - #print "Op V $opref->{name} $set\n"; - eval($set) or die "%Error: Can't eval:\n$set\n $@ "; -} - -sub escapes { - my $str = shift; - my $cmt = shift; - $str =~ s/%tr/\$treeref/g; - $str =~ s/%tg/\$treeref->{signed}/g; - $str =~ s/%tv/\$treeref->{val}/g; - $str =~ s/%tw/\$treeref->{width}/g; - # - $str =~ s/%1r/\$treeref->{op1}/g; - $str =~ s/%1g/\$treeref->{op1}{signed}/g; - $str =~ s/%1n/(\$treeref->{op1}{val}->Word_Read(0))/g; - $str =~ s/%1v/\$treeref->{op1}{val}/g; - $str =~ s/%1w/\$treeref->{op1}{width}/g; - # - $str =~ s/%2r/\$treeref->{op2}/g; - $str =~ s/%2g/\$treeref->{op2}{signed}/g; - $str =~ s/%2n/(\$treeref->{op2}{val}->Word_Read(0))/g; - $str =~ s/%2v/\$treeref->{op2}{val}/g; - $str =~ s/%2w/\$treeref->{op2}{width}/g; - # - $str =~ s/%3r/\$treeref->{op3}/g; - $str =~ s/%3g/\$treeref->{op3}{signed}/g; - $str =~ s/%3n/(\$treeref->{op3}{val}->Word_Read(0))/g; - $str =~ s/%3v/\$treeref->{op3}{val}/g; - $str =~ s/%3w/\$treeref->{op3}{width}/g; - # - $str =~ s/%i/\$treeref->{id}/g; - ($str !~ /%/) or die "%Error: $cmt: Unknown %% escape in $str,"; - return $str; -} - -sub gen_pl { - my $opref = shift; - - my $str = escapes($opref->{pl}, $opref->{name}); - my $func = ("sub { " - ." my \$treeref = shift;" - ." $str;" - ."}"); - my $set = ("\$opref->{pl_sub} = $func; 1;"); - $opref->{pl_sub_text} = $func; # For seeing it in debugging dumps - #print "Op PL $opref->{name} $set\n"; - eval($set) or die "%Error: Can't eval:\n$set\n $@ "; -} - -sub gen_rnd { - my $opref = shift; - - my $str = escapes($opref->{rnd}, $opref->{name}); - - my $func = ("sub { " - ." my \$treeref = shift;" - ." $str;" - ."}"); - my $set = ("\$opref->{rnd_sub} = $func; 1;"); - $opref->{rnd_sub_text} = $func; # For seeing it in debugging dumps - #print "Op RND $opref->{name} $set\n"; - eval($set) or die "%Error: Can't eval:\n$set\n $@ "; -} - -sub stop_text { - return '$stop'; -} - -sub decl_text { - my $var = shift; - my $decl_with = shift; - - my $varref = $Vars{$var}; - return sprintf(" reg %s [%3d:%3d] %s %s; //=%d'h%s" - , ($varref->{signed}?"signed":" ") - , ($varref->{val}->Size)-1+$VarAttrs{$var}{lsb}, - , $VarAttrs{$var}{lsb} - , $var - , (rnd(100)<30 ? "/*verilator public*/":(" "x length("/*verilator public*/"))) - , $varref->{val}->Size - , lc $varref->{val}->to_Hex); -} - -####################################################################### -####################################################################### -####################################################################### -####################################################################### -# Math Functions - -sub selftest { - my $o = {}; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0xff)}, {val=>Bit::Vector->new_Dec(8,0x13)}, 0); - ($o->{val}->Word_Read(0) == 0x0d) or die; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0xff)}, {val=>Bit::Vector->new_Dec(8,0x13)}, 1); - ($o->{val}->Word_Read(0) == 0x08) or die; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0xff), signed=>1}, {val=>Bit::Vector->new_Dec(8,0x13), signed=>1}, 0); - ($o->{val}->Word_Read(0) == 0x00) or die; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0xff), signed=>1}, {val=>Bit::Vector->new_Dec(8,0x13), signed=>1}, 1); - ($o->{val}->Word_Read(0) == 0xff) or die; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0xff), signed=>1}, {val=>Bit::Vector->new_Dec(8,0xdb), signed=>1}, 1); - ($o->{val}->Word_Read(0) == 0xff) or die; - VDIV($o, {val=>Bit::Vector->new_Dec(8,0x72), signed=>1}, {val=>Bit::Vector->new_Dec(8,0xdb), signed=>1}, 1); - ($o->{val}->Word_Read(0) == 0x3) or die; -} -sub val_leaf { return {width=>32, signed=>0, val=>Bit::Vector->new_Dec(32,$_[0]), text=>$_[0],}; } - -sub makebool { return (Bit::Vector->new_Dec(1,$_[0])); } -sub newsized { return (Bit::Vector->new($_[0]->Size)); } -sub max { return $_[0]<$_[1] ? $_[1] : $_[0]; } -sub min { return $_[0]>$_[1] ? $_[1] : $_[0]; } - -sub log2 { - for (my $i=31; $i>=0; $i--) { - return $i+1 if $_[0]>(1<<$i); - } - return 0; -} - -sub countones { - my $out = 0; - for (my $bit=0; $bit < $_[0]->Size; $bit++) { - $out ++ if $_[0]->bit_test($bit); - } - return $out; -} - - -sub VLOGNOT { $_[0]{val} = makebool(($_[1]->is_empty)?1:0); } -sub VNEGATE { $_[0]{val} = my $o = newsized($_[1]); $o->Negate($_[1]); } -sub VCOUNTONES { $_[0]{val} = Bit::Vector->new_Dec(32,countones($_[1])); } -sub VONEHOT { $_[0]{val} = makebool((countones($_[1])==1)?1:0); } -sub VONEHOT0 { $_[0]{val} = makebool((countones($_[1])<=1)?1:0); } -sub VLOGAND { $_[0]{val} = makebool((!($_[1]->is_empty) && !($_[2]->is_empty))?1:0); } -sub VLOGOR { $_[0]{val} = makebool((!($_[1]->is_empty) || !($_[2]->is_empty))?1:0); } - -sub VCOND { if (!($_[1]->is_empty)) { $_[0]{val}=$_[2]->Clone; } else { $_[0]{val}=$_[3]->Clone; } } - -sub VREDAND { $_[0]{val} = makebool(($_[1]->is_full)?1:0); } -sub VREDOR { $_[0]{val} = makebool(($_[1]->is_empty)?0:1); } -sub VREDNAND { $_[0]{val} = makebool(($_[1]->is_full)?0:1); } -sub VREDNOR { $_[0]{val} = makebool(($_[1]->is_empty)?1:0); } -sub VREDXOR { - my $out = 0; - for (my $bit=0; $bit < $_[1]->Size; $bit++) { - $out ^= $_[1]->bit_test($bit); - } - $_[0]{val} = makebool($out); -} -sub VREDXNOR { - my $out = 1; - for (my $bit=0; $bit < $_[1]->Size; $bit++) { - $out ^= $_[1]->bit_test($bit); - } - $_[0]{val} = makebool($out); -} -sub eithercompare { ($_[1]->{signed} && $_[2]->{signed}) - ? $_[1]{val}->Compare($_[2]{val}) - : $_[1]{val}->Lexicompare($_[2]{val}); } -sub VEQ { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])==0) ?1:0); } -sub VNE { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])!=0) ?1:0); } -sub VLT { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])< 0) ?1:0); } -sub VLE { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])<=0) ?1:0); } -sub VGT { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])> 0) ?1:0); } -sub VGE { $_[0]{val} = makebool( (eithercompare($_[0],$_[1],$_[2])>=0) ?1:0); } - -sub VSHIFTLxx { - print "$Vars{vq}->ShiftL($_[0],$_[1]);\n"; - print " ",$_[0]->to_Hex," ",$_[1]->to_Hex,";\n"; - my $out = $_[0]->Clone; - $out->Move_Left($_[1]->Word_Read(0)); - print $out->to_Hex,"\n"; - return $out; } -sub VAND { $_[0]{val}=my $o=newsized($_[1]); $o->Intersection($_[1],$_[2]); } -sub VOR { $_[0]{val}=my $o=newsized($_[1]); $o->Union($_[1],$_[2]); } -sub VNAND { $_[0]{val}=my $o=newsized($_[1]); $o->Intersection($_[1],$_[2]); $o->Complement($o); } -sub VNOR { $_[0]{val}=my $o=newsized($_[1]); $o->Union($_[1],$_[2]); $o->Complement($o); } -sub VXOR { $_[0]{val}=my $o=newsized($_[1]); $o->ExclusiveOr($_[1],$_[2]); } -sub VXNOR{ $_[0]{val}=my $o=newsized($_[1]); $o->ExclusiveOr($_[1],$_[2]); $o->Complement($o); } -sub VNOT { $_[0]{val}=my $o=newsized($_[1]); $o->Complement($_[1]); } -sub VSHIFTL{ $_[0]{val}=my $o=$_[1]->Clone; $o->Move_Left ($_[2]->Word_Read(0)); } -sub VSHIFTR{ $_[0]{val}=my $o=$_[1]->Clone; $o->Move_Right($_[2]->Word_Read(0)); } -sub VSHIFTRS{$_[0]{val}=my $o=$_[1]->Clone; $o->Move_Right($_[2]->Word_Read(0)); - if ($_[1]->msb() && $_[2]->Word_Read(0)>0) {$o->Interval_Fill(max(0,$o->Size-1-$_[2]->Word_Read(0)), $o->Size-1); } - #print (" SHI ",$_[0]{val}->to_Hex,' = ',$_[1]->to_Hex,' >>> ',$_[2]->Word_Read(0),"\n"); -} -sub VCLONE { $_[0]{val}=$_[1]->Clone; } -sub VRESIZE { - $_[0]{val}=$_[1]->Clone; - $_[0]{val}->Resize($_[0]{width}); -} -sub VADD { $_[0]{val}=my $o=newsized($_[1]); $o->add($_[1],$_[2],0); } -sub VSUB { $_[0]{val}=my $o=newsized($_[1]); $o->subtract($_[1],$_[2],0); } -sub VMUL { - # Multiply is signed, so need an additional sign bit - my $a=$_[1]->Clone; $a->Resize($a->Size + 1); - my $b=$_[2]->Clone; $b->Resize($b->Size + 1); - my $mo=Bit::Vector->new($_[1]->Size + $_[2]->Size + 1); - $mo->Multiply($a,$b); - my $o=newsized($_[1]); $o->Interval_Copy($mo,0,0,$_[1]->Size); - $_[0]{val}=$o; -} -sub VDIV { - my $is_mod = $_[3]; - if ($_[2]{val}->is_empty) { # Avoid divide by zero - $_[0]{val}=newsized($_[1]{val}); - return; - } - my $a=$_[1]{val}->Clone; if (!$_[1]->{signed}) { $a->Resize($a->Size + 1); } - my $b=$_[2]{val}->Clone; if (!$_[2]->{signed}) { $b->Resize($b->Size + 1); } - #print ("//DIVpp ",$_[1]->to_Hex,' ',$_[2]->to_Hex,' ',$_[1]->Size,'.',$_[2]->Size," \n"); - #print ("//DIVpp ",$a->to_Hex,' ',$b->to_Hex,' ',$a->Size,'.',$b->Size," \n"); - my $quo=newsized($a); my $rem=newsized($a); - $quo->Divide($a,$b,$rem); # No division by zero - handled by if above - my $o=newsized($_[1]{val}); - $o->Interval_Copy($is_mod ? $rem : $quo,0,0,$_[1]{val}->Size); - #print "//DIV",($_[1]->{signed}?"S":" "),' w',$a->Size,' ',$_[1]{val}->to_Hex,' ',$_[2]{val}->to_Hex,' =',$quo->to_Hex,'.',$rem->to_Hex," \n"; - $_[0]{val}=$o; -} -sub VPOW { # Power is a signed operation - my $a=$_[1]{val}->Clone; if (!$_[1]->{signed}) { $a->Resize($_[1]{val}->Size + 1); } - my $b=$_[2]{val}->Clone; if (!$_[2]->{signed}) { $b->Resize($_[2]{val}->Size + 1); } - print "VVpow = ",$_[1]{val}->to_Hex," ** ",$_[2]{val}->to_Hex,"\n"; - my $mo=Bit::Vector->new($_[1]{val}->Size + 1); - $mo->Power($a,$b); - my $o=Bit::Vector->new($_[0]{width}); $o->Interval_Copy($mo,0,0,$_[1]{val}->Size); - $_[0]{val}=$o; - print "VV = $o\n"; -} -sub VRANGE { - #print "RANGE ",$_[1]->to_Hex,' ',$_[2]->to_Hex,' ',$_[3]->to_Hex," \n"; - return VRANGE_CONST($_[0], $_[1], $_[2]->Word_Read(0), - $_[3]->Word_Read(0), $_[4]); -} -sub VBITSELP { - return VRANGE_CONST($_[0], $_[1], $_[2]->Word_Read(0)+$_[3]->Word_Read(0)-1, - $_[2]->Word_Read(0), $_[4]); -} -sub VBITSELM { - return VRANGE_CONST($_[0],$_[1],$_[2]->Word_Read(0), - $_[2]->Word_Read(0)-$_[3]->Word_Read(0)+1, $_[4]); -} -sub VRANGE_CONST { - # to, from, msb, lsb, variable_lsb_to_subtract - #print "RANGE ",$_[1]->to_Hex,' ',$_[2],' ',$_[3],' ',$_[4]," \n"; - my $size = $_[2] - $_[3] + 1; - my $o=Bit::Vector->new($size); - if ($_[3] < $_[1]->Size) { - $o->Interval_Copy($_[1],0,$_[3]-$_[4],$size); - } - $_[0]{val}=$o; } -sub VCONCAT { - my $o=Bit::Vector->new($_[1]->Size + $_[2]->Size); - $o->Interval_Copy($_[1],$_[2]->Size,0,$_[1]->Size); - $o->Interval_Copy($_[2],0,0,$_[2]->Size); - $_[0]{val}=$o; -} -sub VREPLIC { - my $o=Bit::Vector->new($_[1]->Word_Read(0) * $_[2]->Size); - my $pos = 0; - for (my $time=0; $time<($_[1]->Word_Read(0)); $time++) { - $o->Interval_Copy($_[2],$pos,0,$_[2]->Size); - $pos += $_[2]->Size; - } - $_[0]{val}=$o; -} - -####################################################################### -####################################################################### -####################################################################### - -package Vg::Base; -use Data::Dumper; -use strict; - -#------------------------------------------------------------ -# CREATORS - -sub new { - my $class = shift; - my $self = { - width=>0, # Width of expression, 0=Pick a width - #signed=>0/1, # Undef = pick a sign - @_}; - bless $self, $class; - return $self; -} - -# ACCESSORS - -#------------------------------------------------------------ -# OUTPUTTING - -sub val_to_text { - my $treeref = shift; - my $val = lc $treeref->{val}->to_Hex(); - $val = "0" if $treeref->{val}->is_empty; - return ($treeref->{width} - .($treeref->{signed}?"'sh":"'h") - .$val); -} - -sub tree_dump { - my $treeref = shift; - print Dumper($treeref); -} - -####################################################################### -__END__ - -=pod - -=head1 NAME - -vgen.pl - Generate random verilog code - -=head1 SYNOPSIS - - vgen.pl -o vgen.v - -=head1 DESCRIPTION - -vgen.pl generates automatic random verilog programs. - -=head1 ARGUMENTS - -=over 4 - -=item --help - -Displays this message and program version and exits. - -=item --blockstmts - -Number of statements per block. Defaults to 2. - -=item --depth - -Maximum depth of generated expressions. - -=item --initial - -Put all statements into an initial block. This will probably be optimized -down to a NOP. - -=item --numops - -Number of operations to create. - -=item -o I - -Specify output filename. - -=item --raise - -Pick the specified number of random opcodes, and raise their frequency. - -=item --seed - -Seed for the random number generator. Defaults to 5, 0=randomize. - -=item --signed - -Include some signed arithmetic in the generated code. Experimental. - -=back - -=head1 DISTRIBUTION - -Copyright 2001-2020 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 - -=head1 AUTHORS - -Wilson Snyder - -=head1 SEE ALSO - -=cut - -###################################################################### -### Local Variables: -### compile-command: "./vgen.pl --depth=10 --blockstmts=10 -o obj_dir/vgen.v" -### compile-command: "v4make test_regress/t/t_vgen.pl " -### End: