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
|
|
|
|
2009-08-11 15:30:25 +08:00
|
|
|
autoflush STDOUT 1;
|
|
|
|
|
2009-08-11 03:57:49 +08:00
|
|
|
# list containing the plot data. Each element is a hash describing the extra
|
|
|
|
# plotting options for each curve we're plotting, and the actual data to
|
|
|
|
# plot for each curve. The length of this list grows as the data comes
|
|
|
|
# in
|
|
|
|
my @curves = ();
|
|
|
|
|
2009-02-09 19:17:26 +08:00
|
|
|
# stream in the data by default
|
2009-02-09 17:35:19 +08:00
|
|
|
# point plotting by default
|
2009-02-09 19:17:26 +08:00
|
|
|
my %options = ( "stream" => 1,
|
2009-08-05 04:57:17 +08:00
|
|
|
"points" => 0,
|
2009-08-11 04:04:39 +08:00
|
|
|
"lines" => 0,
|
|
|
|
"ymin" => "",
|
|
|
|
"ymax" => "",
|
|
|
|
"y2min" => "",
|
|
|
|
"y2max" => "");
|
2009-02-09 17:35:19 +08:00
|
|
|
GetOptions(\%options,
|
2009-02-09 19:17:26 +08:00
|
|
|
"stream!",
|
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-08-11 03:57:49 +08:00
|
|
|
"xlen=i",
|
2009-08-11 03:42:48 +08:00
|
|
|
"ymin=f",
|
|
|
|
"ymax=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",
|
|
|
|
"help");
|
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"} )
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
return;
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
2009-08-25 05:31:42 +08:00
|
|
|
if( defined $options{"hardcopy"} && $options{"stream"} )
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
die("If making a hardcopy, we shouldn't be streaming. Doing nothing\n");
|
|
|
|
}
|
|
|
|
if( !defined $options{"xlen"} )
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
die("Must specify the size of the moving x-window. Doing nothing\n");
|
|
|
|
}
|
|
|
|
my $xwindow = $options{"xlen"};
|
2009-12-20 12:46:06 +08:00
|
|
|
|
2009-08-05 04:50:10 +08:00
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
|
|
|
|
|
|
|
|
# now start the data acquisition and plotting threads
|
|
|
|
my $dataQueue = Thread::Queue->new();
|
|
|
|
my $addThr = threads->create(\&mainThread);
|
2009-08-28 01:08:49 +08:00
|
|
|
my $plotThr = threads->create(\&plotThread) if(!$options{"stream"});
|
2009-08-25 05:31:42 +08:00
|
|
|
|
|
|
|
while(<>)
|
|
|
|
{
|
|
|
|
$dataQueue->enqueue($_);
|
|
|
|
}
|
|
|
|
|
|
|
|
$dataQueue->enqueue("Plot now");
|
|
|
|
$dataQueue->enqueue(undef);
|
|
|
|
|
|
|
|
$addThr->join();
|
2009-08-28 01:08:49 +08:00
|
|
|
$plotThr->join() if(!$options{"stream"});
|
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;
|
|
|
|
open PIPE, "|gnuplot" || 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 $temphardcopyfile;
|
|
|
|
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"); }
|
|
|
|
|
|
|
|
if ($outputfileType eq "png")
|
|
|
|
{
|
|
|
|
print PIPE "set terminal png\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
print PIPE "set terminal postscript solid color landscape 10\n";
|
|
|
|
}
|
|
|
|
# write to a temporary file first
|
|
|
|
$temphardcopyfile = $outputfile;
|
|
|
|
$temphardcopyfile =~ s{/}{_}g;
|
|
|
|
$temphardcopyfile = "/tmp/$temphardcopyfile";
|
|
|
|
print PIPE "set output \"$temphardcopyfile\"\n";
|
|
|
|
}
|
|
|
|
|
2009-01-24 08:56:57 +08:00
|
|
|
print PIPE "set xtics\n";
|
2009-08-07 03:09:46 +08:00
|
|
|
print PIPE "set ytics nomirror\n";
|
2009-08-05 04:50:10 +08:00
|
|
|
print PIPE "set y2tics\n";
|
2009-08-11 02:14:16 +08:00
|
|
|
print PIPE "set yrange [". $options{"ymin"} . ":" . $options{"ymax"} ."]\n" if $options{"y2max"};
|
2009-08-05 04:50:10 +08:00
|
|
|
print PIPE "set y2range [". $options{"y2min"} . ":" . $options{"y2max"} ."]\n" if $options{"y2max"};
|
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])
|
|
|
|
{
|
|
|
|
$curves[$y2idx]{"extraopts"} .= $str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
newCurve("", $str);
|
|
|
|
}
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
2009-01-24 08:56:57 +08:00
|
|
|
|
2009-08-07 02:59:38 +08:00
|
|
|
# regexp for a possibly floating point, possibly scientific notation number
|
|
|
|
my $numRE = qr/([-]?[0-9\.]+(?:e[-]?[0-9]+)?)/;
|
2009-08-25 05:31:42 +08:00
|
|
|
my $xlast;
|
|
|
|
while( $_ = $dataQueue->dequeue() )
|
2009-02-09 19:17:26 +08:00
|
|
|
{
|
2009-08-25 05:31:42 +08:00
|
|
|
if(!/Plot now/)
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
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
|
|
|
|
/($numRE)/gc or next;
|
|
|
|
$xlast = $1;
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
while(/([0-9]+) ($numRE)/gc)
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2009-08-25 05:31:42 +08:00
|
|
|
my $idx = $1;
|
|
|
|
my $point = $2;
|
2009-08-11 03:57:49 +08:00
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
# if this curve index doesn't exist, create curve up-to this index
|
|
|
|
while(!exists $curves[$idx])
|
|
|
|
{
|
|
|
|
newCurve("", "");
|
|
|
|
}
|
|
|
|
|
|
|
|
push @{$curves[$idx]->{"data"}}, [$xlast, $point];
|
|
|
|
}
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
elsif($options{"stream"})
|
2009-08-11 03:57:49 +08:00
|
|
|
{
|
2009-08-25 05:31:42 +08:00
|
|
|
cutOld($xlast - $xwindow);
|
|
|
|
plotStoredData($xlast - $xwindow, $xlast);
|
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
|
|
|
|
usleep(100_000) until -e $temphardcopyfile;
|
|
|
|
usleep(100_000) until(system("fuser -s $temphardcopyfile"));
|
|
|
|
|
|
|
|
if($outputfileType eq "pdf")
|
|
|
|
{
|
|
|
|
system("ps2pdf $temphardcopyfile $outputfile");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
system("mv $temphardcopyfile $outputfile");
|
|
|
|
}
|
|
|
|
printf "Wrote output to $outputfile\n";
|
|
|
|
return;
|
|
|
|
}
|
2009-12-20 12:46:06 +08:00
|
|
|
}
|
2009-02-09 19:17:26 +08:00
|
|
|
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) = @_;
|
|
|
|
foreach (@curves)
|
|
|
|
{
|
|
|
|
my $xy = $_->{"data"};
|
2009-08-12 02:18:52 +08:00
|
|
|
|
|
|
|
if( @$xy )
|
2009-08-11 16:54:38 +08:00
|
|
|
{
|
2009-08-12 02:18:52 +08:00
|
|
|
my $firstInWindow = first_index {$_->[0] >= $oldestx} @$xy;
|
|
|
|
splice( @$xy, 0, $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-08-12 05:11:24 +08:00
|
|
|
my @extraopts = map {$_->{"extraopts"}} grep {@{$_->{"data"}}} @curves;
|
2009-08-11 16:54:38 +08:00
|
|
|
print PIPE 'plot ' . join(', ' , map({ '"-"' . $_} @extraopts) ) . "\n";
|
2009-08-11 03:57:49 +08:00
|
|
|
|
|
|
|
foreach my $curve (@curves)
|
2009-02-09 19:17:26 +08:00
|
|
|
{
|
2009-08-11 03:57:49 +08:00
|
|
|
my $buf = $curve->{"data"};
|
2009-08-12 05:11:24 +08:00
|
|
|
next unless @$buf;
|
2009-08-11 03:57:49 +08:00
|
|
|
for my $elem (@$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";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-11 03:57:49 +08:00
|
|
|
sub newCurve()
|
|
|
|
{
|
|
|
|
my ($title, $opts, $newpoint) = @_;
|
|
|
|
if($title) { $opts = "title \"$title\" $opts" }
|
|
|
|
else { $opts = "notitle $opts" }
|
|
|
|
|
2009-08-11 16:54:38 +08:00
|
|
|
$newpoint = [] unless defined $newpoint;
|
2009-08-11 03:57:49 +08:00
|
|
|
push ( @curves,
|
|
|
|
{"extraopts" => " $opts",
|
2009-08-11 16:54:38 +08:00
|
|
|
"data" => $newpoint} );
|
2009-08-11 03:57:49 +08:00
|
|
|
}
|
2009-02-09 19:17:26 +08:00
|
|
|
|
2009-08-25 05:31:42 +08:00
|
|
|
sub usage {
|
|
|
|
print "Usage: $0 <options>\n";
|
|
|
|
print <<OEF;
|
|
|
|
--[no]stream Do [not] display the data a point at a time, as it comes in
|
|
|
|
--[no]lines Do [not] draw lines to connect consecutive points
|
|
|
|
--xlabel xxx Set x-axis label
|
|
|
|
--ylabel xxx Set y-axis label
|
|
|
|
--y2label xxx Set y2-axis label
|
|
|
|
--title xxx Set the title of the plot
|
|
|
|
--legend xxx Set the label for a curve plot. Give this option multiple times for multiple curves
|
|
|
|
--xlen xxx Set the size of the x-window to plot
|
|
|
|
--ymin xxx Set the range for the y axis. Both or neither of these have to be specified
|
|
|
|
--ymax xxx Set the range for the y axis. Both or neither of these have to be specified
|
|
|
|
--y2min xxx Set the range for the y2 axis. Both or neither of these have to be specified
|
|
|
|
--y2max xxx Set the range for the y2 axis. Both or neither of these have to be specified
|
|
|
|
--y2 xxx Plot the data with this index on the y2 axis. These are 0-indexed
|
|
|
|
--hardcopy xxx If not streaming, output to a file specified here. Format inferred from filename
|
|
|
|
OEF
|
|
|
|
}
|
|
|
|
|