mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
gnuPlot.hpp
Go to the documentation of this file.
1/** \file gnuPlot.hpp
2 * \author Jared R. Males (jaredmales@gmail.com)
3 * \brief Declaration and definition of an interface to the gnuplot program
4 * \ingroup plotting_files
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015-2023 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef math_plot_gnuPlot_hpp
28#define math_plot_gnuPlot_hpp
29
30#include <sys/types.h>
31#include <sys/stat.h>
32#include <unistd.h>
33#include <stdio.h>
34#include <fcntl.h>
35#include <poll.h>
36
37#include <stdlib.h>
38
39#include <string>
40#include <iostream>
41#include <fstream>
42#include <vector>
43
44#include <cmath>
45
46#include <map>
47
48#include "../constants.hpp"
49#include "../../mxError.hpp"
50#include "../../sys/timeUtils.hpp"
51
52#ifndef MX_GP_FNAME_SZ
53/// The size of the string for managing the stderr fifo
54/** \ingroup plotting
55 */
56#define MX_GP_FNAME_SZ ( 128 )
57#endif
58
59#ifndef MX_GP_TEMP_SZ
60/// The size of the string for managing temporary file names
61/** \ingroup plotting
62 */
63#define MX_GP_TEMP_SZ ( 128 )
64#endif
65
66#ifndef MX_GP_FC_TIME
67/** \def MX_GP_FC_TIME
68 * Time, in microseconds, to wait for gnuplot startup and file creation to complete.
69 * Opening is retried after each timeout of this length for MX_GP_FC_RETRIES attempts.
70 * \ingroup plotting
71 */
72#define MX_GP_FC_TIME ( 10000 )
73#endif
74
75#ifndef MX_GP_FC_RETRIES
76/** \def MX_GP_FC_RETRIES
77 * Number of times to retry opening the gnuplot stderr file.
78 * \ingroup plotting
79 */
80#define MX_GP_FC_RETRIES ( 10 )
81#endif
82
83namespace mx
84{
85namespace math
86{
87
88// Get the gnuplot binary format string for a type
89// Specializations below.
90template <typename dataT>
91std::string gpBinaryFormat()
92{
93 static_assert( std::is_fundamental<dataT>::value || !std::is_fundamental<dataT>::value,
94 "No gnuplot format specifier available for this type." );
95 return "";
96}
97
98struct gpCurve
99{
100 std::string m_file;
101
102 std::string m_title;
103
104 std::string m_modifiers;
105
106 std::string m_binary;
107
108 gpCurve()
109 {
110 }
111
112 gpCurve( const std::string &file, const std::string &title, const std::string &modifiers )
113 : m_file( file ), m_title( title ), m_modifiers( modifiers )
114 {
115 }
116
117 gpCurve( const std::string &file,
118 const std::string &title,
119 const std::string &modifiers,
120 const std::string &binary )
121 : m_file( file ), m_title( title ), m_modifiers( modifiers ), m_binary( binary )
122 {
123 }
124
125 gpCurve( const gpCurve &gpc )
126 {
127 m_file = gpc.m_file;
128 m_title = gpc.m_title;
129 m_modifiers = gpc.m_modifiers;
130 m_binary = gpc.m_binary;
131 }
132};
133
134/// An interactive c++ interface to gnuplot
135/** Spawns a gnuplot sesssion and communicates with it.
136 * \ingroup plotting
137 *
138 * \todo Use mxError for error reporting.
139 *
140 * An example of using gnuPlot to plot data from a file:
141 * \code
142 * gnuPlot gp; //This automatically connects, and is now ready to plot.
143 *
144 * gp.plot("my_data_file.dat", "u 1:2 w l"); //Plots the data in the file
145 * \endcode
146 *
147 * Arrays can be plotted directly:
148 * \code
149 * std::vector<float> x;
150 * std::vector<double> y;
151 *
152 * //...code to populate x and y
153 *
154 * gp.plot(x, y); //Plots y vs x.
155 * \endcode
156 * Note that the vector data types are different -- this is handled automatically.
157 *
158 * To get a response from the gnuplot session:
159 * \code
160 * std::cout << gp.getResponse("show terminal") << "\n"; //Sends the command to gnuplot, gets the response.
161 * \endcode
162 * Error checking and reporting is not straightforward since gnuplot does not return a result if there is no error. If
163 * there is an error, it can take several hundred milliseconds for a response to be available. So for a typical plot
164 * command, say, one does not want to try to read the stderr ouput just in case as this will just time out with a delay.
165 *
166 * For cases where a response is expected, the following example shows how to check for errors.
167 * \code
168 * errno = 0;
169 * std::string response = gp.getResponse("show terminal"); //Waits up to 0.5 seconds, or a user specified timeout.
170 *
171 * if(response == "") //This indicates some error occurred, or a timeout.
172 * {
173 * if(gp.gpError())
174 * {
175 * std::cerr << "gnuplot returned error:\n" << gp.gpErrorMsg() << "\n";
176 * }
177 * else if(errno)
178 * {
179 * perror("error getting response: ");
180 * }
181 * else
182 * {
183 * std::cerr << "timed out\n";
184 * }
185 * }
186 * else
187 * {
188 * std::cout << response << "\n";
189 * }
190 *
191 * \endcode
192 */
194{
195 protected:
196 int _connected{ 0 };
197
198 /// Set to true if the response indicates a gnuplot error
199 bool _gpError{ false };
200 std::string _gpErrorMsg;
201
202 /// File stream for the gnuplot interface
204
205 /// Where to create gnuplot stderr fifo
206 /** Default is /dev/shm/
207 */
208 std::string _errLocation;
209
210 /// File name of the gnuplot stderr fifo
211 std::string _errFName;
212
213 /// File descriptor for the gnuplot stderr fifo
214 int _errFD{ 0 };
215
216 /// Location of temporary files
217 /** Default is /dev/shm/
218 */
219 std::string _tempLocation;
220
221 /// Vector of all temporary file names opened, used for removal on destruction.
222 std::vector<std::string> _tempFiles;
223
224 /// Flag to control whether temporary files are deleted on destruction. Default is true (files deleted).
225 bool _deleteTemp{ true };
226
227 std::map<std::string, gpCurve> m_curveMap;
228
229 public:
230 gnuPlot();
231
232 ~gnuPlot();
233
234 /// Connect to gnuplot
235 /** Spawns a gnuplot session using popen with stderr redirected to a temporary file. The temporary file
236 * is opened for reading.
237 *
238 * \retval 0 on success
239 * \retval -1 on error
240 */
241 int connect();
242
243 /// Return the value of the gpError flag.
244 /** This flag is set if gnuplot returns an error message (see checkResponse()). The error message
245 * can be accessed with gpErrorMsg().
246 *
247 * \retval 0 if no error
248 * \retval -1 if an error has occurred.
249 */
250 bool gpError();
251
252 /// Return the gnuplot error message
253 /** The error message is extracted by checkResponse() when the response from gnuplot begins with "gnuplot>".
254 */
255 std::string gpErrorMsg();
256
257 /// Send a command to gnuplot
258 /** The newline is appended to the command, and then it is sent to gnuplot.
259 *
260 *
261 * \retval 0 on success
262 * \retval -1 on error
263 */
264 int command( const std::string &com, ///< [in] the command string
265 bool flush = true ///< [in] [optional] if true (default), then the output stream is flushed once the
266 ///< command is written
267 );
268
269 /// Check for a response from gnuplot.
270 /** It typically takes some time for a response from gnuplot to become readable,
271 * whether an error occurs or not. It is often 0.3 seconds or more. If there is no error, there will be no response
272 * so waiting for a response to check for errors after every command can be very time consuming. Thus it is a
273 * choice whether to check for errors from gnuplot after every command, and what timeout to use.
274 *
275 * \todo investigate having a second thread monitor for responses.
276 *
277 * gnuplot terminates all outputs with \c \\n\\n, so this reads up to these two characters. It then strips any
278 * leading and trailing whitespace. If the response begins with \c gnuplot>, then the response is an error. In this
279 * case, checkResponse returns -1 and the gpError flag is set. If gpError is not set but the return value is -1,
280 * then some other error occurred (check \c errno).
281 *
282 * \retval 0 on timeout or successful read
283 * \retval -1 on error, or if the gnuplot response indicates an error.
284 */
286 std::string &response, ///< [out] contains the response from gnuplot, but is empty on timeout or error.
287 double timeout = 0 ///< [in] [optional] the length of time, in seconds, to wait for a response. Default is 0,
288 ///< but a minimum of 0.5 if a response is expected.
289 );
290
291 /// Get a response from gnuplot for a given command.
292 /** This should only be used if a response is expected.
293 *
294 * \returns "" (empty string) on timeout or error (\c errno and gpError() should be checked), and the response from
295 * gnuplot otherwise.
296 */
297 std::string getResponse( const std::string &com, ///< [in] the command string
298 double timeout = 0.5 ///< [in] the length of time, in seconds, to wait for a response.
299 ///< Default is 0.5, which is the minimum that should be used.
300 );
301
302 /// Issue the \b replot command
303 /**
304 * \retval 0 on success
305 * \retval -1 on error
306 */
307 int replot();
308
309 /// Set the x-axis range
310 /** Sends the command:
311 * \verbatim
312 * set xrange [x0:x1]
313 * \endverbatim
314 *
315 * \returns 0 on success
316 * \returns -1 on error
317 */
318 int xrange( double x0, double x1 );
319
320 /// Unset the x-axis range
321 /** Sends the command:
322 * \verbatim
323 * unset xrange
324 * \endverbatim
325 *
326 * \returns 0 on success
327 * \returns -1 on error
328 */
329 int xrange();
330
331 /// Set the y-axis range
332 /** Sends the command:
333 * \verbatim
334 * set yrange [y0:y1]
335 * \endverbatim
336 *
337 * \returns 0 on success
338 * \returns -1 on error
339 */
340 int yrange( double y0, double y1 );
341
342 /// Unset the y-axis range
343 /** Sends the command:
344 * \verbatim
345 * unset yrange
346 * \endverbatim
347 *
348 * \returns 0 on success
349 * \returns -1 on error
350 */
351 int yrange();
352
353 /// Set the y axis to log scale
354 /** Sends the command:
355 * \verbatim
356 set log y
357 \endverbatim
358 *
359 * \retval 0 on success
360 * \retval -1 on error
361 */
362 int logy();
363
364 /// Set the x axis to log scale
365 /** Sends the command:
366 * \verbatim
367 set log x
368 \endverbatim
369 *
370 * \retval 0 on success
371 * \retval -1 on error
372 */
373 int logx();
374
375 /// Set the x and y axes to log scale
376 /** Sends the commands:
377 * \verbatim
378 set log x
379 set log y
380 \endverbatim
381 *
382 * \retval 0 on success
383 * \retval -1 on error
384 */
385 int logxy();
386
387 /// Unset the y axis from log scale
388 /** Sends the command:
389 * \verbatim
390 unset log y
391 \endverbatim
392 *
393 * \retval 0 on success
394 * \retval -1 on error
395 */
396 int ulogy();
397
398 /// Unset the x axis from log scale
399 /** Sends the command:
400 * \verbatim
401 unset log x
402 \endverbatim
403 *
404 * \retval 0 on success
405 * \retval -1 on error
406 */
407 int ulogx();
408
409 /// Unset the x and y axes from log scale
410 /** Sends the command:
411 * \verbatim
412 unset log x
413 unset log y
414 \endverbatim
415 *
416 * \retval 0 on success
417 * \retval -1 on error
418 */
419 int ulogxy();
420
421 /// Plot from a file specifying all curve components.
422 /** Forms the gnuplot plot command as follows:
423 * \verbatim
424 * plot '<fname>' <modifiers> t '<title>'
425 * \endverbatim
426 * The modifiers string can contain any modifiers such as \a using, etc. Should generally not include \a title.
427 *
428 * \retval 0 on success
429 * \retval -1 on error
430 */
431 int plot( const std::string &fname, ///< [in] the name (with full path) of the file containing data to plot
432 const std::string &modifiers, ///< [in] contains any modifiers to the plot command.
433 const std::string &title, ///< [in] the title for this curve
434 const std::string &name ///< [in] the name for this curve. If "" then curve# is used.
435 );
436
437 /// Plot from a file without specifying the title
438 /** Forms the gnuplot plot command as follows:
439 * \verbatim
440 * plot '<fname>' <modifiers> t ''
441 * \endverbatim
442 * The modifiers string can contain any modifiers such as \a using, etc. Should generally not include \a title.
443 *
444 * \retval 0 on success
445 * \retval -1 on error
446 *
447 * \overload
448 */
449 int plot( const std::string &fname, ///< [in] the name (with full path) of the file containing data to plot
450 const std::string &modifiers, ///< [in] contains any modifiers to the plot command.
451 const std::string &name ///< [in] the name for this curve. If "" then the curve# is used.
452 );
453
454 /// Plot from a file without modifiers
455 /** The file name is used for the identifying name
456 *
457 * Forms the gnuplot plot command as follows:
458 * \verbatim
459 * plot '<fname>' <modifiers> t ''
460 * \endverbatim
461 * The modifiers string can contain any modifiers such as \a using, etc. Should generally not include \a title.
462 *
463 * \retval 0 on success
464 * \retval -1 on error
465 *
466 * \overload
467 */
468 int plot( const std::string &fname, ///< [in] the name (with full path) of the file containing data to plot
469 const std::string &name ///< [in] the name for this curve. If "" then the curve# is used
470 );
471
472 /// Plot from a file without specifying the name or any modifiers
473 /** The file name is used for the identifying name
474 *
475 * Forms the gnuplot plot command as follows:
476 * \verbatim
477 * plot '<fname>' t ''
478 * \endverbatim
479 * The modifiers string can contain any modifiers such as \a using, etc. Should generally not include \a title.
480 *
481 * \retval 0 on success
482 * \retval -1 on error
483 *
484 * \overload
485 */
486 int plot( const std::string &fname /**< [in] the name (with full path) of the file containing data to plot */ );
487
488 /// Plot data from an array
489 /** Copies the data in the array to a temporary binary file, and then forms the gnuplot plot command as follows:
490 * \verbatim
491 * plot "temp-file-name" binary format="%dataT" u 1 t "title" <modifiers>
492 * \endverbatim
493 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
494 * can contain any other modifiers. Title is specified so that the name of the temporary file name is not printed
495 * on the plot.
496 *
497 * \retval 0 on success
498 * \retval -1 on error
499 */
500 template <typename dataT>
501 int plot( const dataT *y, ///< [in] a pointer to an array of data
502 size_t N, ///< [in] the length of the array
503 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
504 ///< \b format, \b using, and \b title.
505 const std::string &
506 title, ///< [in] contains the title of the data set, default is an empty string and no key on the plot
507 const std::string &name ///< [in] the identifying name of this curve
508 );
509
510 /// Plot data from an array
511 /** Copies the data in the array to a temporary binary file, and then forms the gnuplot plot command as follows:
512 * \verbatim
513 * plot "temp-file-name" binary format="%dataT" u 1 t '' <modifiers>
514 * \endverbatim
515 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
516 * can contain any other modifiers. Title is specified so that the name of the temporary file name is not printed
517 * on the plot.
518 *
519 * \retval 0 on success
520 * \retval -1 on error
521 */
522 template <typename dataT>
523 int plot( const dataT *y, ///< [in] a pointer to an array of data
524 size_t N, ///< [in] the length of the array
525 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
526 ///< \b format, \b using, and \b title.
527 const std::string &name ///< [in] the identifying name of this curve
528 );
529
530 /// Plot data from an array
531 /** Copies the data in the array to a temporary binary file, and then forms the gnuplot plot command as follows:
532 * \verbatim
533 * plot "temp-file-name" binary format="%dataT" u 1 t '' <modifiers>
534 * \endverbatim
535 *
536 *
537 * \retval 0 on success
538 * \retval -1 on error
539 */
540 template <typename dataT>
541 int plot( const dataT *y, ///< [in] a pointer to an array of data
542 size_t N, ///< [in] the length of the array
543 const std::string &name ///< [in] the identifying name of this curve
544 );
545
546 /// Plot data from an array
547 /** Copies the data in the array to a temporary binary file, and then forms the gnuplot plot command as follows:
548 * \verbatim
549 * plot "temp-file-name" binary format="%dataT" u 1 t ''
550 * \endverbatim
551 *
552 * \retval 0 on success
553 * \retval -1 on error
554 */
555 template <typename dataT>
556 int plot( const dataT *y, ///< [in] a pointer to an array of data
557 size_t N ///< [in] the length of the array
558 );
559
560 /// Plot data from a vector
561 /** Copies the data in the vector to a temporary binary file, and then forms the gnuplot plot command as follows:
562 * \verbatim
563 * plot "temp-file-name" binary format="%dataT" u 1 t "title" <modifiers>
564 * \endverbatim
565 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
566 * can contain any other modifiers. Title is specified so that the name of the temporary file name is not printed
567 * on the plot.
568 *
569 * \retval 0 on success
570 * \retval -1 on error
571 */
572 template <typename dataT>
573 int plot( const std::vector<dataT> &y, ///< [in] the vector containing the data
574 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
575 ///< \b format, \b using, and \b title.
576 const std::string &
577 title, ///< [in] contains the title of the data set, default is an empty string and no key on the plot
578 const std::string &name ///< [in] the identifying name of this curve
579 );
580
581 /// Plot data from a vector
582 /** Copies the data in the vector to a temporary binary file, and then forms the gnuplot plot command as follows:
583 * \verbatim
584 * plot "temp-file-name" binary format="%dataT" u 1 t '' <modifiers>
585 * \endverbatim
586 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
587 * can contain any other modifiers. Title is specified so that the name of the temporary file name is not printed
588 * on the plot.
589 *
590 * \retval 0 on success
591 * \retval -1 on error
592 */
593 template <typename dataT>
594 int plot( const std::vector<dataT> &y, ///< [in] the vector containing the data
595 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
596 ///< \b format, \b using, and \b title.
597 const std::string &name ///< [in] the identifying name of this curve
598 );
599
600 /// Plot data from a vector
601 /** Copies the data in the vector to a temporary binary file, and then forms the gnuplot plot command as follows:
602 * \verbatim
603 * plot "temp-file-name" binary format="%dataT" u 1 t ''
604 * \endverbatim
605 *
606 * \retval 0 on success
607 * \retval -1 on error
608 */
609 template <typename dataT>
610 int plot( const std::vector<dataT> &y, ///< [in] the vector containing the data
611 const std::string &name ///< [in] the identifying name of this curve
612 );
613
614 /// Plot data from a vector
615 /** Copies the data in the vector to a temporary binary file, and then forms the gnuplot plot command as follows:
616 * \verbatim
617 * plot "temp-file-name" binary format="%dataT" u 1 t ''
618 * \endverbatim
619 *
620 * \retval 0 on success
621 * \retval -1 on error
622 */
623 template <typename dataT>
624 int plot( const std::vector<dataT> &y /**< [in] the vector containing the data*/ );
625
626 /// Plot y vs. x data from arrays
627 /** Copies the data in the arrays to a temporary binary file, and then forms the gnuplot plot command as follows:
628 * \verbatim
629 plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t "title" <modifiers>
630 \endverbatim
631 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
632 can contain any other modifiers. Title is
633 * specified so that the name of the temporary file name is not printed on the plot.
634 *
635 * \retval 0 on success
636 * \retval -1 on error
637 */
638 template <typename dataTx, typename dataTy>
639 int plot( const dataTx *x, ///< [in] a pointer to an array of data for the independent variable
640 const dataTy *y, ///< [in] a pointer to an array of data for the dependent variable
641 size_t N, ///< [in] the length of the arrays
642 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
643 ///< \b format, \b using, and \b title.
644 const std::string &title, ///< [in] contains the title of the data set
645 const std::string &name ///< [in] the identifying name of the curve
646 );
647
648 /// Plot y vs. x data from arrays
649 /** Copies the data in the arrays to a temporary binary file, and then forms the gnuplot plot command as follows:
650 * \verbatim
651 * plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t '' <modifiers>
652 * \endverbatim
653 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
654 * can contain any other modifiers. Title is specified so that the name of the temporary file name is not printed
655 * on the plot.
656 *
657 * \retval 0 on success
658 * \retval -1 on error
659 */
660 template <typename dataTx, typename dataTy>
661 int plot( const dataTx *x, ///< [in] a pointer to an array of data for the independent variable
662 const dataTy *y, ///< [in] a pointer to an array of data for the dependent variable
663 size_t N, ///< [in] the length of the arrays
664 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
665 ///< \b format, \b using, and \b title.
666 const std::string &name ///< [in] the identifying name of the curve
667 );
668
669 /// Plot y vs. x data from arrays
670 /** Copies the data in the arrays to a temporary binary file, and then forms the gnuplot plot command as follows:
671 * \verbatim
672 * plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t ''
673 * \endverbatim
674 *
675 * \retval 0 on success
676 * \retval -1 on error
677 */
678 template <typename dataTx, typename dataTy>
679 int plot( const dataTx *x, ///< [in] a pointer to an array of data for the independent variable
680 const dataTy *y, ///< [in] a pointer to an array of data for the dependent variable
681 size_t N, ///< [in] the length of the arrays
682 const std::string &name ///< [in] the identifying name of the curve
683 );
684
685 /// Plot y vs. x data from arrays
686 /** Copies the data in the arrays to a temporary binary file, and then forms the gnuplot plot command as follows:
687 * \verbatim
688 * plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t ''
689 * \endverbatim
690 *
691 * \retval 0 on success
692 * \retval -1 on error
693 */
694 template <typename dataTx, typename dataTy>
695 int plot( const dataTx *x, ///< [in] a pointer to an array of data for the independent variable
696 const dataTy *y, ///< [in] a pointer to an array of data for the dependent variable
697 size_t N ///< [in] the length of the arrays
698 );
699
700 /// Plot y vs. x data from vectors
701 /** Copies the data in the vectors to a temporary binary file, and then forms the gnuplot plot command as follows:
702 * \verbatim
703 plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t "title" <modifiers>
704 \endverbatim
705 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
706 can contain any other modifiers. Title is
707 * specified so that the name of the temporary file name is not printed on the plot.
708 *
709 * \retval 0 on success
710 * \retval -1 on error
711 */
712 template <typename dataTx, typename dataTy>
713 int plot( const std::vector<dataTx> &x, ///< [in] a vector of data for the independent variable
714 const std::vector<dataTy> &y, ///< [in] a vector of data for the dependent variable
715 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
716 ///< \b format, \b using, and \b title.
717 const std::string &title, ///< [in] contains the title of the data set
718 const std::string &name ///< [in] the identifying name of the curve
719 );
720
721 /// Plot y vs. x data from vectors
722 /** Copies the data in the vectors to a temporary binary file, and then forms the gnuplot plot command as follows:
723 * \verbatim
724 plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t '' <modifiers>
725 \endverbatim
726 * The modifiers string \b must \b NOT contain the \b binary, \b format, \b using, or the \b title modifiers, but
727 can contain any other modifiers. Title is
728 * specified so that the name of the temporary file name is not printed on the plot.
729 *
730 * \retval 0 on success
731 * \retval -1 on error
732 */
733 template <typename dataTx, typename dataTy>
734 int plot( const std::vector<dataTx> &x, ///< [in] a vector of data for the independent variable
735 const std::vector<dataTy> &y, ///< [in] a vector of data for the dependent variable
736 const std::string &modifiers, ///< [in] contains any modifiers to the plot command other than \b binary,
737 ///< \b format, \b using, and \b title.
738 const std::string &name ///< [in] the identifying name of the curve
739 );
740
741 /// Plot y vs. x data from vectors
742 /** Copies the data in the vectors to a temporary binary file, and then forms the gnuplot plot command as follows:
743 * \verbatim
744 * plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t ''
745 * \endverbatim
746 *
747 * \retval 0 on success
748 * \retval -1 on error
749 */
750 template <typename dataTx, typename dataTy>
751 int plot( const std::vector<dataTx> &x, ///< [in] a vector of data for the independent variable
752 const std::vector<dataTy> &y, ///< [in] a vector of data for the dependent variable
753 const std::string &name ///< [in] the identifying name of the curve
754 );
755
756 /// Plot y vs. x data from vectors
757 /** Copies the data in the vectors to a temporary binary file, and then forms the gnuplot plot command as follows:
758 * \verbatim
759 * plot "temp-file-name" binary format="%dataTx%dataTy" u 1:2 t ''
760 * \endverbatim
761 *
762 * \retval 0 on success
763 * \retval -1 on error
764 */
765 template <typename dataTx, typename dataTy>
766 int plot( const std::vector<dataTx> &x, ///< [in] a vector of data for the independent variable
767 const std::vector<dataTy> &y ///< [in] a vector of data for the dependent variable
768 );
769
770 /// Plot a single point
771 /** Copies the position as a length-1 vector to a temporary binary file, and then plots it.
772 *
773 *
774 * \retval 0 on success
775 * \retval -1 on error
776 */
777 template <typename dataTx, typename dataTy>
778 int point( dataTx x, ///< [in] the independent axis (x) coordinate of the point
779 dataTy y, ///< [in] the dependent axis (y) coordinate of the point
780 const std::string &modifiers = "", ///< [in] [optional] contains any modifiers to the plot command other
781 ///< than \b binary, \b format, \b using, and \b title.
782 const std::string &title = "" ///< [in] [optional] contains the title of the data set, default is an
783 ///< empty string and no key on the plot
784 );
785
786 /// Draw a circle on the plot
787 /** Creates a circle with specified center and radius, and plots it.
788 */
789 int circle( double xcen, ///< [in] the x-coordinate of the circle center
790 double ycen, ///< [in] the y-coordinate of the circle center
791 double radius, ///< [in] the circle radius
792 const std::string &modifiers = "", ///< [in] [optional] contains any modifiers to the plot command other
793 ///< than \b binary, \b format, \b using, and \b title.
794 const std::string &title = "", ///< [in] [optional] contains the title of the data set, default is an
795 ///< empty string and no key on the plot
796 int npoints = 10 ///< [in] [optional] specifies the number of points in each half circle. Default 10 is
797 ///< usually sufficient.
798 );
799
800 /// Draw a circle on the plot around the origin
801 /** Creates a circle with radius and center (0,0), and plots it.
802 */
803 int circle( double radius, ///< [in] the circle radius
804 const std::string &modifiers = "", ///< [in] [optional] contains any modifiers to the plot command other
805 ///< than \b binary, \b format, \b using, and \b title.
806 const std::string &title = "", ///< [in] [optional] contains the title of the data set, default is an
807 ///< empty string and no key on the plot
808 int npoints = 10 ///< [in] [optional] specifies the number of points in each half circle. Default 10 is
809 ///< usually sufficient.
810 );
811
812 /// Clear all the curves
813 void clear();
814
815 /// Clear all the curves and reset plot configuration
816 void reset();
817
818 /// List the curves and their specifications.
820
821 /// Modify the modifiers for a curve
822 void modifiers( const std::string &name, ///< [in] the name of the curve
823 const std::string &modifiers ///< [in] the new modifiers
824 );
825
826 /// Modify the title for a curve
827 void title( const std::string &name, ///< [in] the name of the curve
828 const std::string &title ///< [in] the new title
829 );
830
831 protected:
832 /// Implementation of 1-D binary plotting.
833 int plotImpl( const void *y,
834 size_t Nbytes,
835 const std::string &binary,
836 const std::string &modifiers,
837 const std::string &title,
838 const std::string &name );
839
840 /// Implementation of 2-d binary plotting.
841 int plotImpl( const void *x,
842 const void *y,
843 size_t Npts,
844 size_t sizex,
845 size_t sizey,
846 const std::string &binaryx,
847 const std::string &binaryy,
848 const std::string &modifiers,
849 const std::string &title,
850 const std::string &name );
851
852 /// Issue the plot command for all the curves
854
855 /// Open a temporary binary file, and provide the filename.
857
858 private:
859 /// Initialize an instance, only used at construction
860 void init();
861};
862
863template <typename dataT>
865 const dataT *y, size_t N, const std::string &modifiers, const std::string &title, const std::string &name )
866{
867 return plotImpl( y, N * sizeof( dataT ), gpBinaryFormat<dataT>(), modifiers, title, name );
868}
869
870template <typename dataT>
871int gnuPlot::plot( const dataT *y, size_t N, const std::string &modifiers, const std::string &name )
872{
873 return plotImpl( y, N * sizeof( dataT ), gpBinaryFormat<dataT>(), modifiers, "", name );
874}
875
876template <typename dataT>
877int gnuPlot::plot( const dataT *y, size_t N, const std::string &name )
878{
879 return plotImpl( y, N * sizeof( dataT ), gpBinaryFormat<dataT>(), "", "", name );
880}
881
882template <typename dataT>
883int gnuPlot::plot( const dataT *y, size_t N )
884{
885 return plotImpl( y, N * sizeof( dataT ), gpBinaryFormat<dataT>(), "", "", "" );
886}
887
888template <typename dataT>
889int gnuPlot::plot( const std::vector<dataT> &y,
890 const std::string &modifiers,
891 const std::string &title,
892 const std::string &name )
893{
894 return plot<dataT>( y.data(), y.size(), modifiers, title, name );
895}
896
897template <typename dataT>
898int gnuPlot::plot( const std::vector<dataT> &y, const std::string &modifiers, const std::string &name )
899{
900 return plot<dataT>( y.data(), y.size(), modifiers, "", name );
901}
902
903template <typename dataT>
904int gnuPlot::plot( const std::vector<dataT> &y, const std::string &name )
905{
906 return plot<dataT>( y.data(), y.size(), "", "", name );
907}
908
909template <typename dataT>
910int gnuPlot::plot( const std::vector<dataT> &y )
911{
912 return plot<dataT>( y.data(), y.size(), "", "", "" );
913}
914
915template <typename dataTx, typename dataTy>
916int gnuPlot::plot( const dataTx *x,
917 const dataTy *y,
918 size_t N,
919 const std::string &modifiers,
920 const std::string &title,
921 const std::string &name )
922{
923 return plotImpl( x,
924 y,
925 N,
926 sizeof( dataTx ),
927 sizeof( dataTy ),
930 modifiers,
931 title,
932 name );
933}
934
935template <typename dataTx, typename dataTy>
936int gnuPlot::plot( const dataTx *x, const dataTy *y, size_t N, const std::string &modifiers, const std::string &name )
937{
938 return plotImpl( x,
939 y,
940 N,
941 sizeof( dataTx ),
942 sizeof( dataTy ),
945 modifiers,
946 "",
947 name );
948}
949
950template <typename dataTx, typename dataTy>
951int gnuPlot::plot( const dataTx *x, const dataTy *y, size_t N, const std::string &name )
952{
953 return plotImpl(
954 x, y, N, sizeof( dataTx ), sizeof( dataTy ), gpBinaryFormat<dataTx>(), gpBinaryFormat<dataTy>(), "", "", name );
955}
956
957template <typename dataTx, typename dataTy>
958int gnuPlot::plot( const dataTx *x, const dataTy *y, size_t N )
959{
960 return plotImpl(
961 x, y, N, sizeof( dataTx ), sizeof( dataTy ), gpBinaryFormat<dataTx>(), gpBinaryFormat<dataTy>(), "", "", "" );
962}
963
964template <typename dataTx, typename dataTy>
965int gnuPlot::plot( const std::vector<dataTx> &x,
966 const std::vector<dataTy> &y,
967 const std::string &modifiers,
968 const std::string &title,
969 const std::string &name )
970{
971 return plot( x.data(), y.data(), x.size(), modifiers, title, name );
972}
973
974template <typename dataTx, typename dataTy>
975int gnuPlot::plot( const std::vector<dataTx> &x,
976 const std::vector<dataTy> &y,
977 const std::string &modifiers,
978 const std::string &name )
979{
980 return plot( x.data(), y.data(), x.size(), modifiers, "", name );
981}
982
983template <typename dataTx, typename dataTy>
984int gnuPlot::plot( const std::vector<dataTx> &x, const std::vector<dataTy> &y, const std::string &name )
985{
986 return plot( x.data(), y.data(), x.size(), "", "", name );
987}
988
989template <typename dataTx, typename dataTy>
990int gnuPlot::plot( const std::vector<dataTx> &x, const std::vector<dataTy> &y )
991{
992 return plot( x.data(), y.data(), x.size(), "", "", "" );
993}
994
995template <typename dataTx, typename dataTy>
996int gnuPlot::point( dataTx x, dataTy y, const std::string &modifiers, const std::string &title )
997{
998
999 return plot( &x, &y, 1, modifiers, title );
1000}
1001
1002template <>
1003std::string gpBinaryFormat<char>();
1004
1005template <>
1006std::string gpBinaryFormat<unsigned char>();
1007
1008template <>
1009std::string gpBinaryFormat<short>();
1010
1011template <>
1012std::string gpBinaryFormat<unsigned short>();
1013
1014template <>
1015std::string gpBinaryFormat<int>();
1016
1017template <>
1018std::string gpBinaryFormat<unsigned int>();
1019
1020template <>
1021std::string gpBinaryFormat<long>();
1022
1023template <>
1024std::string gpBinaryFormat<unsigned long>();
1025
1026template <>
1027std::string gpBinaryFormat<float>();
1028
1029template <>
1030std::string gpBinaryFormat<double>();
1031
1032} // namespace math
1033} // namespace mx
1034
1035#endif // math_plot_gnuPlot_hpp
An interactive c++ interface to gnuplot.
Definition gnuPlot.hpp:194
int circle(double radius, const std::string &modifiers="", const std::string &title="", int npoints=10)
Draw a circle on the plot around the origin.
int doPlotCommand()
Issue the plot command for all the curves.
int plot(const std::string &fname)
Plot from a file without specifying the name or any modifiers.
std::string getResponse(const std::string &com, double timeout=0.5)
Get a response from gnuplot for a given command.
FILE * _pipeH
File stream for the gnuplot interface.
Definition gnuPlot.hpp:203
int plot(const std::string &fname, const std::string &modifiers, const std::string &title, const std::string &name)
Plot from a file specifying all curve components.
bool _gpError
Set to true if the response indicates a gnuplot error.
Definition gnuPlot.hpp:199
int xrange(double x0, double x1)
Set the x-axis range.
void title(const std::string &name, const std::string &title)
Modify the title for a curve.
int plot(const std::string &fname, const std::string &modifiers, const std::string &name)
Plot from a file without specifying the title.
int plotImpl(const void *x, const void *y, size_t Npts, size_t sizex, size_t sizey, const std::string &binaryx, const std::string &binaryy, const std::string &modifiers, const std::string &title, const std::string &name)
Implementation of 2-d binary plotting.
int plot(const std::string &fname, const std::string &name)
Plot from a file without modifiers.
void reset()
Clear all the curves and reset plot configuration.
int connect()
Connect to gnuplot.
int point(dataTx x, dataTy y, const std::string &modifiers="", const std::string &title="")
Plot a single point.
Definition gnuPlot.hpp:996
FILE * openTempFile(char *fname)
Open a temporary binary file, and provide the filename.
std::string _errFName
File name of the gnuplot stderr fifo.
Definition gnuPlot.hpp:211
std::vector< std::string > _tempFiles
Vector of all temporary file names opened, used for removal on destruction.
Definition gnuPlot.hpp:222
int ulogy()
Unset the y axis from log scale.
int plotImpl(const void *y, size_t Nbytes, const std::string &binary, const std::string &modifiers, const std::string &title, const std::string &name)
Implementation of 1-D binary plotting.
bool _deleteTemp
Flag to control whether temporary files are deleted on destruction. Default is true (files deleted).
Definition gnuPlot.hpp:225
int logy()
Set the y axis to log scale.
int _errFD
File descriptor for the gnuplot stderr fifo.
Definition gnuPlot.hpp:214
std::string _errLocation
Where to create gnuplot stderr fifo.
Definition gnuPlot.hpp:208
int ulogxy()
Unset the x and y axes from log scale.
int replot()
Issue the replot command.
int logxy()
Set the x and y axes to log scale.
int yrange()
Unset the y-axis range.
void listCurves()
List the curves and their specifications.
int checkResponse(std::string &response, double timeout=0)
Check for a response from gnuplot.
bool gpError()
Return the value of the gpError flag.
void modifiers(const std::string &name, const std::string &modifiers)
Modify the modifiers for a curve.
int xrange()
Unset the x-axis range.
int ulogx()
Unset the x axis from log scale.
int yrange(double y0, double y1)
Set the y-axis range.
std::string _tempLocation
Location of temporary files.
Definition gnuPlot.hpp:219
int circle(double xcen, double ycen, double radius, const std::string &modifiers="", const std::string &title="", int npoints=10)
Draw a circle on the plot.
int command(const std::string &com, bool flush=true)
Send a command to gnuplot.
int logx()
Set the x axis to log scale.
std::string gpErrorMsg()
Return the gnuplot error message.
void clear()
Clear all the curves.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106