|
|
@@ -15,7 +15,8 @@ use Text::ParseWords; # for shellwords
|
|
|
|
use Pod::Usage;
|
|
|
|
use Pod::Usage;
|
|
|
|
use Time::Piece;
|
|
|
|
use Time::Piece;
|
|
|
|
|
|
|
|
|
|
|
|
my $VERSION = 1.40;
|
|
|
|
# Makefile.PL assumes this is in ''
|
|
|
|
|
|
|
|
my $VERSION = '1.46';
|
|
|
|
|
|
|
|
|
|
|
|
my %options;
|
|
|
|
my %options;
|
|
|
|
interpretCommandline();
|
|
|
|
interpretCommandline();
|
|
|
@@ -41,22 +42,26 @@ my $last_replot_time = [gettimeofday];
|
|
|
|
|
|
|
|
|
|
|
|
# whether the previous replot was timer based
|
|
|
|
# whether the previous replot was timer based
|
|
|
|
my $last_replot_is_from_timer = 1;
|
|
|
|
my $last_replot_is_from_timer = 1;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $prev_timed_replot_time = [gettimeofday];
|
|
|
|
|
|
|
|
my $this_replot_is_from_timer;
|
|
|
|
my $this_replot_is_from_timer;
|
|
|
|
my $stdin = IO::Handle->new();
|
|
|
|
|
|
|
|
die "Couldn't open STDIN" unless $stdin->fdopen(fileno(STDIN),"r");
|
|
|
|
|
|
|
|
my $selector = IO::Select->new( $stdin );
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mainThread();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub getRangeSize
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
my ($id) = @_;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# I'd like to use //, but I guess some people are still on perl 5.8
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
exists $options{rangesize_hash}{$id} ?
|
|
|
|
|
|
|
|
$options{rangesize_hash}{$id} :
|
|
|
|
|
|
|
|
$options{rangesize_default};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub interpretCommandline
|
|
|
|
sub interpretCommandline
|
|
|
|
{
|
|
|
|
{
|
|
|
|
# if I'm using a self-plotting data file with a #! line, then $ARGV[0] will contain ALL of the
|
|
|
|
# if I'm using a self-plotting data file with a #! line, then $ARGV[0] will contain ALL of the
|
|
|
@@ -97,21 +102,24 @@ sub interpretCommandline
|
|
|
|
$options{with} = '';
|
|
|
|
$options{with} = '';
|
|
|
|
|
|
|
|
|
|
|
|
$options{rangesize} = [];
|
|
|
|
$options{rangesize} = [];
|
|
|
|
|
|
|
|
$options{tuplesize} = [];
|
|
|
|
|
|
|
|
|
|
|
|
GetOptions(\%options, 'stream:s', 'domain!', 'dataid!', '3d!', 'colormap!', 'lines!', 'points!',
|
|
|
|
GetOptions(\%options, 'stream:s', 'domain!', 'dataid!', 'vnlog!', '3d!', 'colormap!', 'lines!', 'points!',
|
|
|
|
'circles', 'legend=s{2}', 'autolegend!', 'xlabel=s', 'ylabel=s', 'y2label=s', 'zlabel=s',
|
|
|
|
'circles', 'legend=s{2}', 'autolegend!', 'xlabel=s', 'ylabel=s', 'y2label=s', 'zlabel=s',
|
|
|
|
'title=s', 'xlen=f', 'ymin=f', 'ymax=f', 'xmin=s', 'xmax=s', 'y2min=f', 'y2max=f',
|
|
|
|
'title=s', 'xlen=f', 'ymin=f', 'ymax=f', 'xmin=s', 'xmax=s', 'y2min=f', 'y2max=f',
|
|
|
|
'zmin=f', 'zmax=f', 'y2=s@',
|
|
|
|
'zmin=f', 'zmax=f', 'y2=s@',
|
|
|
|
'style=s{2}', 'curvestyle=s{2}', 'curvestyleall=s', 'styleall=s', 'with=s', 'extracmds=s@', 'set=s@', 'unset=s@',
|
|
|
|
'style=s{2}', 'curvestyle=s{2}', 'curvestyleall=s', 'styleall=s', 'with=s', 'extracmds=s@', 'set=s@', 'unset=s@',
|
|
|
|
'square!', 'square_xy!', 'hardcopy=s', 'maxcurves=i', 'monotonic!', 'timefmt=s',
|
|
|
|
'square!', 'square_xy!', 'hardcopy=s', 'maxcurves=i', 'monotonic!', 'timefmt=s',
|
|
|
|
'equation=s@',
|
|
|
|
'equation=s@',
|
|
|
|
|
|
|
|
'image=s',
|
|
|
|
'histogram=s@', 'binwidth=f', 'histstyle=s',
|
|
|
|
'histogram=s@', 'binwidth=f', 'histstyle=s',
|
|
|
|
'terminal=s',
|
|
|
|
'terminal=s',
|
|
|
|
'rangesize=s{2}', 'rangesizeall=i', 'extraValuesPerPoint=i',
|
|
|
|
'rangesize=s{2}', 'rangesizeall=i',
|
|
|
|
|
|
|
|
'tuplesize=s{2}', 'tuplesizeall=i',
|
|
|
|
|
|
|
|
'extraValuesPerPoint=i', # deprecated and undocumented
|
|
|
|
'help', 'dump', 'exit', 'version',
|
|
|
|
'help', 'dump', 'exit', 'version',
|
|
|
|
'geometry=s') or exit 1;
|
|
|
|
'geometry=s') or exit 1;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# handle various cmdline-option errors
|
|
|
|
# handle various cmdline-option errors
|
|
|
|
if ( $options{help} )
|
|
|
|
if ( $options{help} )
|
|
|
|
{
|
|
|
|
{
|
|
|
@@ -149,6 +157,11 @@ sub interpretCommandline
|
|
|
|
delete $options{with};
|
|
|
|
delete $options{with};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if( $options{dataid} && $options{vnlog} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
print STDERR "--dataid and --vnlog are mutually exclusive. Please just use one.\n";
|
|
|
|
|
|
|
|
exit -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# expand options that are given as comma-separated lists
|
|
|
|
# expand options that are given as comma-separated lists
|
|
|
|
for my $listkey (qw(histogram y2))
|
|
|
|
for my $listkey (qw(histogram y2))
|
|
|
@@ -156,7 +169,7 @@ sub interpretCommandline
|
|
|
|
@{$options{$listkey}} = map split('\s*,\s*', $_), @{$options{$listkey}}
|
|
|
|
@{$options{$listkey}} = map split('\s*,\s*', $_), @{$options{$listkey}}
|
|
|
|
if defined $options{$listkey};
|
|
|
|
if defined $options{$listkey};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for my $listkey (qw(curvestyle rangesize))
|
|
|
|
for my $listkey (qw(curvestyle rangesize tuplesize))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
next unless defined $options{$listkey};
|
|
|
|
next unless defined $options{$listkey};
|
|
|
|
my @in = @{$options{$listkey}};
|
|
|
|
my @in = @{$options{$listkey}};
|
|
|
@@ -176,6 +189,35 @@ sub interpretCommandline
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# convert all tuplesize business to rangesize
|
|
|
|
|
|
|
|
my $domainsize = $options{'3d'} ? 2 : 1;
|
|
|
|
|
|
|
|
if (defined $options{tuplesizeall})
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (defined $options{rangesizeall} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
print STDERR "Only one of --rangesizeall and --tuplesizeall may be given\n";
|
|
|
|
|
|
|
|
exit -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$options{rangesizeall} = $options{tuplesizeall} - $domainsize;
|
|
|
|
|
|
|
|
delete $options{tuplesizeall};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined $options{tuplesize})
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
$options{rangesize} //= [];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $N = @{$options{tuplesize}} / 2;
|
|
|
|
|
|
|
|
for my $i (0..$N-1)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
$options{tuplesize}[2*$i + 1] -= $domainsize;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
push @{$options{rangesize}}, @{$options{tuplesize}};
|
|
|
|
|
|
|
|
delete $options{tuplesize};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we're plotting histograms, then set the default histogram options for
|
|
|
|
# If we're plotting histograms, then set the default histogram options for
|
|
|
|
# each histogram curve
|
|
|
|
# each histogram curve
|
|
|
|
#
|
|
|
|
#
|
|
|
@@ -222,7 +264,10 @@ sub interpretCommandline
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# I now set up the rangesize to always be
|
|
|
|
# I now set up the rangesize to always be
|
|
|
|
|
|
|
|
#
|
|
|
|
# $options{rangesize_hash}{$id} // $options{rangesize_default}
|
|
|
|
# $options{rangesize_hash}{$id} // $options{rangesize_default}
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# which is available as getRangeSize($id)
|
|
|
|
if ( $options{rangesizeall} )
|
|
|
|
if ( $options{rangesizeall} )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
$options{rangesize_default} = $options{rangesizeall};
|
|
|
|
$options{rangesize_default} = $options{rangesizeall};
|
|
|
@@ -281,6 +326,13 @@ sub interpretCommandline
|
|
|
|
$options{curvestyleall} .= ' palette';
|
|
|
|
$options{curvestyleall} .= ' palette';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if ( defined $options{binwidth} && !@{$options{histogram}} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
print STDERR "--binwidth doesn't make sense without any histograms\n";
|
|
|
|
|
|
|
|
exit -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if ( $options{'3d'} )
|
|
|
|
if ( $options{'3d'} )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if ( !$options{domain} )
|
|
|
|
if ( !$options{domain} )
|
|
|
@@ -313,7 +365,7 @@ sub interpretCommandline
|
|
|
|
exit -1;
|
|
|
|
exit -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ( defined $options{binwidth} || @{$options{histogram}} )
|
|
|
|
if ( @{$options{histogram}} )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
print STDERR "--3d does not make sense with histograms\n";
|
|
|
|
print STDERR "--3d does not make sense with histograms\n";
|
|
|
|
exit -1;
|
|
|
|
exit -1;
|
|
|
@@ -347,6 +399,16 @@ sub interpretCommandline
|
|
|
|
print STDERR "--square_xy only makes sense with --3d\n";
|
|
|
|
print STDERR "--square_xy only makes sense with --3d\n";
|
|
|
|
exit -1;
|
|
|
|
exit -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for my $hist_curve(@{$options{histogram}})
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
my $hist_dim = getRangeSize($hist_curve);
|
|
|
|
|
|
|
|
if( $hist_dim != 1 )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
print STDERR "I only support 1D histograms, but curve '$hist_curve' has '$hist_dim'-D data\n";
|
|
|
|
|
|
|
|
exit -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(defined $options{xlen} && !$options{stream} )
|
|
|
|
if(defined $options{xlen} && !$options{stream} )
|
|
|
@@ -356,7 +418,8 @@ sub interpretCommandline
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if($options{stream} && defined $options{xlen} &&
|
|
|
|
if($options{stream} && defined $options{xlen} &&
|
|
|
|
( defined $options{xmin} || defined $options{xmax}))
|
|
|
|
( defined $options{xmin} || defined $options{xmax}) &&
|
|
|
|
|
|
|
|
!defined $options{histogram})
|
|
|
|
{
|
|
|
|
{
|
|
|
|
print STDERR "With --stream and --xlen the X bounds are set, so neither --xmin nor --xmax make sense\n";
|
|
|
|
print STDERR "With --stream and --xlen the X bounds are set, so neither --xmin nor --xmax make sense\n";
|
|
|
|
exit -1;
|
|
|
|
exit -1;
|
|
|
@@ -395,6 +458,27 @@ sub interpretCommandline
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# deal with --image. I just fill in --equation, and reverse the y extents if
|
|
|
|
|
|
|
|
# none are explicitly given
|
|
|
|
|
|
|
|
if( defined $options{image} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
# images generally have the origin at the top-left instead of the
|
|
|
|
|
|
|
|
# bottom-left, so given nothing else, I flip the y axis
|
|
|
|
|
|
|
|
if( !defined $options{ymin} && !defined $options{ymax} &&
|
|
|
|
|
|
|
|
! any { /^ *yrange\b/ } @{$options{set}} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
push @{$options{set}}, "yrange [:] reverse";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if ( ! -r $options{image} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
die "Couldn't read image '$options{image}'";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unshift @{$options{equation}}, qq{"$options{image}" binary filetype=auto flipy with rgbimage title "$options{image}"};
|
|
|
|
|
|
|
|
delete $options{image};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub getGnuplotVersion
|
|
|
|
sub getGnuplotVersion
|
|
|
@@ -447,16 +531,66 @@ sub makeDomainNumeric
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $prev_timed_replot_time = [gettimeofday];
|
|
|
|
|
|
|
|
my $pipe_in;
|
|
|
|
|
|
|
|
my $selector;
|
|
|
|
|
|
|
|
my $line_number = 0;
|
|
|
|
|
|
|
|
my $is_stdin = !@ARGV; # read stdin only if no data files given on the cmdline
|
|
|
|
|
|
|
|
sub openNextFile
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
my $fd;
|
|
|
|
|
|
|
|
if($is_stdin)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
$fd = IO::Handle->new();
|
|
|
|
|
|
|
|
$fd->fdopen(fileno(STDIN), "r") or die "Couldn't open STDIN";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
my $filename = shift @ARGV;
|
|
|
|
|
|
|
|
$fd = IO::File->new($filename, "r") or die "Couldn't open file '$filename'";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $selector = IO::Select->new( $fd );
|
|
|
|
|
|
|
|
return ($fd, $selector);
|
|
|
|
|
|
|
|
}
|
|
|
|
sub getNextLine
|
|
|
|
sub getNextLine
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
sub getline_internal
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
my $line = $pipe_in->getline();
|
|
|
|
|
|
|
|
if( !$is_stdin && !defined $line && $pipe_in->eof() && @ARGV)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
# I got to the end of one file, so open the next one (which I'm
|
|
|
|
|
|
|
|
# sure exists)
|
|
|
|
|
|
|
|
($pipe_in, $selector) = openNextFile();
|
|
|
|
|
|
|
|
next;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return $line;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if( !defined $pipe_in )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
($pipe_in, $selector) = openNextFile();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
$this_replot_is_from_timer = undef;
|
|
|
|
$this_replot_is_from_timer = undef;
|
|
|
|
|
|
|
|
|
|
|
|
# if we're not streaming, or we're doing triggered-only replotting, simply
|
|
|
|
# if we're not streaming, or we're doing triggered-only replotting, simply
|
|
|
|
# do a blocking read
|
|
|
|
# do a blocking read
|
|
|
|
return $stdin->getline()
|
|
|
|
if (! $options{stream} || $options{stream} < 0)
|
|
|
|
if (! $options{stream} || $options{stream} < 0);
|
|
|
|
{
|
|
|
|
|
|
|
|
$line_number++;
|
|
|
|
|
|
|
|
return getline_internal();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $now = [gettimeofday];
|
|
|
|
my $now = [gettimeofday];
|
|
|
@@ -471,7 +605,8 @@ sub getNextLine
|
|
|
|
|
|
|
|
|
|
|
|
if ($selector->can_read($time_remaining))
|
|
|
|
if ($selector->can_read($time_remaining))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return $stdin->getline();
|
|
|
|
$line_number++;
|
|
|
|
|
|
|
|
return getline_internal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@@ -622,11 +757,6 @@ sub mainThread
|
|
|
|
print(PIPE "set xdata time\n");
|
|
|
|
print(PIPE "set xdata time\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# add the extra global options
|
|
|
|
|
|
|
|
print(PIPE "$_\n") foreach (@{$options{extracmds}});
|
|
|
|
|
|
|
|
print(PIPE "set $_\n") foreach (@{$options{set}});
|
|
|
|
|
|
|
|
print(PIPE "unset $_\n") foreach (@{$options{unset}});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# set up histograms
|
|
|
|
# set up histograms
|
|
|
|
$options{binwidth} ||= 1; # if no binwidth given, set it to 1
|
|
|
|
$options{binwidth} ||= 1; # if no binwidth given, set it to 1
|
|
|
|
print PIPE
|
|
|
|
print PIPE
|
|
|
@@ -649,6 +779,11 @@ sub mainThread
|
|
|
|
sendRangeCommand( "zrange", $options{zmin}, $options{zmax} );
|
|
|
|
sendRangeCommand( "zrange", $options{zmin}, $options{zmax} );
|
|
|
|
sendRangeCommand( "cbrange", $options{zmin}, $options{zmax} ) if($options{colormap});
|
|
|
|
sendRangeCommand( "cbrange", $options{zmin}, $options{zmax} ) if($options{colormap});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# add the extra global options
|
|
|
|
|
|
|
|
print(PIPE "$_\n") foreach (@{$options{extracmds}});
|
|
|
|
|
|
|
|
print(PIPE "set $_\n") foreach (@{$options{set}});
|
|
|
|
|
|
|
|
print(PIPE "unset $_\n") foreach (@{$options{unset}});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -658,6 +793,41 @@ sub mainThread
|
|
|
|
# The domain of the current point
|
|
|
|
# The domain of the current point
|
|
|
|
my @domain;
|
|
|
|
my @domain;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# column headers from vnlog
|
|
|
|
|
|
|
|
my @vnlog_headers;
|
|
|
|
|
|
|
|
if($options{vnlog})
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
require Vnlog::Parser;
|
|
|
|
|
|
|
|
require Vnlog::Util;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if ( !defined $pipe_in )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
($pipe_in, $selector) = openNextFile();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $parser = Vnlog::Parser->new();
|
|
|
|
|
|
|
|
while (defined ($_ = Vnlog::Util::get_unbuffered_line($pipe_in)))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if ( !$parser->parse($_) )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
die "Error parsing vnlog: $parser->{error}; looking at line '$_'";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my $keys = $parser->getKeys();
|
|
|
|
|
|
|
|
if (defined $keys)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
@vnlog_headers = @$keys;
|
|
|
|
|
|
|
|
last;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!@vnlog_headers)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
die "Looked through all of the first file, and never saw a vnlog legend";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# The x-axis domain represented as a number. This is exactly the same as
|
|
|
|
# The x-axis domain represented as a number. This is exactly the same as
|
|
|
|
# $domain[0] unless the x-axis domain uses a timefmt. Then this is the
|
|
|
|
# $domain[0] unless the x-axis domain uses a timefmt. Then this is the
|
|
|
|
# number of seconds since the UNIX epoch.
|
|
|
|
# number of seconds since the UNIX epoch.
|
|
|
@@ -741,28 +911,35 @@ sub mainThread
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
$domain[0] = $.;
|
|
|
|
$domain[0] = $line_number;
|
|
|
|
$domain0_numeric = makeDomainNumeric( $domain[0] );
|
|
|
|
$domain0_numeric = makeDomainNumeric( $domain[0] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
my $id = -1;
|
|
|
|
my $id = -1;
|
|
|
|
|
|
|
|
my $i_curve = 0;
|
|
|
|
while(@fields)
|
|
|
|
while(@fields)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if($options{dataid})
|
|
|
|
if ($options{dataid})
|
|
|
|
{
|
|
|
|
{
|
|
|
|
$id = shift @fields;
|
|
|
|
$id = shift @fields;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
elsif($options{vnlog} )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if( $icurve >= @vnlog_headers )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
# Got more columns than vnlog headers. The data is probably
|
|
|
|
|
|
|
|
# bogus, but I don't want to barf at the user, so I silently
|
|
|
|
|
|
|
|
# ignore the data
|
|
|
|
|
|
|
|
last;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
$id = $vnlog_headers[$i_curve];
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
$id++;
|
|
|
|
$id++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# I'd like to use //, but I guess some people are still on perl 5.8
|
|
|
|
my $rangesize = getRangeSize($id);
|
|
|
|
my $rangesize = exists $options{rangesize_hash}{$id} ?
|
|
|
|
|
|
|
|
$options{rangesize_hash}{$id} :
|
|
|
|
|
|
|
|
$options{rangesize_default};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
last if @fields < $rangesize;
|
|
|
|
last if @fields < $rangesize;
|
|
|
|
|
|
|
|
|
|
|
|
pushPoint(getCurve($id),
|
|
|
|
pushPoint(getCurve($id),
|
|
|
@@ -770,6 +947,8 @@ sub mainThread
|
|
|
|
@domain,
|
|
|
|
@domain,
|
|
|
|
splice( @fields, 0, $rangesize ) ) . "\n",
|
|
|
|
splice( @fields, 0, $rangesize ) ) . "\n",
|
|
|
|
$domain0_numeric);
|
|
|
|
$domain0_numeric);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$i_curve++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@@ -886,17 +1065,13 @@ sub updateCurveOptions
|
|
|
|
# as 1 + rangesize). I also need to start the range at the first column
|
|
|
|
# as 1 + rangesize). I also need to start the range at the first column
|
|
|
|
# past the timefmt
|
|
|
|
# past the timefmt
|
|
|
|
|
|
|
|
|
|
|
|
# I'd like to use //, but I guess some people are still on perl 5.8
|
|
|
|
my @rest = map {$_ + $options{timefmt_Ncols}} (1..getRangeSize($id));
|
|
|
|
my $rangesize = exists $options{rangesize_hash}{$id} ?
|
|
|
|
|
|
|
|
$options{rangesize_hash}{$id} :
|
|
|
|
|
|
|
|
$options{rangesize_default};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
my @rest = map {$_ + $options{timefmt_Ncols}} (1..$rangesize);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$usingoptions = "using 1:" . join(':', @rest);
|
|
|
|
$usingoptions = "using 1:" . join(':', @rest);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$curve->{options} = "$histoptions $usingoptions $titleoption $curve->{extraoptions} $options{curvestyleall}";
|
|
|
|
|
|
|
|
|
|
|
|
$curve->{options} = "$histoptions $usingoptions $titleoption $curve->{extraoptions}";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub getCurve
|
|
|
|
sub getCurve
|
|
|
@@ -915,13 +1090,33 @@ sub getCurve
|
|
|
|
|
|
|
|
|
|
|
|
if( !exists $curveIndices{$id} )
|
|
|
|
if( !exists $curveIndices{$id} )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
push @curves, {extraoptions => ' ',
|
|
|
|
push @curves, {# if we have a catch-all style and no specific style, use
|
|
|
|
|
|
|
|
# the catch-all style
|
|
|
|
|
|
|
|
extraoptions => (!exists $options{curvestyle_hash}{$id} &&
|
|
|
|
|
|
|
|
exists $options{curvestyleall}) ?
|
|
|
|
|
|
|
|
"$options{curvestyleall} " : ' ',
|
|
|
|
datastring => '',
|
|
|
|
datastring => '',
|
|
|
|
datastring_meta => [],
|
|
|
|
datastring_meta => [],
|
|
|
|
datastring_offset => 0}; # push a curve with no data and no options
|
|
|
|
datastring_offset => 0}; # push a curve with no data and no options
|
|
|
|
$curveIndices{$id} = $#curves;
|
|
|
|
$curveIndices{$id} = $#curves;
|
|
|
|
|
|
|
|
|
|
|
|
updateCurveOptions($curves[$#curves], $id);
|
|
|
|
updateCurveOptions($curves[$#curves], $id);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# --xlen has a meaning if we're not plotting histograms at all or if we're
|
|
|
|
|
|
|
|
# plotting ONLY histograms. If we're doing both at the same time, there's no
|
|
|
|
|
|
|
|
# consistent way to assign meaning to xlen
|
|
|
|
|
|
|
|
if ( defined $options{xlen} &&
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# have at least some histograms
|
|
|
|
|
|
|
|
@{$options{histogram}} &&
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# there are more curves than histogram curves, i.e. there're some
|
|
|
|
|
|
|
|
# non-histogram curves
|
|
|
|
|
|
|
|
@curves > @{$options{histogram}} ) {
|
|
|
|
|
|
|
|
print STDERR "--xlen only makes sense when plotting ONLY histograms or ONLY NON-histograms\n";
|
|
|
|
|
|
|
|
exit -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $curves[$curveIndices{$id}];
|
|
|
|
return $curves[$curveIndices{$id}];
|
|
|
|
}
|
|
|
|
}
|
|
|
@@ -949,6 +1144,10 @@ sub setCurveAsHistogram
|
|
|
|
my ($id, $str) = @_;
|
|
|
|
my ($id, $str) = @_;
|
|
|
|
|
|
|
|
|
|
|
|
my $curve = getCurve($id);
|
|
|
|
my $curve = getCurve($id);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# With histograms I have 2d plots with rangesize=1. I thus give gnuplot two
|
|
|
|
|
|
|
|
# values for each point: a domain and a range. For histograms I ignore the
|
|
|
|
|
|
|
|
# domain, so I get the statistics of the 2nd column: $2
|
|
|
|
$curve->{histoptions} = 'using (histbin($2)):(1.0) smooth ' . $options{histstyle};
|
|
|
|
$curve->{histoptions} = 'using (histbin($2)):(1.0) smooth ' . $options{histstyle};
|
|
|
|
|
|
|
|
|
|
|
|
updateCurveOptions($curve, $id);
|
|
|
|
updateCurveOptions($curve, $id);
|
|
|
@@ -1016,7 +1215,11 @@ sub replot
|
|
|
|
# seconds-since-the-epoch BACK to the timefmt. Sheesh
|
|
|
|
# seconds-since-the-epoch BACK to the timefmt. Sheesh
|
|
|
|
($xmin, $xmax) = map {Time::Piece->strptime( $_, '%s' )->strftime( $options{timefmt} ) } ($xmin, $xmax);
|
|
|
|
($xmin, $xmax) = map {Time::Piece->strptime( $_, '%s' )->strftime( $options{timefmt} ) } ($xmin, $xmax);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sendRangeCommand( "xrange", $xmin, $xmax );
|
|
|
|
|
|
|
|
|
|
|
|
# if we have any histograms, then I'm not really visualizing the domain at
|
|
|
|
|
|
|
|
# all, and I don't set the range.
|
|
|
|
|
|
|
|
sendRangeCommand( "xrange", $xmin, $xmax )
|
|
|
|
|
|
|
|
unless @{$options{histogram}};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
plotStoredData();
|
|
|
|
plotStoredData();
|
|
|
@@ -1041,6 +1244,9 @@ sub pushPoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mainThread();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
|
|
|
|
feedgnuplot - General purpose pipe-oriented plotting tool
|
|
|
|
feedgnuplot - General purpose pipe-oriented plotting tool
|
|
|
@@ -1058,45 +1264,45 @@ Simple plotting of piped data:
|
|
|
|
|
|
|
|
|
|
|
|
$ seq 5 | awk '{print 2*$1, $1*$1}' |
|
|
|
|
$ seq 5 | awk '{print 2*$1, $1*$1}' |
|
|
|
|
feedgnuplot --lines --points --legend 0 "data 0" --title "Test plot" --y2 1
|
|
|
|
feedgnuplot --lines --points --legend 0 "data 0" --title "Test plot" --y2 1
|
|
|
|
--terminal 'dumb 80,40' --exit
|
|
|
|
--unset grid --terminal 'dumb 80,40' --exit
|
|
|
|
|
|
|
|
|
|
|
|
Test plot
|
|
|
|
Test plot
|
|
|
|
|
|
|
|
|
|
|
|
10 ++------+--------+-------+-------+-------+--------+-------+------*A 25
|
|
|
|
10 +-----------------------------------------------------------------+ 25
|
|
|
|
+ + + + + + + + **#+
|
|
|
|
| + + + + + + + *##|
|
|
|
|
| : : : : : : data 0+**A*** |
|
|
|
|
| data 0 ***A*#* |
|
|
|
|
| : : : : : : :** # |
|
|
|
|
| ** # |
|
|
|
|
9 ++.......................................................**.##....|
|
|
|
|
9 |-+ ** ## |
|
|
|
|
| : : : : : : ** :# |
|
|
|
|
| ** # |
|
|
|
|
| : : : : : : ** # |
|
|
|
|
| ** # |
|
|
|
|
| : : : : : :** ##: ++ 20
|
|
|
|
| ** ## +-| 20
|
|
|
|
8 ++................................................A....#..........|
|
|
|
|
8 |-+ A # |
|
|
|
|
| : : : : : **: # : |
|
|
|
|
| ** # |
|
|
|
|
| : : : : : ** : ## : |
|
|
|
|
| ** ## |
|
|
|
|
| : : : : : ** :# : |
|
|
|
|
| ** # |
|
|
|
|
| : : : : :** B : |
|
|
|
|
| ** B |
|
|
|
|
7 ++......................................**......##................|
|
|
|
|
7 |-+ ** ## |
|
|
|
|
| : : : : ** : ## : : ++ 15
|
|
|
|
| ** ## +-| 15
|
|
|
|
| : : : : ** : # : : |
|
|
|
|
| ** # |
|
|
|
|
| : : : :** : ## : : |
|
|
|
|
| ** ## |
|
|
|
|
6 ++..............................*A.......##.......................|
|
|
|
|
6 |-+ *A ## |
|
|
|
|
| : : : ** : ##: : : |
|
|
|
|
| ** ## |
|
|
|
|
| : : : ** : # : : : |
|
|
|
|
| ** # |
|
|
|
|
| : : :** : ## : : : ++ 10
|
|
|
|
| ** ## +-| 10
|
|
|
|
5 ++......................**........##..............................|
|
|
|
|
5 |-+ ** ## |
|
|
|
|
| : : ** : #B : : : |
|
|
|
|
| ** #B |
|
|
|
|
| : : ** : ## : : : : |
|
|
|
|
| ** ## |
|
|
|
|
| : :** : ## : : : : |
|
|
|
|
| ** ## |
|
|
|
|
4 ++...............A.......###......................................|
|
|
|
|
4 |-+ A ### |
|
|
|
|
| : **: ##: : : : : |
|
|
|
|
| ** ## |
|
|
|
|
| : ** : ## : : : : : ++ 5
|
|
|
|
| ** ## +-| 5
|
|
|
|
| : ** : ## : : : : : |
|
|
|
|
| ** ## |
|
|
|
|
| :** ##B# : : : : : |
|
|
|
|
| ** ##B# |
|
|
|
|
3 ++.....**..####...................................................|
|
|
|
|
3 |-+ ** #### |
|
|
|
|
| **#### : : : : : : |
|
|
|
|
| **#### |
|
|
|
|
| **## : : : : : : : |
|
|
|
|
| #### |
|
|
|
|
B** + + + + + + + +
|
|
|
|
|## + + + + + + + |
|
|
|
|
2 A+------+--------+-------+-------+-------+--------+-------+------++ 0
|
|
|
|
2 +-----------------------------------------------------------------+ 0
|
|
|
|
1 1.5 2 2.5 3 3.5 4 4.5 5
|
|
|
|
1 1.5 2 2.5 3 3.5 4 4.5 5
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -1164,41 +1370,64 @@ with the I<X>-value at the start of that line.
|
|
|
|
|
|
|
|
|
|
|
|
=head3 Curve indexing
|
|
|
|
=head3 Curve indexing
|
|
|
|
|
|
|
|
|
|
|
|
By default, each column represents a separate curve. This is fine unless sparse
|
|
|
|
We index the curves in one of 3 ways: sequentially, explicitly with a
|
|
|
|
data is to be plotted. With the C<--dataid> option, each point is represented by
|
|
|
|
C<--dataid> or by C<--vnlog> headers.
|
|
|
|
2 values: a string identifying the curve, and the value itself. If we add
|
|
|
|
|
|
|
|
C<--dataid> to the original example:
|
|
|
|
By default, each column represents a separate curve. The first column (after any
|
|
|
|
|
|
|
|
domain) is curve C<0>. The next one is curve C<1> and so on. This is fine unless
|
|
|
|
|
|
|
|
sparse data is to be plotted. With the C<--dataid> option, each point is
|
|
|
|
|
|
|
|
represented by 2 values: a string identifying the curve, and the value itself.
|
|
|
|
|
|
|
|
If we add C<--dataid> to the original example:
|
|
|
|
|
|
|
|
|
|
|
|
$ seq 5 | awk '{print 2*$1, $1*$1}' | feedgnuplot --dataid --autolegend
|
|
|
|
$ seq 5 | awk '{print 2*$1, $1*$1}' | feedgnuplot --dataid --autolegend
|
|
|
|
|
|
|
|
|
|
|
|
we get 5 different curves with one point in each. The first column, as produced
|
|
|
|
we get 5 different curves with one point in each. The first column, as produced
|
|
|
|
by C<awk>, is B<2,4,6,8,10>. These are interpreted as the IDs of the curves to
|
|
|
|
by C<awk>, is B<2,4,6,8,10>. These are interpreted as the IDs of the curves to
|
|
|
|
be plotted. The C<--autolegend> option adds a legend using the given IDs to
|
|
|
|
be plotted.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
If we're plotting C<vnlog> data (L<https://www.github.com/dkogan/vnlog>) then we
|
|
|
|
|
|
|
|
can get the curve IDs from the vnlog header. Vnlog is a trivial data format
|
|
|
|
|
|
|
|
where lines starting with C<#> are comments and the first comment contains
|
|
|
|
|
|
|
|
column labels. If we have such data, C<feedgnuplot --vnlog> can interpret these
|
|
|
|
|
|
|
|
column labels if the C<vnlog> perl modules are available.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The C<--autolegend> option adds a legend using the given IDs to
|
|
|
|
label the curves. The IDs need not be numbers; generic strings are accepted. As
|
|
|
|
label the curves. The IDs need not be numbers; generic strings are accepted. As
|
|
|
|
many points as desired can appear on a single line. C<--domain> can be used in
|
|
|
|
many points as desired can appear on a single line. C<--domain> can be used in
|
|
|
|
conjunction with C<--dataid>.
|
|
|
|
conjunction with C<--dataid> or C<--vnlog>.
|
|
|
|
|
|
|
|
|
|
|
|
=head3 Multi-value style support
|
|
|
|
=head3 Multi-value style support
|
|
|
|
|
|
|
|
|
|
|
|
Depending on how gnuplot is plotting the data, more than one value may be needed
|
|
|
|
Depending on how gnuplot is plotting the data, more than one value may be needed
|
|
|
|
to represent the range of a single point. Basic 2D plots have 2 numbers
|
|
|
|
to represent the range of a single point. Basic 2D plots have 2 numbers
|
|
|
|
representing each point: 1 domain and 1 range. But if plotting with
|
|
|
|
representing each point: 1 domain and 1 range. But if plotting with
|
|
|
|
C<--circles>, for instance, then there's an extra range value: the radius. A
|
|
|
|
C<--circles>, for instance, then there's an extra range value: the radius. Many
|
|
|
|
similar situation exists with C<--colormap> where each point contains the
|
|
|
|
other gnuplot styles require more data: errorbars, variable colors (C<with
|
|
|
|
position I<and> the color. There are other gnuplot styles that require more data
|
|
|
|
points palette>), variable sizes (C<with points ps variable>), labels and so on.
|
|
|
|
(such as error bars), but none of these are directly supported by the script.
|
|
|
|
The feedgnuplot tool itself does not know about all these intricacies, but they
|
|
|
|
They can still be used, however, by specifying the specific style with
|
|
|
|
can still be used, by specifying the specific style with C<--style>, and
|
|
|
|
C<--style>, and specifying how many values are needed for each point with
|
|
|
|
specifying how many values are needed for each point with any of
|
|
|
|
C<--rangesizeall> or C<--rangesize> or C<--extraValuesPerPoint>. Those options
|
|
|
|
C<--rangesizeall, C<--tuplesizeall>, C<--rangesize>, C<--tuplesize>. These
|
|
|
|
that specify the range size are required I<only> for styles not explicitly
|
|
|
|
options are required I<only> for styles not explicitly supported by feedgnuplot;
|
|
|
|
supported by feedgnuplot; supported styles do the right thing automatically.
|
|
|
|
supported styles do the right thing automatically.
|
|
|
|
|
|
|
|
|
|
|
|
More examples: if making a 2d plot of y error bars where gnuplot expects a
|
|
|
|
Specific example: if making a 2d plot of y error bars, the exact format can be
|
|
|
|
(x,y,ydelta) tuple for each point, you want C<--rangesizeall 2> because you have
|
|
|
|
queried by running C<gnuplot> and invoking C<help yerrorbars>. This tells us
|
|
|
|
one domain value (x) and 2 range values (y,ydelta). Gnuplot can also plot
|
|
|
|
that there's a 3-column form: C<x y ydelta> and a 4-column form: C<x y ylow
|
|
|
|
lopsided y errorbars by giving a tuple (x,y,ylow,yhigh). This is similar as
|
|
|
|
yhigh>. With 2d plots feedgnuplot will always output the 1-value domain C<x>, so
|
|
|
|
before, but you want C<--rangesizeall 3> instead.
|
|
|
|
the rangesize is 2 and 3 respectively. Thus the following are equivalent:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ echo '1 2 0.3
|
|
|
|
|
|
|
|
2 3 0.4
|
|
|
|
|
|
|
|
3 4 0.5' | feedgnuplot --domain --rangesizeall 2 --with 'yerrorbars'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ echo '1 2 0.3
|
|
|
|
|
|
|
|
2 3 0.4
|
|
|
|
|
|
|
|
3 4 0.5' | feedgnuplot --domain --tuplesizeall 3 --with 'yerrorbars'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ echo '1 2 1.7 2.3
|
|
|
|
|
|
|
|
2 3 2.6 3.4
|
|
|
|
|
|
|
|
3 4 3.5 4.5' | feedgnuplot --domain --rangesizeall 3 --with 'yerrorbars'
|
|
|
|
|
|
|
|
|
|
|
|
=head3 3D data
|
|
|
|
=head3 3D data
|
|
|
|
|
|
|
|
|
|
|
@@ -1265,7 +1494,12 @@ windowsize> can be given. This will create an constantly-updating, scrolling
|
|
|
|
view of the recent past. C<windowsize> should be replaced by the desired length
|
|
|
|
view of the recent past. C<windowsize> should be replaced by the desired length
|
|
|
|
of the domain window to plot, in domain units (passed-in values if C<--domain>
|
|
|
|
of the domain window to plot, in domain units (passed-in values if C<--domain>
|
|
|
|
or line numbers otherwise). If the domain is a time/date via C<--timefmt>, then
|
|
|
|
or line numbers otherwise). If the domain is a time/date via C<--timefmt>, then
|
|
|
|
C<windowsize> is and I<integer> in seconds.
|
|
|
|
C<windowsize> is and I<integer> in seconds. If we're plotting a histogram, then
|
|
|
|
|
|
|
|
C<--xlen> causes a histogram over a moving window to be computed. The subtlely
|
|
|
|
|
|
|
|
here is that with a histogram you don't actually I<see> the domain since only
|
|
|
|
|
|
|
|
the range is analyzed. But the domain is still there, and can be utilized with
|
|
|
|
|
|
|
|
C<--xlen>. With C<--xlen> we can plot I<only> histograms or I<only>
|
|
|
|
|
|
|
|
I<non>-histograms.
|
|
|
|
|
|
|
|
|
|
|
|
=head3 Special data commands
|
|
|
|
=head3 Special data commands
|
|
|
|
|
|
|
|
|
|
|
@@ -1427,6 +1661,19 @@ point in curve ID 20
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
C<--vnlog>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Vnlog is a trivial data format where lines starting with C<#> are comments and
|
|
|
|
|
|
|
|
the first comment contains column labels. Some tools for working with such data
|
|
|
|
|
|
|
|
are available from the C<vnlog> project: L<https://www.github.com/dkogan/vnlog>.
|
|
|
|
|
|
|
|
With the C<vnlog> perl modules installed, we can read the vnlog column headers
|
|
|
|
|
|
|
|
with C<feedgnuplot --vnlog>. This replaces C<--dataid>, and we can do all the
|
|
|
|
|
|
|
|
normal things with these headers. For instance C<feedgnuplot --vnlog
|
|
|
|
|
|
|
|
--autolegend> will generate plot legends for each column in the vnlog, using the
|
|
|
|
|
|
|
|
vnlog column label in the legend.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
C<--[no]3d>
|
|
|
|
C<--[no]3d>
|
|
|
|
|
|
|
|
|
|
|
|
Do [not] plot in 3D. This only makes sense with C<--domain>. Each domain here is
|
|
|
|
Do [not] plot in 3D. This only makes sense with C<--domain>. Each domain here is
|
|
|
@@ -1443,7 +1690,8 @@ Interpret the X data as a time/date, parsed with the given format
|
|
|
|
C<--colormap>
|
|
|
|
C<--colormap>
|
|
|
|
|
|
|
|
|
|
|
|
Show a colormapped xy plot. Requires extra data for the color. zmin/zmax can be
|
|
|
|
Show a colormapped xy plot. Requires extra data for the color. zmin/zmax can be
|
|
|
|
used to set the extents of the colors. Automatically sets the C<--rangesize>.
|
|
|
|
used to set the extents of the colors. Automatically sets the
|
|
|
|
|
|
|
|
C<--rangesize>/C<--tuplesize>.
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1471,7 +1719,8 @@ Do [not] draw points
|
|
|
|
C<--circles>
|
|
|
|
C<--circles>
|
|
|
|
|
|
|
|
|
|
|
|
Plot with circles. This requires a radius be specified for each point.
|
|
|
|
Plot with circles. This requires a radius be specified for each point.
|
|
|
|
Automatically sets the C<--rangesize>. C<Not> supported for 3d plots.
|
|
|
|
Automatically sets the C<--rangesize>/C<--tuplesize>. C<Not> supported for 3d
|
|
|
|
|
|
|
|
plots.
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1499,7 +1748,12 @@ C<--xlen xxx>
|
|
|
|
|
|
|
|
|
|
|
|
When using C<--stream>, sets the size of the x-window to plot. Omit this or set
|
|
|
|
When using C<--stream>, sets the size of the x-window to plot. Omit this or set
|
|
|
|
it to 0 to plot ALL the data. Does not make sense with 3d plots. Implies
|
|
|
|
it to 0 to plot ALL the data. Does not make sense with 3d plots. Implies
|
|
|
|
C<--monotonic>
|
|
|
|
C<--monotonic>. If we're plotting a histogram, then C<--xlen> causes a histogram
|
|
|
|
|
|
|
|
over a moving window to be computed. The subtlely here is that with a histogram
|
|
|
|
|
|
|
|
you don't actually I<see> the domain since only the range is analyzed. But the
|
|
|
|
|
|
|
|
domain is still there, and can be utilized with C<--xlen>. With C<--xlen> we can
|
|
|
|
|
|
|
|
plot I<only> histograms or I<only> I<non>-histograms.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1507,7 +1761,10 @@ C<--xmin/xmax/ymin/ymax/y2min/y2max/zmin/zmax xxx>
|
|
|
|
|
|
|
|
|
|
|
|
Set the range for the given axis. These x-axis bounds are ignored in a streaming
|
|
|
|
Set the range for the given axis. These x-axis bounds are ignored in a streaming
|
|
|
|
plot. The y2-axis bound do not apply in 3d plots. The z-axis bounds apply
|
|
|
|
plot. The y2-axis bound do not apply in 3d plots. The z-axis bounds apply
|
|
|
|
I<only> to 3d plots or colormaps.
|
|
|
|
I<only> to 3d plots or colormaps. Note that there is no C<--xrange> to set both
|
|
|
|
|
|
|
|
sides at once or C<--xinv> to flip the axis around: anything more than the
|
|
|
|
|
|
|
|
basics supported in this option is clearly obtainable by talking to gnuplot, for
|
|
|
|
|
|
|
|
instance C<--set 'xrange [20:10]'> to set the given inverted bounds.
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1541,8 +1798,9 @@ specified for this curve (C<--curvestyle>) or all curves (C<--with>,
|
|
|
|
C<--curvestyleall>) then the default histogram style is set: filled boxes with
|
|
|
|
C<--curvestyleall>) then the default histogram style is set: filled boxes with
|
|
|
|
borders. This is what the user generally wants. This works with C<--domain>
|
|
|
|
borders. This is what the user generally wants. This works with C<--domain>
|
|
|
|
and/or C<--stream>, but in those cases the x-value is used I<only> to cull old
|
|
|
|
and/or C<--stream>, but in those cases the x-value is used I<only> to cull old
|
|
|
|
data because of C<--xlen> or C<--monotonic>. I.e. the x-values are I<not> drawn
|
|
|
|
data because of C<--xlen> or C<--monotonic>. I.e. the domain values are I<not>
|
|
|
|
in any way. Can be passed multiple times, or passed a comma- separated list
|
|
|
|
drawn in any way. Can be passed multiple times, or passed a comma- separated
|
|
|
|
|
|
|
|
list
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1631,15 +1889,26 @@ times.
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
C<--image filename>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Overlays the data on top of a raster image given in C<filename>. This is passed
|
|
|
|
|
|
|
|
through to gnuplot via C<--equation>, and is not interpreted by C<feedgnuplot>
|
|
|
|
|
|
|
|
other than checking for existence. Usually images have their origin at the
|
|
|
|
|
|
|
|
top-left corner, while plots have it in the bottom-left corner instead. Thus if
|
|
|
|
|
|
|
|
the y-axis extents are not specified (C<--ymin>, C<--ymax>, C<--set 'yrange
|
|
|
|
|
|
|
|
...'>) this option will also flip around the y axis to make the image appear
|
|
|
|
|
|
|
|
properly. Since this option is just a passthrough to gnuplot, finer control can
|
|
|
|
|
|
|
|
be achieved by passing in C<--equation> and C<--set yrange ...> directly.
|
|
|
|
|
|
|
|
|
|
|
|
C<--equation xxx>
|
|
|
|
C<--equation xxx>
|
|
|
|
|
|
|
|
|
|
|
|
Gnuplot can plot both data and symbolic equations. C<feedgnuplot> generally
|
|
|
|
Gnuplot can plot both data and symbolic equations. C<feedgnuplot> generally
|
|
|
|
plots data, but with this option can plot symbolic equations I<also>. This is
|
|
|
|
plots data, but with this option can plot symbolic equations I<also>. This is
|
|
|
|
generally intended to augment data plots, since for equation-only plots you
|
|
|
|
generally intended to augment data plots, since for equation-only plots you
|
|
|
|
don't need C<feedgnuplot>. C<--equation> can be passed multiple times for
|
|
|
|
don't need C<feedgnuplot>. C<--equation> can be passed multiple times for
|
|
|
|
multiple equations. The given strings are passed to gnuplot directly without any
|
|
|
|
multiple equations. The given strings are passed to gnuplot directly without
|
|
|
|
thing added or removed, so styling and such should be applied in the string. A
|
|
|
|
anything added or removed, so styling and such should be applied in the string.
|
|
|
|
basic example:
|
|
|
|
A basic example:
|
|
|
|
|
|
|
|
|
|
|
|
seq 100 | awk '{print $1/10, $1/100}' |
|
|
|
|
seq 100 | awk '{print $1/10, $1/100}' |
|
|
|
|
feedgnuplot --with 'lines lw 3' --domain --ymax 1
|
|
|
|
feedgnuplot --with 'lines lw 3' --domain --ymax 1
|
|
|
@@ -1689,7 +1958,7 @@ file type is desired, use both C<--hardcopy> and C<--terminal>
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
C<--maxcurves xxx>
|
|
|
|
C<--maxcurves N>
|
|
|
|
|
|
|
|
|
|
|
|
The maximum allowed number of curves. This is 100 by default, but can be reset
|
|
|
|
The maximum allowed number of curves. This is 100 by default, but can be reset
|
|
|
|
with this option. This exists purely to prevent perl from allocating all of the
|
|
|
|
with this option. This exists purely to prevent perl from allocating all of the
|
|
|
@@ -1699,20 +1968,22 @@ system's memory when reading bogus data
|
|
|
|
|
|
|
|
|
|
|
|
C<--monotonic>
|
|
|
|
C<--monotonic>
|
|
|
|
|
|
|
|
|
|
|
|
If C<--domain> is given, checks to make sure that the x- coordinate in the input
|
|
|
|
If C<--domain> is given, checks to make sure that the x-coordinate in the input
|
|
|
|
data is monotonically increasing. If a given x-variable is in the past, all data
|
|
|
|
data is monotonically increasing. If a given x-variable is in the past, all data
|
|
|
|
currently cached for this curve is purged. Without C<--monotonic>, all data is
|
|
|
|
currently cached for this curve is purged. Without C<--monotonic>, all data is
|
|
|
|
kept. Does not make sense with 3d plots. No C<--monotonic> by default. The data is
|
|
|
|
kept. Does not make sense with 3d plots. No C<--monotonic> by default. The data
|
|
|
|
replotted before being purged
|
|
|
|
is replotted before being purged. This is useful in streaming plots where the
|
|
|
|
|
|
|
|
incoming data represents multiple iterations of the same process (repeated
|
|
|
|
|
|
|
|
simulations of the same period in time, for instance).
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
C<--rangesize curveID xxx>
|
|
|
|
C<--rangesize curveID N>
|
|
|
|
|
|
|
|
|
|
|
|
The options C<--rangesizeall>, C<--rangesize> and C<--extraValuesPerPoint> set
|
|
|
|
The options C<--rangesizeall> and C<--rangesize> set the number of values are
|
|
|
|
the number of values are needed to represent each point being plotted (see
|
|
|
|
needed to represent each point being plotted (see L</"Multi-value style
|
|
|
|
L</"Multi-value style support"> above). These options are I<only> needed if
|
|
|
|
support"> above). These options are I<only> needed if unknown styles are used,
|
|
|
|
unknown styles are used, with C<--styleall> or C<--with> for instance.
|
|
|
|
with C<--styleall> or C<--with> for instance.
|
|
|
|
|
|
|
|
|
|
|
|
C<--rangesize> is used to set how many values are needed to represent the range
|
|
|
|
C<--rangesize> is used to set how many values are needed to represent the range
|
|
|
|
of a point for a particular curve. This overrides any defaults that may exist
|
|
|
|
of a point for a particular curve. This overrides any defaults that may exist
|
|
|
@@ -1724,19 +1995,25 @@ rangesize should apply.
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
C<--rangesizeall xxx>
|
|
|
|
C<--tuplesize curveID N>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Very similar to C<--rangesize>, but instead of specifying the I<range> only,
|
|
|
|
|
|
|
|
this specifies the whole tuple. For instance if we're plotting circles, the
|
|
|
|
|
|
|
|
tuplesize is 3: C<x,y,radius>. In a 2D plot there's a 1-dimensional domain:
|
|
|
|
|
|
|
|
C<x>, so the rangesize is 2: C<y,radius>. This dimensionality can be given
|
|
|
|
|
|
|
|
either way.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
C<--rangesizeall N>
|
|
|
|
|
|
|
|
|
|
|
|
Like C<--rangesize>, but applies to I<all> the curves.
|
|
|
|
Like C<--rangesize>, but applies to I<all> the curves.
|
|
|
|
|
|
|
|
|
|
|
|
C<--extraValuesPerPoint xxx>
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
|
Like C<--rangesizeall>, but instead of overriding the default, adds to it. For
|
|
|
|
C<--tuplesizeall N>
|
|
|
|
example, if plotting non-lopsided y errorbars gnuplot wants (x,y,ydelta) tuples.
|
|
|
|
|
|
|
|
These can be specified both with C<--rangesizeall 2> (because there are 2 range
|
|
|
|
|
|
|
|
values) or C<--extraValuesPerPoint 1> (because there's 1 more value than usual).
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
This option is I<only> needed if unknown styles are used, with C<--styleall> or
|
|
|
|
Like C<--tuplesize>, but applies to I<all> the curves.
|
|
|
|
C<--with> for instance.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=item
|
|
|
|
=item
|
|
|
|
|
|
|
|
|
|
|
@@ -1749,16 +2026,15 @@ is possible to send the output produced this way to gnuplot directly.
|
|
|
|
|
|
|
|
|
|
|
|
C<--exit>
|
|
|
|
C<--exit>
|
|
|
|
|
|
|
|
|
|
|
|
This controls the details of what happens when the input data is exhausted, or
|
|
|
|
This controls what happens when the input data is exhausted, or when some part
|
|
|
|
when some part of the C<feedgnuplot> pipeline is killed. This option does
|
|
|
|
of the C<feedgnuplot> pipeline is killed. This option does different things
|
|
|
|
different things depending on whether C<--stream> is active, so read this
|
|
|
|
depending on whether C<--stream> is active, so read this closely.
|
|
|
|
closely.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
With interactive gnuplot terminals (qt, x11, wxt), the plot windows live in a
|
|
|
|
With interactive gnuplot terminals (qt, x11, wxt), the plot windows live in a
|
|
|
|
separate process from the main C<gnuplot> process. It is thus possible for the
|
|
|
|
separate process from the main C<gnuplot> process. It is thus possible for the
|
|
|
|
main C<gnuplot> process to exit, while leaving the plot windows up (a caveat is
|
|
|
|
main C<gnuplot> process to exit, while leaving the plot windows up (a caveat is
|
|
|
|
that such decapitated windows aren't interactive). To be clear, there are 3
|
|
|
|
that such decapitated windows aren't interactive). There are 3 possible states
|
|
|
|
possible states:
|
|
|
|
of the polotting pipeline:
|
|
|
|
|
|
|
|
|
|
|
|
=over
|
|
|
|
=over
|
|
|
|
|
|
|
|
|
|
|
@@ -1773,35 +2049,35 @@ prompt available
|
|
|
|
|
|
|
|
|
|
|
|
=back
|
|
|
|
=back
|
|
|
|
|
|
|
|
|
|
|
|
The C<--exit> option controls the details of this behavior. The possibilities
|
|
|
|
The possibilities are:
|
|
|
|
are:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=over
|
|
|
|
=over
|
|
|
|
|
|
|
|
|
|
|
|
=item No C<--stream>, input pipe is exhausted (all data read in)
|
|
|
|
=item No C<--stream>, all data read in
|
|
|
|
|
|
|
|
|
|
|
|
=over
|
|
|
|
=over
|
|
|
|
|
|
|
|
|
|
|
|
=item default; no C<--exit>
|
|
|
|
=item no C<--exit> (default)
|
|
|
|
|
|
|
|
|
|
|
|
Alive. Need to Ctrl-C to get back into the shell
|
|
|
|
Alive. Need to Ctrl-C to get back into the shell
|
|
|
|
|
|
|
|
|
|
|
|
=item C<--exit>
|
|
|
|
=item C<--exit>
|
|
|
|
|
|
|
|
|
|
|
|
Half-alive. Non-interactive prompt up, and the shell accepts new commands.
|
|
|
|
Half-alive. Non-interactive prompt up, and the shell accepts new commands.
|
|
|
|
Without C<--stream> the goal is to show a plot, so a Dead state is not useful
|
|
|
|
Without C<--stream> the goal is to show a plot, so a Dead state would not be
|
|
|
|
here.
|
|
|
|
useful.
|
|
|
|
|
|
|
|
|
|
|
|
=back
|
|
|
|
=back
|
|
|
|
|
|
|
|
|
|
|
|
=item C<--stream>, input pipe is exhausted (all data read in) or the
|
|
|
|
=item C<--stream>, all data read in or the C<feedgnuplot> process terminated
|
|
|
|
C<feedgnuplot> process terminated
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=over
|
|
|
|
=over
|
|
|
|
|
|
|
|
|
|
|
|
=item default; no C<--exit>
|
|
|
|
=item no C<--exit> (default)
|
|
|
|
|
|
|
|
|
|
|
|
Alive. Need to Ctrl-C to get back into the shell
|
|
|
|
Alive. Need to Ctrl-C to get back into the shell. This means that when making
|
|
|
|
|
|
|
|
live plots, the first Ctrl-C kills the data feeding process, but leaves the
|
|
|
|
|
|
|
|
final plot up for inspection. A second Ctrl-C kills feedgnuplot as well.
|
|
|
|
|
|
|
|
|
|
|
|
=item C<--exit>
|
|
|
|
=item C<--exit>
|
|
|
|
|
|
|
|
|
|
|
@@ -1883,13 +2159,34 @@ in a Thinkpad.
|
|
|
|
--binwidth 10
|
|
|
|
--binwidth 10
|
|
|
|
--ymin 0 --xlabel 'File size (MB)' --ylabel Frequency
|
|
|
|
--ymin 0 --xlabel 'File size (MB)' --ylabel Frequency
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
=head2 Plotting a live histogram of the ping round-trip times for the past 20 seconds
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ ping -A -D 8.8.8.8 |
|
|
|
|
|
|
|
|
perl -anE 'BEGIN { $| = 1; }
|
|
|
|
|
|
|
|
$F[0] =~ s/[\[\]]//g or next;
|
|
|
|
|
|
|
|
$F[7] =~ s/.*=//g or next;
|
|
|
|
|
|
|
|
say "$F[0] $F[7]"' |
|
|
|
|
|
|
|
|
feedgnuplot --stream --domain --histogram 0 --binwidth 10 \
|
|
|
|
|
|
|
|
--xlabel 'Ping round-trip time (s)' \
|
|
|
|
|
|
|
|
--ylabel Frequency --xlen 20
|
|
|
|
|
|
|
|
|
|
|
|
=head2 Plotting points on top of an existing image
|
|
|
|
=head2 Plotting points on top of an existing image
|
|
|
|
|
|
|
|
|
|
|
|
This can be done by using C<--equation> to pass arbitrary plot input to gnuplot:
|
|
|
|
This can be done with C<--image>:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ < features_xy.data
|
|
|
|
|
|
|
|
feedgnuplot --points --domain --image "image.png"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
or with C<--equation>:
|
|
|
|
|
|
|
|
|
|
|
|
$ < features_xy.data
|
|
|
|
$ < features_xy.data
|
|
|
|
feedgnuplot --points --domain
|
|
|
|
feedgnuplot --points --domain
|
|
|
|
--equation '"image.png" binary filetype=png flipy with rgbimage'
|
|
|
|
--equation '"image.png" binary filetype=auto flipy with rgbimage'
|
|
|
|
|
|
|
|
--set 'yrange [:] reverse'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The C<--image> invocation is a convenience wrapper for the C<--equation>
|
|
|
|
|
|
|
|
version. Finer control is available with C<--equation>.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Here an existing image is given to gnuplot verbatim, and data to plot on top of
|
|
|
|
Here an existing image is given to gnuplot verbatim, and data to plot on top of
|
|
|
|
it is interpreted by feedgnuplot as usual. C<flipy> is useful here because
|
|
|
|
it is interpreted by feedgnuplot as usual. C<flipy> is useful here because
|
|
|
|