123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- /* This file is part of the GNU plotutils package. Copyright (C) 1995,
- 1996, 1997, 1998, 1999, 2000, 2005, 2008, Free Software Foundation, Inc.
- The GNU plotutils package is free software. You may redistribute it
- and/or modify it under the terms of the GNU General Public License as
- published by the Free Software foundation; either version 2, or (at your
- option) any later version.
- The GNU plotutils package is distributed in the hope that it will be
- useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- General Public License for more details.
- You should have received a copy of the GNU General Public License along
- with the GNU plotutils package; see the file COPYING. If not, write to
- the Free Software Foundation, Inc., 51 Franklin St., Fifth Floor,
- Boston, MA 02110-1301, USA. */
- /* This file defines the initialization for any XPlotter object, including
- both private data and public methods. There is a one-to-one
- correspondence between public methods and user-callable functions in the
- C API. */
- #include "sys-defines.h"
- #include <signal.h> /* for kill() */
- #include "extern.h"
- /* Sparse array of pointers to XPlotter instances, and its size. Should be
- initialized to a NULL pointer, and 0, respectively.
- In libplotter these are not global variables: in extern.h, they are
- #define'd to be static data members of the XPlotter class.
- Accessed by _pl_y_initialize() and _pl_y_terminate() in this file, by
- _pl_y_maybe_handle_x_events() in y_openpl.c, and by _pl_y_closepl() in
- y_closepl.c. */
- XPlotter **_xplotters = NULL;
- int _xplotters_len = 0;
- /* initial size for "_xplotters", the sparse array of XPlotter instances */
- #define INITIAL_XPLOTTERS_LEN 4
- /* Mutex for locking _xplotters[] and _xplotters_len. */
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- pthread_mutex_t _xplotters_mutex = PTHREAD_MUTEX_INITIALIZER;
- #endif
- #endif
- #ifndef LIBPLOTTER
- /* In libplot, this is the initialization for the function-pointer part of
- an XPlotter struct. */
- const Plotter _pl_y_default_plotter =
- {
- /* initialization (after creation) and termination (before deletion) */
- _pl_y_initialize, _pl_y_terminate,
- /* page manipulation */
- _pl_y_begin_page, _pl_y_erase_page, _pl_y_end_page,
- /* drawing state manipulation */
- _pl_x_push_state, _pl_x_pop_state,
- /* internal path-painting methods (endpath() is a wrapper for the first) */
- _pl_x_paint_path, _pl_x_paint_paths, _pl_x_path_is_flushable, _pl_x_maybe_prepaint_segments,
- /* internal methods for drawing of markers and points */
- _pl_g_paint_marker, _pl_x_paint_point,
- /* internal methods that plot strings in Hershey, non-Hershey fonts */
- _pl_g_paint_text_string_with_escapes, _pl_x_paint_text_string,
- _pl_x_get_text_width,
- /* private low-level `retrieve font' method */
- _pl_x_retrieve_font,
- /* `flush output' method, called only if Plotter handles its own output */
- _pl_x_flush_output,
- /* internal error handlers */
- _pl_g_warning,
- _pl_g_error,
- };
- #endif /* not LIBPLOTTER */
- /* The private `initialize' method, which is invoked when a Plotter is
- created. It is used for such things as initializing capability flags
- from the values of class variables, allocating storage, etc. When this
- is invoked, _plotter points to the Plotter that has just been
- created. */
- void
- _pl_y_initialize (S___(Plotter *_plotter))
- {
- bool open_slot = false;
- int i, j;
- #ifndef LIBPLOTTER
- /* in libplot, manually invoke superclass initialization method */
- _pl_x_initialize (S___(_plotter));
- #endif
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- /* lock the global variables _xplotters[] and _xplotters_len */
- pthread_mutex_lock (&_xplotters_mutex);
- #endif
- #endif
- /* If this is the first XPlotter to be created, initialize Xt library.
- At least in X11R6, it's OK to initialize it more than once, but we're
- careful. */
- if (_xplotters_len == 0)
- {
- /* first initialize Xlib and Xt thread support if any */
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- #ifdef X_THREAD_SUPPORT
- XInitThreads ();
- XtToolkitThreadInitialize ();
- #endif
- #endif
- #endif
- /* initialize Xt library itself */
- XtToolkitInitialize ();
- }
- /* ensure XPlotter array is set up */
- if (_xplotters_len == 0)
- {
- _xplotters = (XPlotter **)_pl_xmalloc (INITIAL_XPLOTTERS_LEN * sizeof(XPlotter *));
- for (i = 0; i < INITIAL_XPLOTTERS_LEN; i++)
- _xplotters[i] = (XPlotter *)NULL;
- _xplotters_len = INITIAL_XPLOTTERS_LEN;
- }
-
- /* be sure there is an open slot (slot i) */
- for (i = 0; i < _xplotters_len; i++)
- if (_xplotters[i] == NULL)
- {
- open_slot = true;
- break;
- }
- if (!open_slot)
- /* expand array, clearing upper half */
- {
- i = _xplotters_len;
- _xplotters =
- (XPlotter **)_pl_xrealloc (_xplotters,
- 2 * _xplotters_len * sizeof (XPlotter *));
- for (j = _xplotters_len; j < 2 * _xplotters_len; j++)
- _xplotters[j] = (XPlotter *)NULL;
- _xplotters_len *= 2;
- }
-
- /* place just-created Plotter in open slot */
- _xplotters[i] = _plotter;
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- /* unlock the global variables _xplotters[] and _xplotters_len */
- pthread_mutex_unlock (&_xplotters_mutex);
- #endif
- #endif
- /* override superclass initializations, as necessary */
- #ifndef LIBPLOTTER
- /* tag field, differs in derived classes */
- _plotter->data->type = PL_X11;
- #endif
- /* output model */
- _plotter->data->output_model = PL_OUTPUT_VIA_CUSTOM_ROUTINES_TO_NON_STREAM;
- /* initialize data members specific to this derived class */
- _plotter->y_app_con = (XtAppContext)NULL;
- _plotter->y_toplevel = (Widget)NULL;
- _plotter->y_canvas = (Widget)NULL;
- _plotter->y_drawable4 = (Drawable)0;
- _plotter->y_auto_flush = true;
- _plotter->y_vanish_on_delete = false;
- _plotter->y_pids = (pid_t *)NULL;
- _plotter->y_num_pids = 0;
- _plotter->y_event_handler_count = 0;
- /* initialize certain data members from device driver parameters */
- /* determine whether to do an XFlush() after each drawing operation */
- {
- const char *vanish_s;
- vanish_s = (const char *)_get_plot_param (_plotter->data,
- "X_AUTO_FLUSH");
- if (strcasecmp (vanish_s, "no") == 0)
- _plotter->y_auto_flush = false;
- else
- _plotter->y_auto_flush = true;
- }
- /* determine whether windows vanish on Plotter deletion */
- {
- const char *vanish_s;
- vanish_s = (const char *)_get_plot_param (_plotter->data,
- "VANISH_ON_DELETE");
- if (strcasecmp (vanish_s, "yes") == 0)
- _plotter->y_vanish_on_delete = true;
- else
- _plotter->y_vanish_on_delete = false;
- }
- }
- /* The private `terminate' method, which is invoked when a Plotter is
- deleted. It may do such things as write to an output stream from
- internal storage, deallocate storage, etc. When this is invoked,
- _plotter points to the Plotter that is about to be deleted. */
- void
- _pl_y_terminate (S___(Plotter *_plotter))
- {
- int i, j;
- /* kill forked-off processes that are maintaining XPlotter's popped-up
- windows, provided that the VANISH_ON_DELETE parameter was set to "yes"
- at creation time */
- if (_plotter->y_vanish_on_delete)
- {
- for (j = 0; j < _plotter->y_num_pids; j++)
- kill (_plotter->y_pids[j], SIGKILL);
- if (_plotter->y_num_pids > 0)
- {
- free (_plotter->y_pids);
- _plotter->y_pids = (pid_t *)NULL;
- }
- }
- /* remove XPlotter from sparse XPlotter array */
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- /* lock the global variables _xplotters[] and _xplotters_len */
- pthread_mutex_lock (&_xplotters_mutex);
- #endif
- #endif
- for (i = 0; i < _xplotters_len; i++)
- if (_xplotters[i] == _plotter)
- {
- _xplotters[i] = (XPlotter *)NULL;
- break;
- }
- #ifdef PTHREAD_SUPPORT
- #ifdef HAVE_PTHREAD_H
- /* unlock the global variables _xplotters[] and _xplotters_len */
- pthread_mutex_unlock (&_xplotters_mutex);
- #endif
- #endif
- #ifndef LIBPLOTTER
- /* in libplot, manually invoke superclass termination method */
- _pl_x_terminate (S___(_plotter));
- #endif
- }
- #ifdef LIBPLOTTER
- XPlotter::XPlotter (FILE *infile, FILE *outfile, FILE *errfile)
- :XDrawablePlotter (infile, outfile, errfile)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (FILE *outfile)
- :XDrawablePlotter (outfile)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (istream& in, ostream& out, ostream& err)
- : XDrawablePlotter (in, out, err)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (ostream& out)
- : XDrawablePlotter (out)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter ()
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (FILE *infile, FILE *outfile, FILE *errfile, PlotterParams ¶meters)
- :XDrawablePlotter (infile, outfile, errfile, parameters)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (FILE *outfile, PlotterParams ¶meters)
- :XDrawablePlotter (outfile, parameters)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (istream& in, ostream& out, ostream& err, PlotterParams ¶meters)
- : XDrawablePlotter (in, out, err, parameters)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (ostream& out, PlotterParams ¶meters)
- : XDrawablePlotter (out, parameters)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::XPlotter (PlotterParams ¶meters)
- : XDrawablePlotter (parameters)
- {
- _pl_y_initialize ();
- /* add to _xplotters sparse array */
- }
- XPlotter::~XPlotter ()
- {
- /* if luser left the Plotter open, close it */
- if (_plotter->data->open)
- _API_closepl ();
- _pl_y_terminate ();
- /* remove from _xplotters sparse array */
- }
- #endif
|