2009-12-20 12:46:06 +08:00
|
|
|
#!/usr/bin/perl -w
|
|
|
|
use strict;
|
2009-02-09 17:35:19 +08:00
|
|
|
use Getopt::Long;
|
2009-08-25 05:31:42 +08:00
|
|
|
use Time::HiRes qw( usleep gettimeofday tv_interval);
|
2009-08-11 15:09:12 +08:00
|
|
|
use IO::Handle;
|
2009-08-12 02:18:52 +08:00
|
|
|
use List::MoreUtils qw( first_index );
|
2009-02-09 19:17:26 +08:00
|
|
|
use Data::Dumper;
|
2009-08-25 05:31:42 +08:00
|
|
|
use threads;
|
|
|
|
use Thread::Queue;
|
2009-02-09 17:35:19 +08:00
|
|
|
|
2010-01-13 13:54:49 +08:00
|
|
|
open(GNUPLOT_VERSION, "gnuplot --version |");
|
|
|
|
my ($gnuplotVersion) = <GNUPLOT_VERSION> =~ /gnuplot\s*([0-9]*\.[0-9]*)/;
|
|
|
|
if(!$gnuplotVersion)
|
|
|
|
{
|
|
|
|
print STDERR "Couldn't find the version of gnuplot. Does it work? Trying anyway...\n";
|
|
|
|
$gnuplotVersion = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
close(GNUPLOT_VERSION);
|
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
my $usage = <<OEF;
|
2010-02-06 06:33:46 +08:00
|
|
|
Usage: $0 [options] file1 file2 ...
|
|
|
|
any number of data files can be given on the cmdline. They will be processed
|
|
|
|
in sequence. If no data files are given, data will be read in from standard
|
|
|
|
input.
|
2009-12-07 06:49:44 +08:00
|
|
|
|
|
|
|
--[no]domain If enabled, the first element of each line is the
|
|
|
|
domain variable. If not, the point index is used
|
|
|
|
|
|
|
|
--[no]dataindex If enabled, each data point is preceded by the index
|
|
|
|
of the data set that point corresponds to. If not
|
|
|
|
enabled, the order of the point is used.
|
|
|
|
|
|
|
|
As an example, if line 3 of the input is "0 9 1 20"
|
|
|
|
'--nodomain --nodataindex' would parse the 4 numbers as points in 4
|
|
|
|
different curves at x=3
|
|
|
|
|
|
|
|
'--domain --nodataindex' would parse the 4 numbers as points in 3 different
|
|
|
|
curves at x=0. Here, 0 is the x-variable and 9,1,20 are the data values
|
|
|
|
|
|
|
|
'--nodomain --dataindex' would parse the 4 numbers as points in 2 different
|
|
|
|
curves at x=3. Here 0 and 1 are the data indices and 9 and 20 are the
|
|
|
|
data values
|
|
|
|
|
|
|
|
'--domain --dataindex' would parse the 4 numbers as a single point at
|
|
|
|
x=0. Here 9 is the data index and 1 is the data value. 20 is an extra
|
|
|
|
value, so it is ignored. If another value followed 20, we'd get another
|
|
|
|
point in curve number
|
|
|
|
|
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--[no]stream Do [not] display the data a point at a time, as it
|
|
|
|
comes in
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--[no]lines Do [not] draw lines to connect consecutive points
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--[no]points Do [not] draw points
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--xlabel xxx Set x-axis label
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--ylabel xxx Set y-axis label
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--y2label xxx Set y2-axis label
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--title xxx Set the title of the plot
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--legend xxx Set the label for a curve plot. Give this option multiple
|
|
|
|
times for multiple curves
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2010-02-12 06:14:57 +08:00
|
|
|
--xlen xxx Set the size of the x-window to plot. Omit this or set it
|
|
|
|
to 0 to plot ALL the data
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--xmin xxx Set the range for the x axis. These are ignored in a
|
|
|
|
streaming plot
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--xmax xxx Set the range for the x axis. These are ignored in a
|
|
|
|
streaming plot
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--ymin xxx Set the range for the y axis.
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--ymax xxx Set the range for the y axis.
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--y2min xxx Set the range for the y2 axis.
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 04:48:50 +08:00
|
|
|
--y2max xxx Set the range for the y2 axis.
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--y2 xxx Plot the data with this index on the y2 axis. These are
|
|
|
|
0-indexed
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--hardcopy xxx If not streaming, output to a file specified here. Format
|
|
|
|
inferred from filename
|
2010-01-15 06:34:58 +08:00
|
|
|
|
2010-01-15 07:09:33 +08:00
|
|
|
--maxcurves xxx The maximum allowed number of curves. This is 100 by default,
|
2010-01-15 06:45:24 +08:00
|
|
|
but can be reset with this option. This exists purely to
|
|
|
|
prevent perl from allocating all of the system's memory when
|
|
|
|
reading bogus data
|
|
|
|
|
2010-03-29 09:57:38 +08:00
|
|
|
--monotonic If --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 currently
|
|
|
|
cached for this curve is purged. Without --monotonic, all
|
|
|
|
data is kept. No --monotonic by default
|
|
|
|
|
2009-12-07 06:22:03 +08:00
|
|
|
--dump Instead of printing to gnuplot, print to STDOUT. For
|
|
|
|
debugging.
|
2009-12-07 04:48:50 +08:00
|
|
|
OEF
|
|
|
|
|
2010-02-06 04:05:52 +08:00
|
|
|
# if I'm using a self-plotting data file with a #! line, then $ARGV[0] will contain ALL of the
|
|
|
|
# options and $ARGV[1] will contain the data file to plot. In this case I need to split $ARGV[0] so
|
|
|
|
# that GetOptions() can parse it correctly. On the other hand, if I'm plotting normally (not with
|
|
|
|
# #!) a file with spaces in the filename, I don't want to split the filename. Hopefully this logic
|
|
|
|
# takes care of both those cases.
|
|
|
|
if(exists $ARGV[0] && !-r $ARGV[0])
|
|
|
|
{
|
|
|
|
unshift @ARGV, split(/\s+/, shift(@ARGV));
|
|
|
|
}
|
|
|
|
|
2009-12-19 04:48:15 +08:00
|
|
|
# do not stream in the data by default
|
2010-03-29 09:57:38 +08:00
|
|
|
# point plotting by default.
|
|
|
|
# no monotonicity checks by default
|
2009-12-19 04:48:15 +08:00
|
|
|
my %options = ( "stream" => 0,
|
2009-12-07 06:49:44 +08:00
|
|
|
"domain" => 0,
|
|
|
|
"dataindex" => 0,
|
|
|
|
"points" => 0,
|
|
|
|
"lines" => 0,
|
2010-02-12 06:14:57 +08:00
|
|
|
"xlen" => 0,
|
2009-12-07 06:49:44 +08:00
|
|
|
"ymin" => "",
|
|
|
|
"ymax" => "",
|
|
|
|
"y2min" => "",
|
2010-01-15 06:45:24 +08:00
|
|
|
"y2max" => "",
|
2010-01-15 07:09:33 +08:00
|
|
|
"maxcurves" => 100);
|
2010-01-15 06:45:24 +08:00
|
|
|
|
2009-02-09 17:35:19 +08:00
|
|
|
GetOptions(\%options,
|
2009-02-09 19:17:26 +08:00
|
|
|
"stream!",
|
2009-12-07 06:49:44 +08:00
|
|
|
"domain!",
|
|
|
|
"dataindex!",
|
2009-08-05 01:37:13 +08:00
|
|
|
"lines!",
|
2009-08-11 03:54:21 +08:00
|
|
|
"points!",
|
2009-08-05 04:50:10 +08:00
|
|
|
"legend=s@",
|
|
|
|
"xlabel=s",
|
|
|
|
"ylabel=s",
|
|
|
|
"y2label=s",
|
|
|
|
"title=s",
|
2009-09-09 02:32:22 +08:00
|
|
|
"xlen=f",
|
2009-08-11 03:42:48 +08:00
|
|
|
"ymin=f",
|
|
|
|
"ymax=f",
|
2009-12-04 03:22:33 +08:00
|
|
|
"xmin=f",
|
|
|
|
"xmax=f",
|
2009-08-05 04:50:10 +08:00
|
|
|
"y2min=f",
|
|
|
|
"y2max=f",
|
2009-08-05 07:03:57 +08:00
|
|
|
"y2=i@",
|
2009-08-11 03:57:49 +08:00
|
|
|
"hardcopy=s",
|
2010-01-15 06:45:24 +08:00
|
|
|
"maxcurves=i",
|
2010-03-29 09:57:38 +08:00
|
|
|
"monotonic!",
|
2009-09-16 07:25:13 +08:00
|
|
|
"help",
|
2009-12-07 04:48:50 +08:00
|
|
|
"dump") or die($usage);
|
2009-02-09 17:35:19 +08:00
|
|
|
|
|
|
|
# set up plotting style
|
2009-08-05 04:57:17 +08:00
|
|
|
my $style = "";
|
|
|
|
if($options{"lines"}) { $style .= "lines";}
|
|
|
|
if($options{"points"}) { $style .= "points";}
|
|
|
|
|
|
|
|
if(!$style) { $style = "points"; }
|
2009-12-20 12:46:06 +08:00
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
if( defined $options{"help"} )
|
|
|
|
{
|
2009-12-07 04:48:50 +08:00
|
|
|
die($usage);
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
2009-08-25 05:31:42 +08:00
|
|
|
|
2009-12-07 06:20:27 +08:00
|
|
|
|
2010-03-29 07:46:27 +08:00
|
|
|
# list containing the plot data. Each element is a reference to a list, representing the data for
|
|
|
|
# one curve. The first "point" is a hash describing various curve parameters. The rest are all
|
|
|
|
# references to lists of (x,y) tuples
|
2009-12-07 06:20:27 +08:00
|
|
|
my @curves = ();
|
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
# now start the data acquisition and plotting threads
|
2009-09-11 06:11:51 +08:00
|
|
|
my $dataQueue;
|
2009-09-09 02:35:59 +08:00
|
|
|
my $xwindow;
|
2009-08-25 05:31:42 +08:00
|
|
|
|
2009-08-28 01:30:55 +08:00
|
|
|
if($options{"stream"})
|
2009-08-25 05:31:42 +08:00
|
|
|
{
|
2009-09-09 02:32:44 +08:00
|
|
|
if( defined $options{"hardcopy"})
|
|
|
|
{
|
|
|
|
$options{"stream"} = undef;
|
|
|
|
}
|
|
|
|
|
2009-09-11 06:11:51 +08:00
|
|
|
$dataQueue = Thread::Queue->new();
|
2009-08-28 01:30:55 +08:00
|
|
|
my $addThr = threads->create(\&mainThread);
|
|
|
|
my $plotThr = threads->create(\&plotThread);
|
2009-08-25 05:31:42 +08:00
|
|
|
|
2009-08-28 01:30:55 +08:00
|
|
|
while(<>)
|
|
|
|
{
|
2009-12-07 06:49:44 +08:00
|
|
|
# place every line of input to the queue, so that the plotting thread can process it.
|
2009-08-28 01:30:55 +08:00
|
|
|
$dataQueue->enqueue($_);
|
2009-12-07 06:49:44 +08:00
|
|
|
|
|
|
|
# if we are using an implicit domain (x = line number), then we send it on the data queue also,
|
|
|
|
# since $. is not meaningful in the plotting thread
|
|
|
|
if(!$options{domain})
|
|
|
|
{
|
|
|
|
$dataQueue->enqueue($.);
|
|
|
|
}
|
2009-08-28 01:30:55 +08:00
|
|
|
}
|
2009-08-25 05:31:42 +08:00
|
|
|
|
2009-08-28 01:30:55 +08:00
|
|
|
$dataQueue->enqueue("Plot now");
|
|
|
|
$dataQueue->enqueue(undef);
|
2009-08-25 05:31:42 +08:00
|
|
|
|
2009-08-28 01:30:55 +08:00
|
|
|
$addThr->join();
|
|
|
|
$plotThr->join();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mainThread();
|
|
|
|
}
|
2009-08-25 05:31:42 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub plotThread
|
|
|
|
{
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
sleep(1);
|
|
|
|
$dataQueue->enqueue("Plot now");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub mainThread {
|
2009-01-24 08:56:57 +08:00
|
|
|
local *PIPE;
|
2009-08-28 04:23:08 +08:00
|
|
|
my $dopersist = "";
|
2010-01-13 13:54:49 +08:00
|
|
|
|
|
|
|
if($gnuplotVersion >= 4.3)
|
|
|
|
{
|
|
|
|
$dopersist = "--persist" if(!$options{"stream"});
|
|
|
|
}
|
2009-08-28 04:23:08 +08:00
|
|
|
|
2009-09-16 07:25:13 +08:00
|
|
|
if(exists $options{"dump"})
|
|
|
|
{
|
|
|
|
*PIPE = *STDOUT;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
open PIPE, "|gnuplot $dopersist" || die "Can't initialize gnuplot\n";
|
|
|
|
}
|
2009-08-11 15:09:12 +08:00
|
|
|
autoflush PIPE 1;
|
2009-08-11 03:57:49 +08:00
|
|
|
|
2009-08-05 07:03:57 +08:00
|
|
|
my $outputfile;
|
|
|
|
my $outputfileType;
|
|
|
|
if( defined $options{"hardcopy"})
|
|
|
|
{
|
|
|
|
$outputfile = $options{"hardcopy"};
|
|
|
|
($outputfileType) = $outputfile =~ /\.(ps|pdf|png)$/;
|
|
|
|
if(!$outputfileType) { die("Only .ps, .pdf and .png supported\n"); }
|
|
|
|
|
2009-12-20 11:14:11 +08:00
|
|
|
my %terminalOpts =
|
|
|
|
( ps => 'postscript solid color landscape 10',
|
|
|
|
pdf => 'pdfcairo solid color font ",10" size 11in,8.5in',
|
|
|
|
png => 'png' );
|
|
|
|
|
|
|
|
print PIPE "set terminal $terminalOpts{$outputfileType}\n";
|
|
|
|
print PIPE "set output \"$outputfile\"\n";
|
2009-08-05 07:03:57 +08:00
|
|
|
}
|
2009-09-16 03:13:59 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
print PIPE "set terminal x11\n";
|
|
|
|
}
|
2009-08-05 07:03:57 +08:00
|
|
|
|
2009-01-24 08:56:57 +08:00
|
|
|
print PIPE "set xtics\n";
|
2009-09-09 02:17:44 +08:00
|
|
|
if($options{"y2"})
|
|
|
|
{
|
|
|
|
print PIPE "set ytics nomirror\n";
|
|
|
|
print PIPE "set y2tics\n";
|
2009-12-04 03:22:33 +08:00
|
|
|
print PIPE "set y2range [". $options{"y2min"} . ":" . $options{"y2max"} ."]\n" if( $options{"y2min"} || $options{"y2max"} );
|
2009-09-09 02:17:44 +08:00
|
|
|
}
|
2009-12-04 03:22:33 +08:00
|
|
|
print PIPE "set xrange [". $options{"xmin"} . ":" . $options{"xmax"} ."]\n" if( $options{"xmin"} || $options{"xmax"} );;
|
|
|
|
print PIPE "set yrange [". $options{"ymin"} . ":" . $options{"ymax"} ."]\n" if( $options{"ymin"} || $options{"ymax"} );;
|
2009-02-09 17:35:19 +08:00
|
|
|
print PIPE "set style data $style\n";
|
2009-01-24 08:56:57 +08:00
|
|
|
print PIPE "set grid\n";
|
|
|
|
|
2009-08-05 04:50:10 +08:00
|
|
|
print(PIPE "set xlabel \"" . $options{"xlabel" } . "\"\n") if $options{"xlabel"};
|
|
|
|
print(PIPE "set ylabel \"" . $options{"ylabel" } . "\"\n") if $options{"ylabel"};
|
|
|
|
print(PIPE "set y2label \"" . $options{"y2label"} . "\"\n") if $options{"y2label"};
|
|
|
|
print(PIPE "set title \"" . $options{"title" } . "\"\n") if $options{"title"};
|
|
|
|
|
2009-08-11 03:57:49 +08:00
|
|
|
# For the specified values, set the legend entries to 'title "blah blah"'
|
|
|
|
if($options{"legend"})
|
|
|
|
{
|
|
|
|
foreach (@{$options{"legend"}}) { newCurve($_, "") }
|
|
|
|
}
|
2009-08-05 01:37:13 +08:00
|
|
|
|
2009-08-05 04:50:10 +08:00
|
|
|
# For the values requested to be printed on the y2 axis, set that
|
2009-08-11 03:57:49 +08:00
|
|
|
foreach my $y2idx (@{$options{"y2"}})
|
|
|
|
{
|
|
|
|
my $str = " axes x1y2 linewidth 3";
|
|
|
|
if(exists $curves[$y2idx])
|
|
|
|
{
|
2010-03-29 07:46:27 +08:00
|
|
|
$curves[$y2idx][0]{"options"} .= $str;
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-11-05 10:20:51 +08:00
|
|
|
newCurve("", $str, undef, $y2idx);
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
2009-01-24 08:56:57 +08:00
|
|
|
|
2009-12-07 06:23:21 +08:00
|
|
|
# regexp for a possibly floating point, possibly scientific notation number, fully captured
|
2010-03-03 09:13:12 +08:00
|
|
|
my $numRE = qr/([-]?[0-9\.]+(?:e[-+]?[0-9]+)?)/o;
|
2009-08-25 05:31:42 +08:00
|
|
|
my $xlast;
|
2009-12-07 06:55:31 +08:00
|
|
|
my $haveNewData;
|
2009-09-11 06:11:51 +08:00
|
|
|
|
2009-12-07 06:22:52 +08:00
|
|
|
while( $_ = ($dataQueue && $dataQueue->dequeue()) // <> )
|
2009-02-09 19:17:26 +08:00
|
|
|
{
|
2010-01-11 15:46:53 +08:00
|
|
|
next if /^#/o;
|
|
|
|
|
2009-09-11 06:11:51 +08:00
|
|
|
if($_ ne "Plot now")
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2009-12-07 06:55:31 +08:00
|
|
|
$haveNewData = 1;
|
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
# parse the incoming data lines. The format is
|
|
|
|
# x idx0 dat0 idx1 dat1 ....
|
|
|
|
# where idxX is the index of the curve that datX corresponds to
|
2009-12-07 06:49:44 +08:00
|
|
|
#
|
|
|
|
# $options{domain} indicates whether the initial 'x' is given or not (if not, the line
|
|
|
|
# number is used)
|
|
|
|
# $options{dataindex} indicates whether idxX is given or not (if not, the point order in the
|
|
|
|
# line is used)
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-12-07 06:49:44 +08:00
|
|
|
if($options{domain})
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2009-12-07 06:49:44 +08:00
|
|
|
/$numRE/go or next;
|
|
|
|
$xlast = $1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# since $. is not meaningful in the plotting thread if we're using the data queue, we pass
|
|
|
|
# $. on the data queue in that case. This construct pulls it from the queue if we're using
|
|
|
|
# it, otherwise it uses $. directly
|
|
|
|
$xlast = ($dataQueue && $dataQueue->dequeue()) // $.;
|
|
|
|
}
|
2009-08-11 03:57:49 +08:00
|
|
|
|
2009-12-07 06:49:44 +08:00
|
|
|
if($options{dataindex})
|
|
|
|
{
|
|
|
|
while(/([0-9]+)\s+$numRE/go)
|
|
|
|
{
|
|
|
|
my $idx = $1;
|
|
|
|
my $point = $2;
|
|
|
|
|
2010-03-29 09:57:38 +08:00
|
|
|
pushPoint($idx, [$xlast, $point]);
|
2009-12-07 06:49:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
my $idx = 0;
|
|
|
|
foreach my $point (/$numRE/go)
|
|
|
|
{
|
2010-03-29 09:57:38 +08:00
|
|
|
pushPoint($idx, [$xlast, $point]);
|
2009-12-07 06:49:44 +08:00
|
|
|
$idx++;
|
|
|
|
}
|
2009-08-25 05:31:42 +08:00
|
|
|
}
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
|
|
|
|
2010-03-29 07:04:51 +08:00
|
|
|
elsif($options{"stream"})
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2010-03-29 07:04:51 +08:00
|
|
|
# only redraw a streaming plot if there's new data to plot
|
2009-12-07 06:55:31 +08:00
|
|
|
next unless $haveNewData;
|
|
|
|
$haveNewData = undef;
|
|
|
|
|
2010-02-12 06:14:57 +08:00
|
|
|
if( $options{"xlen"} )
|
|
|
|
{
|
|
|
|
cutOld($xlast - $options{"xlen"});
|
|
|
|
plotStoredData($xlast - $options{"xlen"}, $xlast);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plotStoredData();
|
|
|
|
}
|
2009-02-09 19:17:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-11 15:45:41 +08:00
|
|
|
# finished reading in all of the data
|
2009-02-09 19:17:26 +08:00
|
|
|
if($options{"stream"})
|
|
|
|
{
|
|
|
|
print PIPE "exit;\n";
|
|
|
|
close PIPE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-08-11 16:54:38 +08:00
|
|
|
plotStoredData();
|
2009-08-05 07:03:57 +08:00
|
|
|
|
|
|
|
if( defined $options{"hardcopy"})
|
|
|
|
{
|
|
|
|
print PIPE "set output\n";
|
|
|
|
# sleep until the plot file exists, and it is closed. Sometimes the output is
|
|
|
|
# still being written at this point
|
2009-12-20 11:14:11 +08:00
|
|
|
usleep(100_000) until -e $outputfile;
|
|
|
|
usleep(100_000) until(system("fuser -s $outputfile"));
|
2009-08-05 07:03:57 +08:00
|
|
|
|
2009-12-04 03:37:50 +08:00
|
|
|
print "Wrote output to $outputfile\n";
|
2009-08-05 07:03:57 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-08-28 04:23:08 +08:00
|
|
|
|
|
|
|
# we persist gnuplot, so we shouldn't need this sleep. However, once
|
|
|
|
# gnuplot exist, but the persistent window sticks around, you can no
|
|
|
|
# longer interactively zoom the plot. So we still sleep
|
|
|
|
sleep(100000);
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-11 16:54:38 +08:00
|
|
|
sub cutOld
|
2009-02-09 19:17:26 +08:00
|
|
|
{
|
2009-08-11 16:54:38 +08:00
|
|
|
my ($oldestx) = @_;
|
2009-08-12 02:18:52 +08:00
|
|
|
|
2009-09-11 06:11:51 +08:00
|
|
|
foreach my $xy (@curves)
|
|
|
|
{
|
|
|
|
if( @$xy > 1 )
|
2009-08-11 16:54:38 +08:00
|
|
|
{
|
2009-09-11 06:11:51 +08:00
|
|
|
my $firstInWindow = first_index {$_->[0] >= $oldestx} @{$xy}[1..$#$xy];
|
|
|
|
splice( @$xy, 1, $firstInWindow ) unless $firstInWindow == -1;
|
2009-08-11 16:54:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-08-11 16:54:38 +08:00
|
|
|
sub plotStoredData
|
|
|
|
{
|
|
|
|
my ($xmin, $xmax) = @_;
|
|
|
|
print PIPE "set xrange [$xmin:$xmax]\n" if defined $xmin;
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-09-11 06:11:51 +08:00
|
|
|
# get the options for those curves that have any data
|
2009-09-16 03:44:23 +08:00
|
|
|
my @nonemptyCurves = grep {@$_ > 1} @curves;
|
2010-03-29 07:46:27 +08:00
|
|
|
my @extraopts = map {$_->[0]{"options"}} @nonemptyCurves;
|
2009-09-11 06:11:51 +08:00
|
|
|
|
2009-08-11 16:54:38 +08:00
|
|
|
print PIPE 'plot ' . join(', ' , map({ '"-"' . $_} @extraopts) ) . "\n";
|
2009-08-11 03:57:49 +08:00
|
|
|
|
2009-09-16 03:44:23 +08:00
|
|
|
foreach my $buf (@nonemptyCurves)
|
2009-02-09 19:17:26 +08:00
|
|
|
{
|
2009-09-11 06:11:51 +08:00
|
|
|
# send each point to gnuplot. Ignore the first "point" since it's the
|
2010-03-29 07:46:27 +08:00
|
|
|
# curve options
|
2009-09-11 06:11:51 +08:00
|
|
|
for my $elem (@{$buf}[1..$#$buf]) {
|
2009-08-11 16:54:38 +08:00
|
|
|
my ($x, $y) = @$elem;
|
|
|
|
print PIPE "$x $y\n";
|
2009-02-09 19:17:26 +08:00
|
|
|
}
|
|
|
|
print PIPE "e\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-29 09:57:38 +08:00
|
|
|
sub newCurve
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2009-11-05 10:20:51 +08:00
|
|
|
my ($title, $opts, $newpoint, $idx) = @_;
|
|
|
|
|
2010-01-15 06:45:24 +08:00
|
|
|
if(scalar @curves >= $options{maxcurves})
|
|
|
|
{
|
|
|
|
say STDERR "Tried to exceed the --maxcurves setting.";
|
|
|
|
say STDERR "Invoke with a higher --maxcurves limit if you really want to do this.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-11-05 10:20:51 +08:00
|
|
|
# if this curve index doesn't exist, create curve up-to this index
|
|
|
|
if(defined $idx)
|
|
|
|
{
|
|
|
|
while(!exists $curves[$idx])
|
|
|
|
{
|
|
|
|
pushNewEmptyCurve();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# if we're not given an index, create a new one at the end, and fill it in
|
|
|
|
pushNewEmptyCurve();
|
|
|
|
$idx = $#curves;
|
|
|
|
}
|
|
|
|
|
2009-08-11 03:57:49 +08:00
|
|
|
if($title) { $opts = "title \"$title\" $opts" }
|
|
|
|
else { $opts = "notitle $opts" }
|
|
|
|
|
2009-09-11 06:11:51 +08:00
|
|
|
if( defined $newpoint )
|
|
|
|
{
|
2010-03-29 07:46:27 +08:00
|
|
|
$curves[$idx] = [{"options" => " $opts"}, $newpoint];
|
2009-09-11 06:11:51 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-03-29 07:46:27 +08:00
|
|
|
$curves[$idx] = [{"options" => " $opts"}];
|
2009-09-11 06:11:51 +08:00
|
|
|
}
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-11-05 10:20:51 +08:00
|
|
|
sub pushNewEmptyCurve
|
|
|
|
{
|
|
|
|
my $opts = "notitle ";
|
2010-03-29 07:46:27 +08:00
|
|
|
push @curves, [{"options" => " $opts"}];
|
2009-11-05 10:20:51 +08:00
|
|
|
}
|
2010-03-29 09:57:38 +08:00
|
|
|
|
|
|
|
sub pushPoint
|
|
|
|
{
|
|
|
|
my ($idx, $xy) = @_;
|
|
|
|
|
|
|
|
if ( !exists $curves[$idx] )
|
|
|
|
{
|
|
|
|
newCurve("", "", undef, $idx);
|
|
|
|
}
|
|
|
|
elsif($options{monotonic})
|
|
|
|
{
|
|
|
|
my $curve = $curves[$idx];
|
|
|
|
if( @$curve > 1 && $xy->[0] < $curve->[$#{$curve}][0] )
|
|
|
|
{
|
|
|
|
# the x-coordinate of the new point is in the past, so I wipe out all the data for this curve
|
|
|
|
# and start anew
|
|
|
|
splice( @$curve, 1, @$curve-1 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
push @{$curves[$idx]}, $xy;
|
|
|
|
}
|