#!/usr/bin/perl -w # $Id: tbw 1089 2018-12-19 10:45:41Z mueller $ # # Copyright 2007-2018 by Walter F.J. Mueller # # This program is free software; you may redistribute and/or modify it under # the terms of the GNU General Public License as published by the Free # Software Foundation, either version 3, or (at your option) any later version. # # This program is distributed in the hope that it will be useful, but # WITHOUT ANY WARRANTY, without even the implied warranty of MERCHANTABILITY # or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License # for complete details. # # Revision History: # Date Rev Version Comment # 2018-12-18 1089 1.5.5 add and use bailout # 2016-09-03 805 1.5.4 use {} as delimiter for immediate mode data # 2016-08-28 804 1.5.3 BUGFIX: xsim: append -R to ARGV (was prepended...) # 2016-07-02 782 1.5.2 add TBW_GHDL_OPTS # 2016-06-25 778 1.5.1 support all sim modes # 2016-04-17 762 1.5 make '-run' default for [IX]Sim, add '-norun' # 2016-03-20 748 1.4 recode OPTIONS handling and -fifo handling # 2016-02-06 727 1.3 add XSim support # 2015-01-04 629 1.2.6 BUGFIX: setup proper dsc values after -fifo # 2014-12-23 619 1.2.5 add -fifo and -verbose options # 2014-07-27 575 1.2.4 use xtwi to start ISim models # 2011-11-06 420 1.2.3 fix tbw.dat parsing (allow / in file names) # 2010-05-23 294 1.2.2 handle tb_code's in non-local directories # 2010-04-18 279 1.2.1 add -help and more text to print_usage() # 2009-11-22 252 1.2 add ISim support # 2007-09-15 82 1.1.1 test for ambigous matches of name arguments; for # "suff=[l1;l2;..]" style inlines use linkname_tmp.tmp # as filename # 2007-09-09 81 1.1 add fifo setup to tbw; allow multiple action lines # per target; support immediate mode data # "[line1;line2;...]" values # 2007-08-03 71 1.0.1 handle redefinition of existing symlink correctly # 2007-06-30 62 1.0 Initial version # # 'test bench wrapper' to setup symlink refering to stimulus file(s) # # usage: tbw [file] [args] # # will look for file (default is _stim.dat) and setup a symlink # _stim_dat to refer to it. All args are passed along to # use 5.14.0; # require Perl 5.14 or higher use strict; # require strict checking use POSIX qw(mkfifo); use FileHandle; use File::Spec; use Cwd 'abs_path'; my $tb_code; my $is_ghdl; # uses ghdl simulator my $is_isim; # uses ISE simulator my $is_isim_run; my $is_xsim; # uses vivado simulator my $opt_run; my $opt_norun; my $opt_fifo; my $opt_verbose; my @args_pos; # list of positional args my @args_nam; # list of named args my @file_dsc; # file descriptors from tbw.dat my $ghdl_opts = $ENV{TBW_GHDL_OPTS}; # ghdl extra options autoflush STDOUT 1; # autoflush, so nothing lost on exec later if (scalar(@ARGV) && $ARGV[0] =~ m/^-+help$/) { # -help or --help given print_usage(); exit 0; } if (scalar(@ARGV) == 0) { print "tbw-E: name of test bench code missing\n"; print_usage(); exit 1; } # process test-bench-filename $tb_code = shift @ARGV; my $tb_code_path = "."; my $tb_code_name = $tb_code; if ($tb_code =~ m|^(.*)/(.*)$|) { $tb_code_path = $1; $tb_code_name = $2; } # process -norun, -fifo and -verbose options (can be in any order now) while (scalar(@ARGV)) { my $opt = $ARGV[0]; if ($opt =~ m/^-+norun$/) { $opt_norun = 1; shift @ARGV;} elsif ($opt =~ m/^-+fifo$/) { $opt_fifo = 1; shift @ARGV;} elsif ($opt =~ m/^-+verbose$/) { $opt_verbose = 1; shift @ARGV;} elsif ($opt =~ m/^-+run$/) { print "tbw-I: legacy option '-run' seen and ignored; is now default\n"; shift @ARGV; } else { last;} } my $tb_code_stem = $tb_code_name; $tb_code_stem =~ s/_[fsorept]sim$//; # drop sim mode suffix if ($tb_code_stem =~ /_ISim$/) { # is it an ISim executable ? $tb_code_stem =~ s/_ISim$//; # drop _ISim $is_isim = 1; $is_isim_run = not $opt_norun; } if ($tb_code_stem =~ /_XSim$/) { # is it an XSim executable ? $tb_code_stem =~ s/_XSim$//; # drop _XSim $is_xsim = 1; push @ARGV,'-R' unless $opt_norun; # run all unless '-norun' given } $is_ghdl = not ($is_isim or $is_xsim); bailout("'$tb_code' not existing or not executable") if (not -e $tb_code); # # read tbw.dat file in current directory or directory of executable # my $tbwdat_file = "tbw.dat"; $tbwdat_file = "$tb_code_path/tbw.dat" unless (-r "tbw.dat"); if (-r $tbwdat_file) { my $ok = 0; my $done = 0; open (TBW, $tbwdat_file) or bailout("failed to open '$tbwdat_file': $!"); while () { chomp; next if /^#/; if ( m{^\s*\[([\.\/a-zA-Z0-9_]*)\]\s*$} ) { last if $done; $ok = 0; $ok = 1 if ($1 eq $tb_code || $1 eq $tb_code_stem); } elsif ( m{^\s*([a-zA-Z0-9_]*)\s*=\s*([a-zA-Z0-9_./<>]*)\s*$} ) { if ($ok) { push @file_dsc, {tag=>$1, val=>$2}; $done = 1; } } else { print "tbw-E: bad line in tbw.dat:\n $_\n"; } } } else { print "tbw-I: didn't find ${tbwdat_file}, using defaults\n"; } # # if no tbw.dat or no matching stanza found, setup defaults # unless (scalar (@file_dsc)) { if ($opt_fifo) { push @file_dsc, {tag=>'rlink_cext_fifo_rx', val=>''}; push @file_dsc, {tag=>'rlink_cext_fifo_tx', val=>''}; push @file_dsc, {tag=>'rlink_cext_conf', val=>''}; } else { push @file_dsc, {tag=>$tb_code_stem . "_stim", val=>$tb_code_stem . "_stim.dat"}; } } # # now process argument list # { my $ind = 0; while (scalar(@ARGV)>0 && not $ARGV[0] =~ /^-/) { my $arg = shift @ARGV; my $ok; if ($arg =~ /([a-zA-Z0-9_]*)=(.*)/) { # named argument my $tag = $1; my $val = $2; foreach my $dsc (@file_dsc) { if ($dsc->{tag} =~ /$tag$/) { $dsc->{val} = $val; $ok += 1; } } bailout("can't match named argument: $arg") if ($ok == 0); bailout("ambiguous match for named argument: $arg") if ($ok > 1); } else { # positional argument if ($ind < scalar(@file_dsc)) { $file_dsc[$ind]->{val} = $arg; } else { bailout("too many positional arguments: $arg"); } $ind += 1; } } } if ($opt_verbose) { foreach my $dsc (@file_dsc) { my $tag = $dsc->{tag}; my $val = $dsc->{val}; printf " %s = %s\n", $tag, $val; } } # # now handle all specified file descriptors # foreach my $dsc (@file_dsc) { my $tag = $dsc->{tag}; my $val = $dsc->{val}; if ($val eq "") { # handle FIFO's next if (-p $tag); print "tbw-I: create FIFO $tag\n"; mkfifo($tag, 0666) or bailout("can't mkfifo '$tag': $!"); } else { # handle link to file cases if ($val =~ /^\{(.*)\}$/) { # immediate data case: "{line1;line2;...}" my @lines = split /;/, $1; my $fname = "$tag\_tmp.tmp"; open TFILE,">$fname" or bailout("can't create temporary file '$fname': $!"); foreach (@lines) { s/^\s*//; s/\s*$//; print TFILE "$_\n"; } close TFILE; $val = $fname; } else { unlink "$tag\_tmp.tmp" if (-e "$tag\_tmp.tmp"); # remove old .tmp file $val = "/dev/null" if ($val eq ""); # null file case } # handle file names # - if absolute path keep then # - if relative path seen note that it is relative to test bench code path # --> prepent test bench path, canonize, and convert to a relative path # name relavive to cwd ! unless ($val =~ m|^/|) { $val = $tb_code_path . '/' . $val; $val = File::Spec->abs2rel(abs_path($val)); } if (not -r $val) { bailout("file for '$tag' not existing or not readable: $val"); } if (-l $tag) { my $cur_link = readlink $tag; if ($cur_link ne $val) { print "tbw-I: redefine $tag -> $val\n"; unlink $tag or bailout("failed to unlink '$tag': $!"); symlink $val, $tag or bailout("failed to symlink 1: $!"); } } else { if (-e $tag) { bailout("$tag exists but is not a symlink"); } else { print "tbw-I: define $tag -> $val\n"; symlink $val, $tag or bailout("failed to symlink 2: $!"); } } } } # # additional ghdl options # if ($is_ghdl && defined $ghdl_opts) { push @ARGV, split /\s+/,$ghdl_opts; } # # here all ok, finally exec test bench # if ($is_isim_run) { # handle for isim 'run all' my $cmd = "xtwi" . " " . $tb_code . " " . join " ",@ARGV; open (ISIM_RUN, "| $cmd") or bailout("failed to open process pipe to isim: $!"); print ISIM_RUN "run all\n"; print ISIM_RUN "quit\n"; close (ISIM_RUN) or bailout("failed to close process pipe to isim: $!"); } else { # otherwise just exec # print ($tb_code . " " . join(" ",@ARGV) . "\n"); exec $tb_code,@ARGV or bailout("failed to exec: $!"); } #------------------------------------------------------------------------------- sub bailout { my ($msg) = @_; print STDERR "tbw-F: $msg\n"; exit 1; } # ---------------------------------------------------------------------------- sub print_usage { print "usage: tbw [opts] [filedefs] [ghdl-opts]\n"; print " opts\n"; print " -norun for _ISim tb's, runs the tb without 'run all' command\n"; print " -fifo use rlink_cext fifo, ignore tbw.dat\n"; print " -verbose show the used tag,value settings before execution\n"; print " filedefs define tb input, either filename in tbw.dat order or\n"; print " tag=name or tag=[] pairs with tag matching one in in\n"; print " tbw.dat. The [] form allows to give data inline, e.g.\n"; print " like \"_conf={.rpmon 1}\"\n"; print " ghdl-opts are all other options starting with a '-', they are\n"; print " passed to the testbench. Some useful ghdl options are:\n"; print " --wave=x.ghw\n"; print " --stack-max-size=16384\n"; print " --stop-time=1ns --disp-time --trace-processes\n"; }