/* This file contains the point plotter half of GNU graph.  The point
   plotter could also be linked with other software.  It translates a
   sequence of points, regarded as defining a polyline or a sequence of
   polylines, to a sequence of libplot calls.
   
   A `point' is a structure.  Each point structure contains the following
   fields:

      x and y coordinates of the point
      xmin and xmax (meaningful only if have_x_errorbar is set, see below)
      ymin and ymax (meaningful only if have_y_errorbar is set, see below)
      a `pendown' flag

      a symbol type (a small integer, interpreted as a marker type)
      a symbol size (a fraction of the size of the plotting box)
      a symbol font name (relevant only for symbol types >= 32)
      a linemode (a small integer)
      a linewidth (a fraction of the size of the display device)
      a polyline fill-fraction (in the interval [0,1], <0 means no fill)
      a use_color flag (true or false)
      a `have_x_errorbar' flag (true or false)
      a `have_y_errorbar' flag (true or false)

   The point plotter constructs a polyline from each successive run of
   points that have the pendown flag set.  The final nine fields are
   assumed to be the same for each point in such a run.  At each point on
   the polyline, the appropriate marker symbol, if any, will be plotted.
   Symbol types greater than or equal to 32 are interpreted as single
   characters to be plotted, rather than symbols.

   Points without the pendown flag set cause the polyline to be broken, and
   a new one to begin, before the symbol (if any) is plotted.  

   The plotter supports five basic linemodes: 1 through 5.  The
   interpretation of `linemode' depends on the polyline's use_color flag.

	   linemode		If monochrome 		If color

	       1		solid			red
	       2		dotted			green
	       3		dotdashed		blue
	       4		shortdashed		magenta
	       5		longdashed		cyan

   In the monochrome case, the pattern simply repeats: 6,7,8,9,10 are
   equivalent to 1,2,3,4,5, etc.  In the colored case, the sequence of
   colors also repeats.  But linemodes 1,2,3,4,5 are drawn solid, while
   6,7,8,9,10 are drawn dotted, 11,12,13,14,15 are drawn dotdashed, etc.
   So there are 25 distinct colored linemodes, and 5 distinct monochrome
   (black) ones.

   The color of a symbol will be the same as the color of the polyline on
   which it is plotted.

   linemodes -1, -2, etc. have a special interpretation.  They are
   `disconnected' linemodes: no polyline will appear, but if color is
   being used, the color of the plotted symbols (if any) will be
   linemode-dependent.  -1,-2,-3,-4,5 signify red,green,blue,magenta,cyan
   (the same sequence as for 1,2,3,4,5); thereafter the sequence repeats.

   linemode 0 is special (for backward compatibility).  No line is drawn;
   symbol #1 (a point) will be used.  So using linemode 0 is the same as
   using linemode -1, symbol 1.

   The point plotter is invoked by calling the following, in order.

   initialize_plotter() initializes global structures used by 
   	the plot_frame() and plot_point() routines.  These include
	the structures that specify the linear transformation from user
	coordinates to the coordinates used by libplot, and structures
	that specify the style of the plot frame.  
   open_plotter() opens the output device, by calling libplot routines.
   plot_frame() plots the plot frame, using libplot routines.  [Optional.]
   plot_point() uses libplot routines to plot a single point, together 
   	with (possibly)	a line extending to it from the last point, and 
	a symbol. [Alternatively, plot_point_array() can be used, to plot 
	an array of points.]
   close_plotter() closes the plotter. */

#include "sys-defines.h"
#include "plot.h"
#include "extern.h"

#define FUZZ 0.000001		/* bd. on floating pt. roundoff error */

#define NEAR_EQUALITY(a, b, scale) (fabs((a) - (b)) < (FUZZ * fabs(scale)))

typedef unsigned int outcode;	/* for Cohen-Sutherland clipper */
enum { TOP = 0x1, BOTTOM = 0x2, RIGHT = 0x4, LEFT = 0x8 };
enum { ACCEPTED = 0x1, CLIPPED_FIRST = 0x2, CLIPPED_SECOND = 0x4 };

#define TRIAL_NUMBER_OF_TICK_INTERVALS 5
#define MAX_NUM_SUBTICKS 29	/* max num. of linearly spaced subticks */
#define RELATIVE_SUBTICK_SIZE 0.4 /* subtick_size / tick_size */
/* if a log axis spans >5.0 orders of magnitude, don't plot log subsubticks */
#define MAX_DECADES_WITH_LOG_SUBSUBTICKS 5.0 

/* inter-tick spacing types, returned by scale1() and spacing_type() */
#define S_ONE 0
#define S_TWO 1
#define S_FIVE 2
#define S_TWO_FIVE 3	/* we don't use this one, but user may request it */
#define S_UNKNOWN -2

/* valid axis layout types; A_LOG2, anyone? */
#define A_LINEAR 0
#define A_LOG10 1

/* The x_trans and y_trans structures specify the linear transformation
from user coordinates to device coordinates.  They are used both in the
plotting of the graph frame, and in the plotting of data points. */

typedef struct
{
  /* Input (user) coordinates, all floating point.  These are the
     coordinates used in the original data points (or their base-10 logs,
     for an axis of log type).  We'll map them to the unit interval
     [0.0,1.0]. */
  double input_min, input_max;	/* min, max */
  double input_range;		/* max - min, precomputed for speed */
  /* If we're reversing axes, we'll then map [0.0,1.0] to [1.0,0.0] */
  bool reverse;
  /* We'll map [0.0,1.0] to another (smaller) interval, linearly */
  double squeezed_min, squeezed_max; /* min, max */
  double squeezed_range;	/* max - min */
  /* Output [i.e., libplot] coordinates.  The interval [0.0,1.0] will be
     mapped to this range, and the squeezed interval to a sub-range.  This
     is so that the box within which points are plotted will be smaller
     than the full area of the graphics display. */
  double output_min, output_max; /* min */
  double output_range;		/* max - min */
} Transform;

static Transform x_trans, y_trans; /* transformations applied to user coors
				      to turn them into device coordinates */

/* Affine transformation macros */

/* X Scale: convert from user x value to normalized x coordinate (floating
   point, 0.0 to 1.0). */
#define XS(x) (((x) - x_trans.input_min)/x_trans.input_range)
/* X Reflect: map [0,1] to [1,0], if that's called for */
#define XR(x) (x_trans.reverse ? 1.0 - (x) : (x))
/* X Squeeze: map [0,1] range for normalized x coordinate into a smaller
   interval, the x range for the plotting area within the graphics display */
#define XSQ(x) (x_trans.squeezed_min + (x) * x_trans.squeezed_range)
/* X Plot: convert from normalized x coordinate to floating point libplot
   coordinate. */
#define XP(x) (x_trans.output_min + (x) * x_trans.output_range)
/* X Value: convert from user x value (floating point) to floating point
   libplot coordinate. */
#define XV(x) XP(XSQ(XR(XS(x))))
/* X Normalize: the same, but do not perform reflection if any.  (We use
   this for plotting of axes and their labels.) */
#define XN(y) XP(XSQ(XS(y)))

/* Y Scale: convert from user y value to normalized y coordinate (floating
   point, 0.0 to 1.0). */
#define YS(y) (((y) - y_trans.input_min)/y_trans.input_range)
/* Y Reflect: map [0,1] to [1,0], if that's called for */
#define YR(y) (y_trans.reverse ? 1.0 - (y) : (y))
/* Y Squeeze: map [0,1] range for normalized y coordinate into a smaller
   interval, the y range for the plotting area within the graphics display */
#define YSQ(y) (y_trans.squeezed_min + (y) * y_trans.squeezed_range)
/* Y Plot: convert from normalized y coordinate to floating point libplot
   coordinate. */
#define YP(y) (y_trans.output_min + (y) * y_trans.output_range)
/* Y Value: convert from user y value (floating point) to floating point
   libplot coordinate.  (We use this for plotting of points.) */
#define YV(y) YP(YSQ(YR(YS(y))))
/* Y Normalize: the same, but do not perform reflection if any.  (We use
   this for plotting of axes and their labels.) */
#define YN(y) YP(YSQ(YS(y)))

/* Size Scale: convert distances, or sizes, from normalized coors to
   libplot coordinates.  (Used for tick, symbol, and font sizes.)  The min
   should really be precomputed. */
#define SS(x) \
(DMIN(x_trans.output_range * x_trans.squeezed_range, \
	     y_trans.output_range * y_trans.squeezed_range) * (x))

/* The `x_axis' and `y_axis' structs specify layout of the two axes, for
   the construction of the plot frame.  All structure elements that are
   doubles are expressed in user coordinates (unless the axis is
   logarithmic, in which case logs are taken before this structure is
   filled in). */

/* These structs are filled in by calls to prepare_axis(), when
   initialize_plotter() is called.  The only exceptions to this are the
   elements `max_width' and `non_user_ticks', which are filled in by
   plot_frame(), as the frame for the plot is being drawn. */

typedef struct 
{
  char *font_name;		/* fontname for axis label and tick labels */
  double font_size;		/* font size for axis label and tick labels */
  char *label;			/* axis label (a string) */
  int type;			/* axis layout type (A_LINEAR or A_LOG10) */
  double tick_spacing;		/* distance between ticks */
  int min_tick_count, max_tick_count; /* tick location = count * spacing */
  bool have_lin_subticks;	/* does axis have linearly spaced subticks? */
  double lin_subtick_spacing;	/* distance between linearly spaced subticks */
  int min_lin_subtick_count, max_lin_subtick_count; 
  bool have_normal_subsubticks; /* does axis have logarithmic subsubticks?*/
  bool user_specified_subsubticks; /* axis has user-spec'd subsubticks? */
  double subsubtick_spacing;	/* spacing for user-specified ones */
  double other_axis_loc;	/* location of intersection w/ other axis */
  double alt_other_axis_loc;	/* alternative loc. (e.g. right end vs. left)*/
  bool switch_axis_end;	/* other axis at right/top, not left/bottom? */
  bool omit_ticks;		/* just plain omit them (and their labels) ? */
  double max_label_width;	/* max width of labels placed on axis, in
				   libplot coors (we update this during
				   plotting, for y axis only) */
  int labelled_ticks;		/* number of labelled ticks, subticks, and
				   subsubticks drawn on the axis
				   (we update this during plotting, so we
				   can advise the user to specify a tick
				   spacing by hand if labelled_ticks <= 2) */
} Axis;

static Axis x_axis, y_axis;

/* This structure contains plotter attributes which are not axis-specific
   and hence are not part of the above structures.  The final elements are
   updated during plotter operation. */

typedef struct
{
  /* following elements are parameters (not updated during plotter operation)*/
  int handle;			/* Plotter handle from libplot */
  char *display_type;		/* mnemonic: type of libplot device driver */
  int rotation_angle;		/* one of ROT_{0,90,180,270} */
  bool save_screen;		/* erase display when opening plotter? */
  char *bg_color;		/* color of background, if non-NULL */
  grid_type grid_spec;		/* frame specification */
  double blankout_fraction;	/* 1.0 means blank whole box before plot */
  bool no_rotate_y_label;	/* useful for pre-X11R6 X servers */
  double tick_size;		/* fractional tick size */
  double subtick_size;		/* fractional subtick size (for linear axes) */
  double frame_line_width;	/* fractional width of lines in the frame */
  double half_line_width;	/* approx. half of this, in libplot coors */
  char *frame_color;		/* color for frame (and plot, if no -C) */
  char *title;			/* plot title */
  char *title_font_name;	/* font for plot title */
  double title_font_size;	/* fractional height of plot title */
  int clip_mode;		/* 0, 1, or 2 (cf. clipping in gnuplot) */
  /* following elements are updated during plotter operation; they're the
     repository for plotter state */
  bool first_point_of_plot;	/* true only at beginning of each plot */
  bool need_break;		/* break polyline before next point? */
  double oldpoint_x, oldpoint_y; /* last-plotted point */
  int symbol;			/* symbol being plotted at each point */
  int linemode;			/* linemode used for polyline */
} Plotter;
  
static Plotter plotter;

/* forward references */
static int clip_line ____P((double *x0_p, double *y0_p, double *x1_p, double *y1_p));
static int spacing_type ____P((double spacing));
static outcode compute_outcode ____P((double x, double y, bool tolerant));
static void plot_errorbar ____P((const Point *p));
static void prepare_axis ____P((Axis *axisp, Transform *trans, 
		       double min, double max, double spacing, 
		       char *font_name, double font_size, char *label, 
		       double subsubtick_spacing, 
		       bool user_specified_subsubticks, 
		       bool round_to_next_tick, bool log_axis,
		       bool reverse_axis, bool switch_axis_end, 
		       bool omit_ticks));
static void plot_abscissa_log_subsubtick ____P((double xval));
static void plot_ordinate_log_subsubtick ____P((double xval));
static void print_tick_label ____P((char *labelbuf, Axis axis, Transform transform, double val));
static void scale1 ____P((double min, double max, 
		 double *tick_spacing, int *tick_spacing_type));
static void set_line_style ____P((int style, bool use_color));
static void transpose_portmanteau ____P((int *val));


/* print_tick_label() prints a label on an axis tick.  The format depends
 * on whether the axis is a log axis or a linear axis; also, the magnitude
 * of the axis labels.
 */

static void
#ifdef _HAVE_PROTOS
print_tick_label (char *labelbuf, Axis axis, Transform transform, double val)
#else
print_tick_label (labelbuf, axis, transform, val)
     char *labelbuf;
     Axis axis;
     Transform transform;
     double val;
#endif
{
  int prec;
  char *eloc, *ptr;
  char labelbuf_tmp[64], incrbuf[64];
  double spacing;
  bool big_exponents;
  double min, max;

  /* two possibilities: large/small exponent magnitudes */

  min = (axis.type == A_LOG10 
	 ? pow (10.0, transform.input_min) : transform.input_min);
  max = (axis.type == A_LOG10 
	 ? pow (10.0, transform.input_max) : transform.input_max);
	  
  big_exponents = (((min != 0.0 && fabs (log10 (fabs (min))) >= 4.0)
		    || (max != 0.0 && fabs (log10 (fabs (max))) >= 4.0))
		   ? true : false);

  if (big_exponents)
    /* large exponents, rewrite as foo x 10^bar, using escape sequences */
    {
      char *src = labelbuf_tmp, *dst = labelbuf;
      int exponent;
      char floatbuf[64];
      char *fptr = floatbuf;
      double prefactor;
      
      sprintf (labelbuf_tmp, "%e", val);
      if ((eloc = strchr (labelbuf_tmp, (int)'e')) == NULL)
	return;

      if (axis.type == A_LOG10 && !axis.user_specified_subsubticks)
	/* a hack: this must be a power of 10, so just print "10^bar" */
	{
	  sscanf (++eloc, "%d", &exponent);	      
	  sprintf (dst, "10\\sp%d\\ep", exponent);
	  return;
	}

      /* special case: zero prints as `0', not 0.0x10^whatever */
      if (val == 0.0)
	{
	  *dst++ = '0';
	  *dst = '\0';
	  return;
	}

      while (src < eloc)
	*fptr++ = *src++;
      *fptr = '\0';
      sscanf (floatbuf, "%lf", &prefactor); /* get foo */
      sscanf (++src, "%d", &exponent); /* get bar */
      
      spacing = (axis.type == A_LINEAR
		 ? axis.tick_spacing
		 : axis.subsubtick_spacing); /* user-specified, for log axis */
      sprintf (incrbuf, "%f", 
	       spacing / pow (10.0, (double)exponent));
      ptr = strchr (incrbuf, (int)'.');
      prec = 0;
      if (ptr != NULL)
	{
	  int count = 0;
	  
	  while (*(++ptr))
	    {
	      count++;
	      if (*ptr != '0')
		prec = count;
	    }
	}
      
      /* \sp ... \ep is start_superscript ... end_superscript, and \r6 is
	 right-shift by 1/6 em.  \mu is the `times' character.  The \r8\l8
	 foolishness is to work around a bug in xfig 3.1. */
      sprintf (dst, "%.*f\\r6\\mu\\r8\\l810\\sp%d\\ep", 
	       prec, prefactor, exponent);

      return;
    }

  else	/* small-size exponent magnitudes */
    {
      if (axis.type == A_LOG10 && !axis.user_specified_subsubticks)
	/* a hack: this must be a (small) power of 10, so we'll just use
	   %g format (same as %f, no trailing zeroes) */
	{
	  sprintf (labelbuf, "%.9g", val);
	  return;
	}
      
      /* always use no. of digits of precision present in increment */
      spacing = (axis.type == A_LINEAR
		 ? axis.tick_spacing
		 : axis.subsubtick_spacing); /* user-specified, for log axis */
      sprintf (incrbuf, "%.9f", spacing);
      ptr = strchr (incrbuf, (int)'.');
      prec = 0;
      if (ptr != NULL)
	{
	  int count = 0;
	  
	  while (*(++ptr))
	    {
	      count++;
	      if (*ptr != '0')
		prec = count;
	    }
	}
      sprintf (labelbuf, "%.*f", prec, val);
      return;
    }
}

/* Algorithm SCALE1, for selecting an inter-tick spacing that will yield a
 * good-looking linear-format axis.  The spacing is always 1.0, 2.0, or 5.0
 * times a power of ten.
 *
 * Reference: Lewart, C. R., "Algorithms SCALE1, SCALE2, and
 *	SCALE3 for Determination of Scales on Computer Generated
 *	Plots", Communications of the ACM, 10 (1973), 639-640.  
 *      Also cited as ACM Algorithm 463.
 *
 * We call this routine even when the axis is logarithmic rather than
 * linear.  In that case the arguments are the logs of the actual
 * arguments, so that it computes an optimal inter-tick factor rather than
 * an optimal inter-tick spacing.
 */

static void
#ifdef _HAVE_PROTOS
scale1 (double min, double max, double *tick_spacing, int *tick_spacing_type)
#else
scale1 (min, max, tick_spacing, tick_spacing_type)
     double min;		/* Data min */
     double max;		/* Data max */
     double *tick_spacing;	/* Inter-tick spacing */
     int *tick_spacing_type;	/* Inter-tick spacing type (0, 1, or 2,
				   i.e. S_ONE, S_TWO, or S_FIVE) */
#endif
{
  int k;
  double nal;
  double a, b;
  
  /* valid interval lengths */
  static const double vint[] =
    {
      1.0, 2.0, 5.0, 10.0
    };
  
  /* Corresponding breakpoints.  Can choose geometric means, i.e. sqrt(2),
     sqrt(10), sqrt(50), but using sqrt(10)=3.16...  will (if nticks=5, as
     we choose it to be) cause intervals of length 1.5 to yield an
     inter-tick distance of 0.2 rather than 0.5.  So we could reduce it to
     2.95.  Similarly we could reduce sqrt(50) to 6.95 so that intervals of
     length 3.5 will yield an inter-tick distance of 1.0 rather than 0.5. */
  static const double sqr[] =
    {
      M_SQRT2, 3.16228, 7.07107
    };

  /* compute trial inter-tick interval length */
  a = (max - min) / TRIAL_NUMBER_OF_TICK_INTERVALS;
  a *= (max > min) ? 1.0 : -1.0; /* paranoia, max>min always */
  if (a <= 0.0)			
    {
      fprintf(stderr, "%s: error: bad trial inter-tick spacing %g\n",
	      progname, a);
      exit (EXIT_FAILURE);
    }
  nal = floor(log10(a));
  b = a * pow (10.0, -nal);	/* 1.0 <= b < 10.0 */

  /* round to closest permissible inter-tick interval length */
  k = 0;
  do
    {
      if (b < sqr[k])
	break;
      k++;
    }
  while (k < 3);

  *tick_spacing = (max > min ? 1.0 : -1.0) * vint[k] * pow (10.0, nal);
  /* for increment type, 0,1,2 means 1,2,5 times a power of 10 */
  *tick_spacing_type = (k == 3 ? 0 : k);
  return;
}

/* Determine whether an inter-tick spacing (in practice, one specified by
   the user) is 1.0, 2.0, or 5.0 times a power of 10. */
static int 
#ifdef _HAVE_PROTOS
spacing_type (double incr)
#else
spacing_type (incr)
     double incr;
#endif
{
  int i;
  int i_tenpower = (int)(floor(log10(incr)));
  double tenpower = 1.0;
  bool neg_power = false;

  if (i_tenpower < 0)
    {
      neg_power = true;
      i_tenpower = -i_tenpower;
    }

  for (i = 0; i < i_tenpower; i++)
    tenpower *= 10;
  if (neg_power)
    tenpower = 1.0 / tenpower;

  if (NEAR_EQUALITY(incr, tenpower, tenpower))
    return S_ONE;
  else if (NEAR_EQUALITY(incr, 2 * tenpower, tenpower))
    return S_TWO;
  else if (NEAR_EQUALITY(incr, 2.5 * tenpower, tenpower))
    return S_TWO_FIVE;
  else if (NEAR_EQUALITY(incr, 5 * tenpower, tenpower))
    return S_FIVE;
  else
    return S_UNKNOWN;
}


/* prepare_axis() fills in the Axis structure for an axis, and some of
 * the linear transformation variables in the Transform structure also.
 */

static void 
#ifdef _HAVE_PROTOS
prepare_axis (Axis *axisp, Transform *trans, double min, double max, 
	      double spacing, char *font_name, double font_size, char *label,
	      double subsubtick_spacing, 
	      bool user_specified_subsubticks, bool round_to_next_tick,
	      bool log_axis, bool reverse_axis, 
	      bool switch_axis_end, bool omit_ticks)
#else
prepare_axis (axisp, trans, 
	      min, max, spacing, 
	      font_name, font_size, label,
	      subsubtick_spacing, 
	      user_specified_subsubticks, 
	      round_to_next_tick, log_axis,
	      reverse_axis, switch_axis_end,
	      omit_ticks)
     Axis *axisp;
     Transform *trans;
     double min, max, spacing;
     char *font_name;
     double font_size;
     char *label;
     double subsubtick_spacing;
     bool user_specified_subsubticks; /* i.e. linear ticks on a log axis */
     bool round_to_next_tick; /* round limits to the next tick mark */
     bool log_axis;	/* log axis */
     bool reverse_axis;	/* will reverse min, max */
     bool switch_axis_end;	/* intersection w/ other axis in alt. pos. */
     bool omit_ticks;	/* suppress all ticks and tick labels */
#endif
{
  double range;
  int tick_spacing_type = 0;
  double tick_spacing, lin_subtick_spacing;
  int min_tick_count, max_tick_count;
  int min_lin_subtick_count, max_lin_subtick_count;
  bool have_lin_subticks;

  if (min > max)
    /* paranoia, max < min is swapped at top level */
    {
      fprintf(stderr, "%s: error: min > max for axis, not allowed\n",
	      progname);
      exit (EXIT_FAILURE);
    }

  if (min == max)		/* expand in a clever way */
    {
      max = floor (max + 1.0);
      min = ceil (min - 1.0);
    }

  if (log_axis)		/* log axis, data are stored in logarithmic form */
    /* compute a tick spacing; user can't specify it */
    {
      scale1 (min, max, &tick_spacing, &tick_spacing_type);
      if (tick_spacing <= 1.0)
	{
	  tick_spacing = 1.0;
	  tick_spacing_type = S_ONE;
	}
    }
  else				/* linear axis */
    {
      if (spacing == 0.0)	/* i.e., not specified by user */
	scale1 (min, max, &tick_spacing, &tick_spacing_type);
      else			/* luser is boss, don't use SCALE1 */
	{
	  tick_spacing = spacing;
	  tick_spacing_type = spacing_type (spacing);
	}
    }

  range = max - min;		/* range is not negative */

  if (round_to_next_tick)	/* expand both limits to next tick */
    {
      if (user_specified_subsubticks)
	/* Special Case.  If user specified the `spacing' argument to -x or
	   -y on a logarithmic axis, our usual tick-generating and
	   tick-plotting algorithms are disabled.  So we don't bother with
	   min_tick_count or several other fields of the axis struct;
	   instead we just compute a new (rounded) max, min, and range.
	   Since most data are stored as logs, this is complicated. */
	{
	  double true_min = pow (10.0, min), true_max = pow (10.0, max);
	  double true_range = true_max - true_min;
	  int min_count, max_count;
	  
	  min_count = (int)(floor ((true_min + FUZZ * true_range)
				   / subsubtick_spacing));
	  max_count = (int)(ceil ((true_max - FUZZ * true_range) 
				  / subsubtick_spacing));
	  /* avoid core dump, do *not* reduce minimum to zero! */
	  if (min_count > 0)
	    min = log10 (min_count * subsubtick_spacing);
	  max = log10 (max_count * subsubtick_spacing);	  
	  range = max - min;
	  min_tick_count = max_tick_count = 0; /* keep gcc happy */
	}
      else
	{
	  min_tick_count = (int)(floor((min + FUZZ * range)/ tick_spacing));
	  max_tick_count = (int)(ceil((max - FUZZ * range)/ tick_spacing));
	  /* max_tick_count > min_tick_count always */
	  /* tickval = tick_spacing * count, 
	     for all count in [min_count,max_count]; must have >=2 ticks */
	  min = tick_spacing * min_tick_count;
	  max = tick_spacing * max_tick_count;
	  range = max - min;
	}
    }
  else		/* don't expand limits to next tick */
    {
      min_tick_count = (int)(ceil((min - FUZZ * range)/ tick_spacing));
      max_tick_count = (int)(floor((max + FUZZ * range)/ tick_spacing)); 
      /* max_tick_count <= min_tick_count is possible */
      /* tickval = incr * count, 
	 for all count in [min_count,max_count]; can have 0,1,2,3... ticks */
    }
  
  /* Allow 5 subticks per tick if S_FIVE or S_TWO_FIVE, 2 if S_TWO.  Case
     S_ONE is special; we try 10, 5, and 2 in succession */
  switch (tick_spacing_type)
    {
    case S_FIVE:
    case S_TWO_FIVE:
      lin_subtick_spacing = tick_spacing / 5;      
      break;
    case S_TWO:
      lin_subtick_spacing = tick_spacing / 2;
      break;
    case S_ONE:
      lin_subtick_spacing = tick_spacing / 10;
      min_lin_subtick_count = (int)(ceil((min - FUZZ * range)/ lin_subtick_spacing));
      max_lin_subtick_count = (int)(floor((max + FUZZ * range)/ lin_subtick_spacing)); 
      if (max_lin_subtick_count - min_lin_subtick_count > MAX_NUM_SUBTICKS)
	{
	  lin_subtick_spacing = tick_spacing / 5;
	  min_lin_subtick_count = (int)(ceil((min - FUZZ * range)/ lin_subtick_spacing));
	  max_lin_subtick_count = (int)(floor((max + FUZZ * range)/ lin_subtick_spacing)); 
	  if (max_lin_subtick_count - min_lin_subtick_count > MAX_NUM_SUBTICKS)
	    lin_subtick_spacing = tick_spacing / 2;
	}
      break;
    default:	/* in default case, i.e. S_UNKNOWN, we won't plot linear subticks */
      lin_subtick_spacing = tick_spacing; /* not actually needed, since not plotted */
      break;
    }

  /* smallest possible inter-subtick factor for a log axis is 10.0 */
  if (log_axis && lin_subtick_spacing <= 1.0)
    lin_subtick_spacing = 1.0;

  min_lin_subtick_count = (int)(ceil((min - FUZZ * range)/ lin_subtick_spacing));
  max_lin_subtick_count = (int)(floor((max + FUZZ * range)/ lin_subtick_spacing)); 
  have_lin_subticks 
    = ((tick_spacing_type != S_UNKNOWN /* S_UNKNOWN -> no subticks */
	&& (max_lin_subtick_count - min_lin_subtick_count) <= MAX_NUM_SUBTICKS)
       ? true : false);

  /* fill in parameters for axis-specific affine transformation */
  trans->input_min = min;
  trans->input_max = max;
  trans->input_range = range;	/* precomputed for speed */
  trans->reverse = reverse_axis;

  /* fill in axis-specific plot frame variables */
  axisp->switch_axis_end = switch_axis_end;
  axisp->omit_ticks = omit_ticks;
  axisp->label = label;
  axisp->font_name = font_name;
  axisp->font_size = font_size;
  axisp->max_label_width = 0.0;
  axisp->type = log_axis ? A_LOG10 : A_LINEAR;
  axisp->tick_spacing = tick_spacing;
  axisp->min_tick_count = min_tick_count;
  axisp->max_tick_count = max_tick_count;
  axisp->have_lin_subticks = have_lin_subticks;
  axisp->lin_subtick_spacing = lin_subtick_spacing;
  axisp->min_lin_subtick_count = min_lin_subtick_count;
  axisp->max_lin_subtick_count = max_lin_subtick_count;
  axisp->user_specified_subsubticks = user_specified_subsubticks;
  axisp->subsubtick_spacing = subsubtick_spacing;
  axisp->labelled_ticks = 0;	/* updated during drawing of frame */

  if (log_axis)		/* logarithmic axis */
    /* do we have special logarithmic subsubticks, and should we label them? */
    {
      if (max - min <= 
	  MAX_DECADES_WITH_LOG_SUBSUBTICKS + FUZZ) 
	/* not too many orders of magnitude, so plot normal log subsubticks */
	axisp->have_normal_subsubticks = true;
      else
	/* too many orders of magnitude, don't plot log subsubticks */
	axisp->have_normal_subsubticks = false;
    }
  else				/* linear axes don't have log subsubticks */
    axisp->have_normal_subsubticks = false;
}

/* The following routines [initialize_plotter(), open_plotter(),
 * plot_frame(), plot_point(), close_plotter()] are the basic routines of
 * the point-plotter.  See descriptions at the head of this file.
 */

void 
#ifdef _HAVE_PROTOS
initialize_plotter(char *display_type, int rotation_angle, 
		   bool save_screen, char *bg_color,
		   double frame_line_width, char *frame_color, char *title, 
		   char *title_font_name, double title_font_size, 
		   double tick_size, grid_type grid_spec, double x_min, 
		   double x_max, double x_spacing, double y_min, double y_max,
		   double y_spacing, bool spec_x_spacing, 
		   bool spec_y_spacing, double width, double height, 
		   double up, double right, char *x_font_name, 
		   double x_font_size, char *x_label, char *y_font_name, 
		   double y_font_size, char *y_label, 
		   bool no_rotate_y_label, int log_axis,
		   int round_to_next_tick, int switch_axis_end, 
		   int omit_ticks, int clip_mode, double blankout_fraction,
		   bool transpose_axes)
#else
initialize_plotter(display_type, rotation_angle, save_screen, bg_color,
		   frame_line_width, frame_color,
		   title, title_font_name, title_font_size,
		   tick_size, grid_spec, 
		   x_min, x_max, x_spacing, y_min, y_max, y_spacing, 
		   spec_x_spacing, 
		   spec_y_spacing, 
		   width,
		   height, up, right, x_font_name, x_font_size, x_label,
		   y_font_name, y_font_size, y_label, no_rotate_y_label,
		   log_axis,
		   round_to_next_tick,
		   switch_axis_end, omit_ticks, 
		   clip_mode, blankout_fraction,
		   transpose_axes)
     char *display_type;	/* mnemonic: type of libplot display driver */
     int rotation_angle;	/* ROT_{0, 90, 180, 270} */
     bool save_screen;		/* whether or not to erase */
     char *bg_color;		/* color of background, if non-NULL */
     double frame_line_width;	/* fractional width of lines in the frame */
     char *frame_color;		/* color for frame (and plot if no -C option)*/
     char *title;		/* plot title */
     char *title_font_name;	/* font for plot title (string) */
     double title_font_size;	/* font size for plot title  */
     double tick_size;		/* fractional size of ticks */
     grid_type grid_spec;	/* gridstyle (and tickstyle) spec */
     double x_min, x_max, x_spacing;
     double y_min, y_max, y_spacing;
     bool spec_x_spacing;
     bool spec_y_spacing;
     double width, height, up, right;
     char *x_font_name;
     double x_font_size;
     char *x_label;
     char *y_font_name; 
     double y_font_size;
     char *y_label;
     bool no_rotate_y_label;
     /* portmanteaux */
     int log_axis;		/* whether axes should be logarithmic */
     int round_to_next_tick;	/* round limits to the next tick mark */
     int switch_axis_end;	/* put axes at right/top, not left/bottom? */
     int omit_ticks;		/* omit all ticks, tick labels from an axis? */
     /* other args */
     int clip_mode;		/* clip mode = 0, 1, or 2 */
     double blankout_fraction;	/* 1.0 means blank out whole box before plot*/
     bool transpose_axes;
#endif
{
  double x_subsubtick_spacing = 0.0, y_subsubtick_spacing = 0.0;
  /* local portmanteau variables */
  int reverse_axis = 0;		/* min > max on an axis? */
  int user_specified_subsubticks = 0; /* i.e. linear ticks on a log axis? */

  plotter.display_type = xstrdup (display_type);

  if (log_axis & X_AXIS)
    {
      if (spec_x_spacing)
	/* spacing is handled specially for log axes */
	{
	  spec_x_spacing = false;
	  user_specified_subsubticks |= X_AXIS;
	  x_subsubtick_spacing = x_spacing;
	}
    }
  
  if (log_axis & Y_AXIS)
    {
      if (spec_y_spacing)
	{
	  /* spacing is handled specially for log axes */
	  spec_y_spacing = false;
	  user_specified_subsubticks |= Y_AXIS;
	  y_subsubtick_spacing = y_spacing;
	}
    }
  
  /* check for reversed axes (min > max) */
  if (x_max < x_min)
    {
      reverse_axis |= X_AXIS;
      {
	double temp;
	
	temp = x_min;
	x_min = x_max;
	x_max = temp;
      }
    }
  if (x_max == x_min)
    {
      fprintf (stderr, 
	       "%s: separating identical upper and lower x limits\n",
	       progname);
      /* separate them */
      x_max += 1.0;
      x_min -= 1.0;
    }
  /* check for reversed axes (min > max) */
  if (y_max < y_min)
    {
      reverse_axis |= Y_AXIS;
      {
	double temp;
	
	temp = y_min;
	y_min = y_max;
	y_max = temp;
      }
    }
  if (y_max == y_min)
    {
      fprintf (stderr, 
	       "%s: separating identical upper and lower y limits\n",
	       progname);
      /* separate them */
      y_max += 1.0;
      y_min -= 1.0;
    }
  
  /* At this point, min < max for each axis, if the user specified the two
     limits on an axis; reverse_axis portmanteau variable keeps track of
     whether either axis was discovered to be reversed. */
  
  /* silently accept negative spacing as equivalent as positive */
  if (spec_x_spacing)
    {
      if (x_spacing == 0.0)
	{
	  fprintf (stderr, 
		   "%s: error: zero spacing between ticks on an axis\n",
		   progname);
	  exit (EXIT_FAILURE);
	}
      x_spacing = fabs (x_spacing);
    }
  if (spec_y_spacing)
    {
      if (y_spacing == 0.0)
	{
	  fprintf (stderr, 
		   "%s: error: zero spacing between ticks on an axis\n",
		   progname);
	  exit (EXIT_FAILURE);
	}
      y_spacing = fabs (y_spacing);
    }
	      
  /* now transpose the two axes (i.e. their portmanteau variables, labels,
     limits etc.) if transpose_axes was set */
  if (transpose_axes)
    {
      char *temp_string;
      double temp_double;
      
      transpose_portmanteau (&log_axis);
      transpose_portmanteau (&round_to_next_tick);
      transpose_portmanteau (&switch_axis_end);
      transpose_portmanteau (&omit_ticks);

      transpose_portmanteau (&reverse_axis);
      transpose_portmanteau (&user_specified_subsubticks);
      
      temp_string = x_label;
      x_label = y_label;
      y_label = temp_string;
      
      temp_double = x_min;
      x_min = y_min;
      y_min = temp_double;
      
      temp_double = x_max;
      x_max = y_max;
      y_max = temp_double;
      
      temp_double = x_spacing;
      x_spacing = y_spacing;
      y_spacing = temp_double;
      
      temp_double = x_subsubtick_spacing;
      x_subsubtick_spacing = y_subsubtick_spacing;
      y_subsubtick_spacing = temp_double;
    }
	      
  /* fill in the Plotter struct */

  plotter.save_screen = save_screen;
  plotter.bg_color = bg_color;
  plotter.frame_line_width = frame_line_width;
  plotter.frame_color = frame_color;
  plotter.no_rotate_y_label = no_rotate_y_label;
  plotter.blankout_fraction = blankout_fraction;

  if (title != NULL)
    plotter.title = xstrdup (title);
  else
    plotter.title = NULL;
  if (title_font_name != NULL)
      plotter.title_font_name = xstrdup (title_font_name);
  else
    plotter.title_font_name = NULL;
  plotter.title_font_size = title_font_size;
  plotter.tick_size = tick_size;
  plotter.subtick_size = RELATIVE_SUBTICK_SIZE * tick_size;
  plotter.grid_spec = grid_spec;
  plotter.clip_mode = clip_mode;
  plotter.rotation_angle = rotation_angle;

  /* fill in the Transform and Axis structures for each coordinate */
  prepare_axis(&x_axis, &x_trans,
	       x_min, x_max, x_spacing,
	       x_font_name, x_font_size, x_label, 
	       x_subsubtick_spacing,
	       (bool)(user_specified_subsubticks & X_AXIS), 
	       (bool)(round_to_next_tick & X_AXIS),
	       (bool)(log_axis & X_AXIS), 
	       (bool)(reverse_axis & X_AXIS),
	       (bool)(switch_axis_end & X_AXIS),
	       (bool)(omit_ticks & X_AXIS));
  prepare_axis(&y_axis, &y_trans,
	       y_min, y_max, y_spacing,
	       y_font_name, y_font_size, y_label, 
	       y_subsubtick_spacing,
	       (bool)(user_specified_subsubticks & Y_AXIS), 
	       (bool)(round_to_next_tick & Y_AXIS),
	       (bool)(log_axis & Y_AXIS), 
	       (bool)(reverse_axis & Y_AXIS),
	       (bool)(switch_axis_end & Y_AXIS),
	       (bool)(omit_ticks & Y_AXIS));
  
  /* fill in additional parameters in the two Transform structures */
  x_trans.squeezed_min = right;
  x_trans.squeezed_max = right + width;
  x_trans.squeezed_range = width;  
  y_trans.squeezed_min = up;
  y_trans.squeezed_max = up + height;
  y_trans.squeezed_range = height;

  /* specify interval range for each coordinate, in libplot units */
  x_trans.output_min = 0.0;
  x_trans.output_max = (double)PLOT_SIZE;
  x_trans.output_range = x_trans.output_max - x_trans.output_min;
  x_trans.output_min = 0.0;
  y_trans.output_max = (double)PLOT_SIZE;
  y_trans.output_range = y_trans.output_max - y_trans.output_min;

  /* fill in fields in Axis structs dealing with location of other axis */
  if (plotter.grid_spec != AXES_AT_ORIGIN)
    /* Normal case */
    {
      /* axes are at left/bottom */
      x_axis.other_axis_loc = x_trans.input_min;
      y_axis.other_axis_loc = y_trans.input_min;
      /* secondary axes (used only if --switch-axis-end is specified) */
      x_axis.alt_other_axis_loc = x_trans.input_max;
      y_axis.alt_other_axis_loc = y_trans.input_max;
    }
  else
    /* Special case: grid type #4, AXES_AT_ORIGIN */
    {
      /* In this case (grid type #4), we don't allow the user to move the
         axis position by using the --switch-axis-end option.  Each axis is
         at the value 0 (the origin) if the value 0 is between the limits
         of the opposing axis.  Otherwise, the position is at the end
         closer to the value of 0. */
      x_axis.other_axis_loc 
	= (x_trans.input_min * x_trans.input_max <= 0.0) ? 0.0 : 
	  (x_trans.input_min > 0.0 ? x_trans.input_min : x_trans.input_max);
      y_axis.other_axis_loc 
	= (y_trans.input_min * y_trans.input_max <= 0.0) ? 0.0 : 
	  (y_trans.input_min > 0.0 ? y_trans.input_min : y_trans.input_max);
      /* secondary axes are the same */
      x_axis.alt_other_axis_loc = x_axis.other_axis_loc;
      y_axis.alt_other_axis_loc = y_axis.other_axis_loc;
      x_axis.switch_axis_end = (((x_trans.input_max - x_axis.other_axis_loc)
				 < (x_axis.other_axis_loc - x_trans.input_min))
				? true : false);
      y_axis.switch_axis_end = (((y_trans.input_max - y_axis.other_axis_loc)
				 < (y_axis.other_axis_loc - y_trans.input_min))
				? true : false);
    }

  /* The following is a version of (plotter.frame_line_width)/2 (expressed
     in terms of libplot coordinates) which the plotter uses as an offset,
     to get highly accurate positioning of ticks and labels. */
  if (frame_line_width < 0.0 || pl_havecap ("WIDE_LINES") == 0)
    plotter.half_line_width = 0.0;/* N.B. <0.0 -> default width, pres. small */
  else
    plotter.half_line_width = 0.5 * frame_line_width * x_trans.output_range;

  /* initialize the plotter state variables */
  plotter.first_point_of_plot = true;
  plotter.oldpoint_x = 0.0; 
  plotter.oldpoint_y = 0.0;
}

int
#ifdef _HAVE_PROTOS
open_plotter(void)
#else
open_plotter()
#endif
{
  if (plotter.bg_color)
    /* select user-specified background color */
    pl_parampl ("BG_COLOR", plotter.bg_color);
  if ((plotter.handle = pl_newpl (plotter.display_type, NULL, stdout, stderr)) < 0)
    return -1;
  else
    pl_selectpl (plotter.handle);
  if (pl_openpl () < 0)
    return -1;
  if (!plotter.save_screen || plotter.bg_color)
    pl_erase ();
  switch ((int)(plotter.rotation_angle))
    {
    case (int)ROT_0:
    default:
      pl_fspace (0.0, 0.0, (double)PLOT_SIZE, (double)PLOT_SIZE);
      break;
    case (int)ROT_90:
      pl_fspace2 (0.0, (double)PLOT_SIZE,
		  0.0, 0.0,
		  (double)PLOT_SIZE, (double)PLOT_SIZE);
      break;
    case (int)ROT_180:
      pl_fspace2 ((double)PLOT_SIZE, (double)PLOT_SIZE,
		  0.0, (double)PLOT_SIZE,
		  (double)PLOT_SIZE, 0.0);
      break;
    case (int)ROT_270:
      pl_fspace2 ((double)PLOT_SIZE, 0.0,
		  (double)PLOT_SIZE, (double)PLOT_SIZE,
		  0.0, 0.0);
      break;
    }

  return 0;
}

int
#ifdef _HAVE_PROTOS
close_plotter(void)
#else
close_plotter()
#endif
{
  int retval;

  retval = pl_closepl ();
  if (retval < 0)
    return -1;
  else
    {
      pl_selectpl (0);
      return pl_deletepl (plotter.handle);
    }
}


/* plot_frame() plots the graph frame (grid plus axis labels and title),
   using the plotter's global variables (the x_axis, y_axis, x_trans,
   y_trans structures).  It comprises several almost-independent tasks:

   0. draw opaque white rectangle (``canvas'') for the plot
   1. draw title, if any, above drawing box
   2. draw axes, and a full drawing box if requested
   3. draw ticks, grid lines, and tick labels on abscissa;
      also subticks, if abscissa is a linear axis
   4. draw ticks, grid lines, and tick labels on ordinate,
      also subticks, if ordinate is a linear axis
   5. draw sub-tick marks, sub-grid lines, and sub-labels,
      if abscissa is a logarithmic axis
   6. draw sub-tick marks, sub-grid lines, and sub-labels,
      if ordinate is a logarithmic axis
   7. draw axis label on abscissa
   8. draw axis label on ordinate (normally rotated 90 degrees)

   A savestate()--restorestate() pair is wrapped around nine tasks.  They
   are not quite independent because in (4) and (6) we keep track of the
   maximum width of the tick labels on the ordinate, to position the
   rotated ordinate label properly in (8).

   At the conclusion of the eight tasks, we warn the user if (i) he/she
   didn't specify a tick spacing for a logarithmic axis by hand, and (ii)
   our default algorithm for drawing ticks on a logarithmic axis has drawn
   too few ticks (i.e. <= 2 ticks) on the axis.

   Task #0 (drawing a white canvas) isn't always performed; only if the
   argument draw_canvas is set.  It isn't set when we call this function
   for the first time; see graph.c.
*/

void
#ifdef _HAVE_PROTOS
plot_frame (bool draw_canvas)
#else
plot_frame (draw_canvas)
     bool draw_canvas;
#endif
{
  static bool tick_warning_printed = false; /*when too few labelled ticks*/

  pl_savestate();   /* wrap savestate()--restorestate() around all 9 tasks */

  /* set color for plot frame (and for plot also, if a monochrome one) */
  if (plotter.frame_color)
    pl_pencolorname (plotter.frame_color);

  /* set line width as a fraction of size of display, <0.0 means default */
  pl_flinewidth (plotter.frame_line_width * (double)PLOT_SIZE);
  pl_linemod ("solid");		/* axes and plotting box will be solid */

  /* 0.  DRAW AN OPAQUE WHITE BOX */

  if (draw_canvas)
    {
      pl_savestate();
      /* use user-specified background color (if any) instead of white */
      if (pl_havecap ("SETTABLE_BACKGROUND") != 0 && plotter.bg_color)
	pl_colorname (plotter.bg_color);
      else
	pl_colorname ("white");
      pl_filltype (1);		/* turn on filling */
      pl_fbox (XP(XSQ(0.5 - 0.5 * plotter.blankout_fraction)), 
	       YP(YSQ(0.5 - 0.5 * plotter.blankout_fraction)),
	       XP(XSQ(0.5 + 0.5 * plotter.blankout_fraction)),
	       YP(YSQ(0.5 + 0.5 * plotter.blankout_fraction)));
      pl_restorestate();
    }

  /* 1.  DRAW THE TITLE, I.E. THE TOP LABEL */

  if (plotter.grid_spec != NO_AXES 
      && !y_axis.switch_axis_end /* no title if x axis is at top of plot */
      && plotter.title != NULL && *plotter.title != '\0')
    {
      double title_font_size;

      /* switch to our font for drawing title */
      pl_fontname (plotter.title_font_name);
      title_font_size = pl_ffontsize (SS(plotter.title_font_size));

      pl_fmove (XP(XSQ(0.5)), 
		YP(YSQ(1.0 
		       + (((plotter.grid_spec == AXES_AND_BOX 
			    || plotter.grid_spec == AXES)
			   && (plotter.tick_size <= 0.0) ? 1.0 : 0.5)
			  * fabs(plotter.tick_size))))
		+ 0.65 * title_font_size
		+ plotter.half_line_width);
      pl_alabel ('c', 'b', plotter.title); /* title centered, bottom spec'd */
    }

  /* 2.  DRAW AXES FOR THE PLOT */

  switch (plotter.grid_spec)
    {
    case AXES_AND_BOX_AND_GRID:
    case AXES_AND_BOX:
      /* draw a box, not just a pair of axes */
      pl_fbox (XP(XSQ(0.0)), YP(YSQ(0.0)),
	       XP(XSQ(1.0)), YP(YSQ(1.0)));
      break;
    case AXES:
      {
	double xstart, ystart, xmid, ymid, xend, yend;
	
	xstart = (x_axis.switch_axis_end 
		  ? XN(x_axis.other_axis_loc) - plotter.half_line_width
		  : XN(x_axis.alt_other_axis_loc) + plotter.half_line_width);
	ystart = (y_axis.switch_axis_end 
		  ? YN(y_axis.alt_other_axis_loc)
		  : YN(y_axis.other_axis_loc));
	xmid = (x_axis.switch_axis_end 
		? XN(x_axis.alt_other_axis_loc)
		: XN(x_axis.other_axis_loc));
	ymid = ystart;
	xend = xmid;
	yend = (y_axis.switch_axis_end 
		? YN(y_axis.other_axis_loc) - plotter.half_line_width
		: YN(y_axis.alt_other_axis_loc) + plotter.half_line_width);
	
	pl_fmove (xstart, ystart);
	pl_fcont (xmid, ymid);
	pl_fcont (xend, yend);
      }
      break;
    case AXES_AT_ORIGIN:
      {
	double xpos, ypos;
	
	xpos = (x_axis.switch_axis_end 
		? XN(x_axis.other_axis_loc)
		: XN(x_axis.alt_other_axis_loc));
	ypos = (y_axis.switch_axis_end 
		? YN(y_axis.alt_other_axis_loc)
		: YN(y_axis.other_axis_loc));
	
	pl_fline (xpos, YP(YSQ(0.0)) - plotter.half_line_width,
		  xpos, YP(YSQ(1.0)) + plotter.half_line_width);
	pl_fline (XP(XSQ(0.0)) - plotter.half_line_width, ypos, 
		  XP(XSQ(1.0)) + plotter.half_line_width, ypos);	
      }
      break;
    case NO_AXES:
    default:
      break;
    }

  /* 3.  PLOT TICK MARKS, GRID LINES, AND TICK LABELS ON ABSCISSA */

  if (plotter.grid_spec != NO_AXES && !x_axis.omit_ticks
      && !x_axis.user_specified_subsubticks)
    {
      int i;
      double xval, xrange = x_trans.input_max - x_trans.input_min;
      /* there is no way you could use longer labels on tick marks! */
      char labelbuf[2048];

      /* switch to our font for drawing x axis label and tick labels */
      pl_fontname (x_axis.font_name);
      pl_ffontsize (SS(x_axis.font_size));
      
      for (i = x_axis.min_tick_count; i <= x_axis.max_tick_count; i++) 
	/* tick range can be empty */
	{
	  xval = i * x_axis.tick_spacing;
	  	  
	  /* discard tick locations outside plotting area */
	  if (xval < x_trans.input_min - FUZZ * xrange 
	      || xval > x_trans.input_max + FUZZ * xrange)
	    continue;

	  /* Plot the abscissa tick labels. */
	  if (!y_axis.switch_axis_end
	      && !(plotter.grid_spec == AXES_AT_ORIGIN
		   /* don't plot label if it could run into an axis */
		   && NEAR_EQUALITY (xval, x_axis.other_axis_loc, 
				     x_trans.input_range)
		   && (y_axis.other_axis_loc != y_trans.input_min)
		   && (y_axis.other_axis_loc != y_trans.input_max)))
	    /* print labels below bottom boundary */
	    {
	      pl_fmove (XV (xval),
			YN (y_axis.other_axis_loc)
			- (SS ((plotter.tick_size >= 0.0 ? 0.75 : 1.75) * fabs(plotter.tick_size))
			   + plotter.half_line_width));
	      print_tick_label (labelbuf, x_axis, x_trans,
				(x_axis.type == A_LOG10) ? pow (10.0, xval) : xval);
	      pl_alabel('c', 't', labelbuf);
	      x_axis.labelled_ticks++;
	    }
	  else
	    /* print labels above top boundary */
	    if (y_axis.switch_axis_end
	      && !(plotter.grid_spec == AXES_AT_ORIGIN
		   /* don't plot label if it could run into an axis */
		   && NEAR_EQUALITY (xval, x_axis.other_axis_loc, 
				     x_trans.input_range)
		   && (y_axis.other_axis_loc != y_trans.input_min)
		   && (y_axis.other_axis_loc != y_trans.input_max)))
	      {
		pl_fmove (XV (xval),
			  YN (y_axis.alt_other_axis_loc)
			  + (SS ((plotter.tick_size >= 0.0 ? 0.75 : 1.75) * fabs(plotter.tick_size))
			     + plotter.half_line_width));
		print_tick_label (labelbuf, x_axis, x_trans,
				  (x_axis.type == A_LOG10) ? pow (10.0, xval) : xval);
		pl_alabel('c', 'b', labelbuf);
		x_axis.labelled_ticks++;
	      }
	  
	  /* Plot the abscissa tick marks, and vertical grid lines. */
	  switch (plotter.grid_spec)
	    {
	    case AXES_AND_BOX_AND_GRID:
	      pl_linemod ("dotted");
	      pl_fmove (XV(xval), YP(YSQ(0.0)));
	      pl_fcont (XV(xval), YP(YSQ(1.0)));
	      pl_linemod ("solid");
	      /* fall through */
	    case AXES_AND_BOX:
	      if (!y_axis.switch_axis_end)
		{
		  pl_fmove (XV (xval), 
			 YN (y_axis.alt_other_axis_loc));
		  pl_fcont (XV (xval), 
			 YN (y_axis.alt_other_axis_loc)
			 - (SS (plotter.tick_size)
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)));
		}
	      else
		{
		  pl_fmove (XV (xval), 
			 YN (y_axis.other_axis_loc));
		  pl_fcont (XV (xval), 
			 YN (y_axis.other_axis_loc)
			 + (SS (plotter.tick_size)
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)));
		}
	      /* fall through */
	    case AXES:
	    case AXES_AT_ORIGIN:
	      if (!y_axis.switch_axis_end)
		{
		  pl_fmove (XV (xval), 
			 YN (y_axis.other_axis_loc));
		  pl_fcont (XV (xval), 
			 YN (y_axis.other_axis_loc)
			 + (SS (plotter.tick_size)
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)));
		}
	      else
		{
		  pl_fmove (XV (xval), 
			 YN (y_axis.alt_other_axis_loc));
		  pl_fcont (XV (xval), 
			 YN (y_axis.alt_other_axis_loc)
			 - (SS (plotter.tick_size)
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)));
		}
	      break;
	    default:		/* shouldn't happen */
	      break;
	    }
	}
      
      if (x_axis.have_lin_subticks)
	{
	  double subtick_size;	/* libplot coordinates */
	  
	  /* linearly spaced subticks on log axes are as long as reg. ticks */
	  subtick_size = (x_axis.type == A_LOG10 
			  ? SS(plotter.tick_size) : SS(plotter.subtick_size));

	  /* Plot the linearly spaced subtick marks on the abscissa */
	  for (i = x_axis.min_lin_subtick_count; i <= x_axis.max_lin_subtick_count; i++) 
	    /* tick range can be empty */
	    {
	      xval = i * x_axis.lin_subtick_spacing;
	  	  
	      /* discard subtick locations outside plotting area */
	      if (xval < x_trans.input_min - FUZZ * xrange 
		  || xval > x_trans.input_max + FUZZ * xrange)
		continue;

	      switch (plotter.grid_spec)
		{
		case AXES_AND_BOX_AND_GRID:
		case AXES_AND_BOX:
		  /* draw on both sides */
		  if (!y_axis.switch_axis_end)
		    {
		      pl_fmove (XV (xval), 
			     YN (y_axis.alt_other_axis_loc));
		      pl_fcont (XV (xval), 
			     YN (y_axis.alt_other_axis_loc)
			     - (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)));
		    }
		  else
		    {
		      pl_fmove (XV (xval), 
			     YN (y_axis.other_axis_loc));
		      pl_fcont (XV (xval), 
			     YN (y_axis.other_axis_loc)
			     + (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)));
		    }
		  /* fall through */
		case AXES:
		case AXES_AT_ORIGIN:
		  if (!y_axis.switch_axis_end)
		    /* draw on only one side */
		    {
		      pl_fmove (XV (xval), 
			     YN (y_axis.other_axis_loc));
		      pl_fcont (XV (xval), 
			     YN (y_axis.other_axis_loc)
			     + (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)));
		    }
		  else
		    {
		      pl_fmove (XV (xval), 
			     YN (y_axis.alt_other_axis_loc));
		      pl_fcont (XV (xval), 
			     YN (y_axis.alt_other_axis_loc)
			     - (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)));
		    }
		  break;
		default:		/* shouldn't happen */
		  break;
		}
	    }
	}

      /* plot a vertical dotted line at x = 0 */
      if (plotter.grid_spec != AXES_AT_ORIGIN 
	  && x_axis.type == A_LINEAR
	  && x_trans.input_min * x_trans.input_max < 0.0)
	{
	  pl_linemod ("dotted");
	  pl_fline (XV(0.0), YP(YSQ(0.0)),
		 XV(0.0), YP(YSQ(1.0)));
	  pl_linemod ("solid");	  
	}
    }
  
  /* 4.  PLOT TICK MARKS, GRID LINES, AND TICK LABELS ON ORDINATE */

  if (plotter.grid_spec != NO_AXES && !y_axis.omit_ticks
      && !y_axis.user_specified_subsubticks)
    {
      int i;
      double yval, yrange = y_trans.input_max - y_trans.input_min;
      /* there is no way you could use longer labels on tick marks! */
      char labelbuf[2048];

      /* switch to our font for drawing y axis label and tick labels */
      pl_fontname (y_axis.font_name);
      pl_ffontsize (SS(y_axis.font_size));
      
      for (i = y_axis.min_tick_count; i <= y_axis.max_tick_count; i++) 
	/* range can be empty */
	{
	  yval = i * y_axis.tick_spacing;
	  
	  /* discard tick locations outside plotting area */
	  if (yval < y_trans.input_min - FUZZ * yrange 
	      || yval > y_trans.input_max + FUZZ * yrange)
	    continue;

	  /* Plot the ordinate tick labels. */
	  if (!x_axis.switch_axis_end
	      && !(plotter.grid_spec == AXES_AT_ORIGIN
		   /* don't plot label if it could run into an axis */
		   && NEAR_EQUALITY (yval, y_axis.other_axis_loc, 
				     y_trans.input_range)
		   && (x_axis.other_axis_loc != x_trans.input_min)
		   && (x_axis.other_axis_loc != x_trans.input_max)))
	    /* print labels to left of left boundary */
	    {
	      double new_width;

	      pl_fmove (XN (x_axis.other_axis_loc)
		     - (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			   * fabs(plotter.tick_size))
			+ plotter.half_line_width),
		     YV (yval));
	      print_tick_label (labelbuf, y_axis, y_trans,
				(y_axis.type == A_LOG10) ? pow (10.0, yval) : yval);
	      new_width = pl_flabelwidth (labelbuf);
	      pl_alabel ('r', 'c', labelbuf);
	      y_axis.max_label_width = DMAX(y_axis.max_label_width, new_width);
	      y_axis.labelled_ticks++;
	    }
	  else
	    /* print labels to right of right boundary */
	    if (x_axis.switch_axis_end
		&& !(plotter.grid_spec == AXES_AT_ORIGIN
		     /* don't plot label if it could run into an axis */
		     && NEAR_EQUALITY (yval, y_axis.other_axis_loc, 
				       y_trans.input_range)
		     && (x_axis.other_axis_loc != x_trans.input_min)
		     && (x_axis.other_axis_loc != x_trans.input_max)))
	    {
	      double new_width;

	      pl_fmove (XN (x_axis.alt_other_axis_loc)
		    + (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			  * fabs(plotter.tick_size))
		       + plotter.half_line_width),
		     YV (yval));
	      print_tick_label (labelbuf, y_axis, y_trans,
				(y_axis.type == A_LOG10) ? pow (10.0, yval) : yval);
	      new_width = pl_flabelwidth (labelbuf);
	      pl_alabel ('l', 'c', labelbuf);
	      y_axis.max_label_width = DMAX(y_axis.max_label_width, new_width);
	      y_axis.labelled_ticks++;
	    }
	  
	  /* Plot the tick marks on the y-axis, and horizontal grid lines. */
	  switch (plotter.grid_spec)
	    {
	    case AXES_AND_BOX_AND_GRID:
	      pl_linemod ("dotted");
	      pl_fmove (XP(XSQ(0.0)), YV (yval));
	      pl_fcont (XP(XSQ(1.0)), YV (yval));
	      pl_linemod ("solid");
	      /* fall through */
	    case AXES_AND_BOX:
	      if (!x_axis.switch_axis_end)
		{
		  pl_fmove (XN (x_axis.alt_other_axis_loc),
			 YV (yval));
		  pl_fcont (XN (x_axis.alt_other_axis_loc)
			 - (SS (plotter.tick_size) 
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)),
			 YV (yval));
		}
	      else
		{
		  pl_fmove (XN (x_axis.other_axis_loc),
			 YV (yval));
		  pl_fcont (XN (x_axis.other_axis_loc)
			 + (SS (plotter.tick_size) 
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)),
			 YV (yval));
		}
	      /* fall through */
	    case AXES:
	    case AXES_AT_ORIGIN:
	      if (!x_axis.switch_axis_end)
		{
		  pl_fmove (XN (x_axis.other_axis_loc),
			 YV (yval));
		  pl_fcont (XN (x_axis.other_axis_loc)
			 + (SS (plotter.tick_size) 
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)),
			 YV (yval));
		}
	      else
		{
		  pl_fmove (XN (x_axis.alt_other_axis_loc),
			 YV (yval));
		  pl_fcont (XN (x_axis.alt_other_axis_loc)
			 - (SS (plotter.tick_size) 
			    + (plotter.tick_size > 0.0 ? plotter.half_line_width
			       : -plotter.half_line_width)),
			 YV (yval));
		}
	      break;
	    default:		/* shouldn't happen */
	      break;
	    }
	}

      if (y_axis.have_lin_subticks)
	{
	  double subtick_size;	/* libplot coordinates */

	  /* linearly spaced subticks on a log axis are as long as regular ticks */
	  subtick_size = (y_axis.type == A_LOG10 
			  ? SS(plotter.tick_size) : SS(plotter.subtick_size));

	  /* Plot the linearly spaced subtick marks on the ordinate */
	  for (i = y_axis.min_lin_subtick_count; i <= y_axis.max_lin_subtick_count; i++) 
	    /* range can be empty */
	    {
	      yval = i * y_axis.lin_subtick_spacing;
	      
	      /* discard subtick locations outside plotting area */
	      if (yval < y_trans.input_min - FUZZ * yrange 
		  || yval > y_trans.input_max + FUZZ * yrange)
		continue;

	      /* Plot the tick marks on the y-axis, and horizontal grid lines. */
	      switch (plotter.grid_spec)
		{
		case AXES_AND_BOX_AND_GRID:
		case AXES_AND_BOX:
		  if (!x_axis.switch_axis_end)
		    {
		      pl_fmove (XN (x_axis.alt_other_axis_loc),
			     YV (yval));
		      pl_fcont (XN (x_axis.alt_other_axis_loc)
			     - (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)),
			     YV (yval));
		    }
		  else
		    {
		      pl_fmove (XN (x_axis.other_axis_loc),
			     YV (yval));
		      pl_fcont (XN (x_axis.other_axis_loc)
			     + (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)),
			     YV (yval));
		    }
		  /* fall through */
		case AXES:
		case AXES_AT_ORIGIN:
		  if (!x_axis.switch_axis_end)
		    {
		      pl_fmove (XN (x_axis.other_axis_loc),
			     YV (yval));
		      pl_fcont (XN (x_axis.other_axis_loc)
			     + (subtick_size
				+ (subtick_size > 0.0 ? plotter.half_line_width
				   : -plotter.half_line_width)),
			     YV (yval));
		    }
		  else
		    {
		      pl_fmove (XN (x_axis.alt_other_axis_loc),
			     YV (yval));
		      pl_fcont (XN (x_axis.alt_other_axis_loc)
			     - (subtick_size
			       + (subtick_size > 0.0 ? plotter.half_line_width
				  : -plotter.half_line_width)),
			     YV (yval));
		    }
		  break;
		default:		/* shouldn't happen */
		  break;
		}
	    }
	}
	  
      /* plot a horizontal dotted line at y = 0 */
      if (plotter.grid_spec != AXES_AT_ORIGIN 
	  && y_axis.type == A_LINEAR
	  && y_trans.input_min * y_trans.input_max < 0.0)
	{
	  pl_linemod ("dotted");
	  pl_fline (XP(XSQ(0.0)), YV(0.0),
		 XP(XSQ(1.0)), YV(0.0));
	  pl_linemod ("solid");	  
	}
    }

  /* 5.  DRAW LOGARITHMIC SUBSUBTICKS AND THEIR LABELS ON ABSCISSA */

  /* first, draw normal logarithmic subsubticks if any */
  if (plotter.grid_spec != NO_AXES && x_axis.have_normal_subsubticks
      && !x_axis.user_specified_subsubticks && !x_axis.omit_ticks)
    {
      int i, m, imin, imax;
      double xval, xrange = x_trans.input_max - x_trans.input_min;

      /* compute an integer range (of powers of 10) large enough to include
	 the entire desired axis */
      imin = (int)(floor (x_trans.input_min - FUZZ * xrange));
      imax = (int)(ceil (x_trans.input_max + FUZZ * xrange));

      for (i = imin; i < imax; i++)
	{
	  for (m = 1; m <= 9 ; m++)
	    {
	      xval = i + log10 ((double)m);

	      /* Plot subsubtick and label, if desired. */
	      /* N.B. if tick is outside axis range, nothing will be printed */
	      plot_abscissa_log_subsubtick (xval);
	    }
	}
    }

  /* second, draw user-specified logarithmic subsubticks instead, if any */
  if (plotter.grid_spec != NO_AXES && x_axis.user_specified_subsubticks
      && !x_axis.omit_ticks)
    {
      int i, imin, imax;
      double xval, xrange = x_trans.input_max - x_trans.input_min;
      
      /* compute an integer range large enough to include the entire
	 desired axis */
      imin = (int)(floor (pow (10.0, x_trans.input_min - FUZZ * xrange) 
			  / x_axis.subsubtick_spacing));
      imax = (int)(ceil (pow (10.0, x_trans.input_max + FUZZ * xrange) 
			 / x_axis.subsubtick_spacing));
      
      /* draw user-specified subsubticks */
      for (i = imin; i <= imax; i++)
	{
	  xval = log10 (i * x_axis.subsubtick_spacing);

	  /* Plot subsubtick and label, if desired. */
	  /* N.B. if tick is outside axis range, nothing will be printed */
	  plot_abscissa_log_subsubtick (xval);
	}
    }

  /* 6.  DRAW LOGARITHMIC SUBSUBTICKS AND THEIR LABELS ON ORDINATE */
  
  /* first, draw normal logarithmic subsubticks if any */
  if (plotter.grid_spec != NO_AXES && y_axis.have_normal_subsubticks
      && !y_axis.user_specified_subsubticks && !y_axis.omit_ticks)
    {
      int i, m, imin, imax;
      double yval, yrange = y_trans.input_max - y_trans.input_min;

      /* compute an integer range (of powers of 10) large enough to include
	 the entire desired axis */
      imin = (int)(floor (y_trans.input_min - FUZZ * yrange));
      imax = (int)(ceil (y_trans.input_max + FUZZ * yrange));

      /* draw normal subticks */
      for (i = imin; i < imax; i++)
	{
	  for (m = 1; m <= 9; m++)
	    {
	      yval = i + log10 ((double)m);

	      /* Plot subsubtick and label, if desired. */
	      /* N.B. if tick is outside axis range, nothing will be printed */
	      plot_ordinate_log_subsubtick (yval);
	    }
	}
    }

  /* second, draw user-specified logarithmic subsubticks instead, if any */
  if (plotter.grid_spec != NO_AXES && y_axis.user_specified_subsubticks
      && !y_axis.omit_ticks)
    {
      int i, imin, imax;
      double yval, yrange = y_trans.input_max - y_trans.input_min;
      
      /* compute an integer range large enough to include the entire
	 desired axis */
      imin = (int)(floor (pow (10.0, y_trans.input_min - FUZZ * yrange) 
			  / y_axis.subsubtick_spacing));
      imax = (int)(ceil (pow (10.0, y_trans.input_max + FUZZ * yrange) 
			 / y_axis.subsubtick_spacing));
      
      /* draw user-specified subsubticks */
      for (i = imin; i <= imax; i++)
	{
	  yval = log10 (i * y_axis.subsubtick_spacing);

	  /* Plot subsubtick and label, if desired. */
	  /* N.B. if tick is outside axis range, nothing will be printed */
	  plot_ordinate_log_subsubtick (yval);
	}
    }

  /* 7.  DRAW THE ABSCISSA LABEL */

      if ((plotter.grid_spec != NO_AXES)
      && x_axis.label != NULL && x_axis.label != '\0')
    {
      double x_axis_font_size;
      double xloc;

      /* switch to our font for drawing x axis label and tick labels */
      pl_fontname (x_axis.font_name);
      x_axis_font_size = pl_ffontsize (SS(x_axis.font_size));

      if (plotter.grid_spec != AXES_AT_ORIGIN)
	/* center the label on the axis */
	xloc = 0.5 * (x_trans.input_max + x_trans.input_min);
      else
	{
	  if ((y_axis.other_axis_loc == y_trans.input_min)
	      || (y_axis.other_axis_loc == y_trans.input_max))

	    xloc = 0.5 * (x_trans.input_max + x_trans.input_min);
	  else
	    /* center label in the larger of the two halves */
	    xloc = 
	      x_trans.input_max-x_axis.other_axis_loc >= x_axis.other_axis_loc-x_trans.input_min ?
		0.5 * (x_trans.input_max + x_axis.other_axis_loc) :
		  0.5 * (x_axis.other_axis_loc + x_trans.input_min);
	}
      
      if (!y_axis.switch_axis_end) /* axis on bottom, label below it */
	{
	  pl_fmove (XV (xloc), 
		 YN (y_axis.other_axis_loc)
		 - (SS ((plotter.tick_size >= 0.0 ? 0.875 : 2.125) 
			* fabs(plotter.tick_size))
		    + (6*x_axis_font_size)/5
		    + plotter.half_line_width));
	  pl_alabel ('c', 't', x_axis.label);
	}
      else			/* axis on top, label above it */
	{
	  pl_fmove (XV (xloc), 
		 YN (y_axis.alt_other_axis_loc)
		 + (SS ((plotter.tick_size >= 0.0 ? 0.875 : 2.125) 
			* fabs(plotter.tick_size))
		    + (6*x_axis_font_size)/5
		    + plotter.half_line_width));
	  pl_alabel ('c', 'b', x_axis.label);
	}
    }

  /* 8.  DRAW THE ORDINATE LABEL */

  if ((plotter.grid_spec != NO_AXES)
      && (y_axis.label != NULL && *(y_axis.label) != '\0'))
    {
      double y_axis_font_size;
      double yloc;

      /* switch to our font for drawing y axis label and tick labels */
      pl_fontname (y_axis.font_name);
      y_axis_font_size = pl_ffontsize (SS(y_axis.font_size));

      if (plotter.grid_spec != AXES_AT_ORIGIN)
	/* center the label on the axis */
	yloc = 0.5 * (y_trans.input_min + y_trans.input_max);
      else
	{
	  if ((x_axis.other_axis_loc == x_trans.input_min)
	      || (x_axis.other_axis_loc == x_trans.input_max))
	    yloc = 0.5 * (y_trans.input_min + y_trans.input_max);
	  else
	    /* center label in the larger of the two halves */
	    yloc = 
	      y_trans.input_max-y_axis.other_axis_loc >= y_axis.other_axis_loc-y_trans.input_min ?
		0.5 * (y_trans.input_max + y_axis.other_axis_loc) :
		  0.5 * (y_axis.other_axis_loc + y_trans.input_min);
	}
      
/* a relic of temps perdus */
#define libplot_has_font_metrics 1

      if (!x_axis.switch_axis_end)
	{
	  pl_fmove (XN (x_axis.other_axis_loc)
		 - (libplot_has_font_metrics ?
		    (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			* fabs(plotter.tick_size)) 
		     + 1.15 * y_axis.max_label_width
		     + 0.5 * y_axis_font_size
		     + plotter.half_line_width)
		    : (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			  * fabs(plotter.tick_size)) /* backup */
		       + 1.0 * y_axis_font_size
		       + plotter.half_line_width)),
		 YV(yloc));
	  
	  if (libplot_has_font_metrics
	      && !plotter.no_rotate_y_label) /* can rotate label */
	    {
	      pl_textangle (90);
	      pl_alabel ('c', 'x', y_axis.label);
	      pl_textangle (0);
	    }
	  else
	    /* non-rotated axis label, right justified */
	    pl_alabel ('r', 'c', y_axis.label);
	}
      else
	{
	  pl_fmove (XN (x_axis.alt_other_axis_loc)
		 + (libplot_has_font_metrics ?
		    (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			* fabs(plotter.tick_size)) 
		     + 1.15 * y_axis.max_label_width 
		     + 0.5 * y_axis_font_size
		     + plotter.half_line_width)
		    : (SS((plotter.tick_size >= 0.0 ? 0.75 : 1.75) 
			  * fabs(plotter.tick_size)) /* backup */
		       + 1.0 * y_axis_font_size
		       + plotter.half_line_width)), 
		 YV(yloc));
	  
	  if (libplot_has_font_metrics
	      && !plotter.no_rotate_y_label) /* can rotate label */
	    {
	      pl_textangle (90);
	      pl_alabel ('c', 't', y_axis.label);
	      pl_textangle (0);
	    }
	  else
	    /* non-rotated axis label, left justified */
	    pl_alabel ('l', 'c', y_axis.label);
	}
    }

  /* END OF TASKS */

  /* flush frame to device */
  pl_flushpl();

  pl_restorestate();

  if (plotter.grid_spec != NO_AXES)
    {
      if (!tick_warning_printed && 
	  ((!x_axis.omit_ticks && x_axis.labelled_ticks <= 2)
	   || (!y_axis.omit_ticks && y_axis.labelled_ticks <= 2)))
	{
	  fprintf (stderr,
		   "%s: too few labelled axis ticks, adjust tick spacing manually\n",
		   progname);
	  tick_warning_printed = true;
	}
    }
}
  


/* plot_abscissa_log_subsubtick() and plot_ordinate_log_subsubtick() are
   called to plot both normal log subticks and special (user-requested)
   ones */

static void
#ifdef _HAVE_PROTOS
plot_abscissa_log_subsubtick (double xval)
#else
plot_abscissa_log_subsubtick (xval)
	double xval;		/* log of location */
#endif
{
  double xrange = x_trans.input_max - x_trans.input_min;
  /* there is no way you could use longer labels on tick marks! */
  char labelbuf[2048];
  double tick_size = SS(plotter.tick_size); /* for positioning labels */
  double subsubtick_size = SS(plotter.subtick_size);
      
  /* switch to our font for drawing x axis label and tick labels */
  pl_fontname (x_axis.font_name);
  pl_ffontsize (SS(x_axis.font_size));
  
  /* discard subsubtick locations outside plotting area */
  if (xval < x_trans.input_min - FUZZ * xrange
      || xval > x_trans.input_max + FUZZ * xrange)
    return;
  
  /* label subsubtick if it seems appropriate */
  if (x_axis.user_specified_subsubticks)
    {
      print_tick_label (labelbuf, x_axis, x_trans, 
			pow (10.0, xval));
      if (!y_axis.switch_axis_end)
	{
	  pl_fmove (XV (xval),
		 YN (y_axis.other_axis_loc)
		 - ((tick_size >= 0 ? 0.75 : 1.75)
		    * fabs((double)tick_size)
		    + plotter.half_line_width));
	  pl_alabel('c', 't', labelbuf);
	  x_axis.labelled_ticks++;
	}
      else
	{
	  pl_fmove (XV (xval),
		 YN (y_axis.alt_other_axis_loc)
		 + ((tick_size >= 0 ? 0.75 : 1.75) 
		    * fabs((double)tick_size)
		    + plotter.half_line_width));
	  pl_alabel('c', 'b', labelbuf);
	  x_axis.labelled_ticks++;
	}
    }
  
  /* draw subsubtick */
  switch (plotter.grid_spec)
    {
    case AXES_AND_BOX_AND_GRID:
      pl_linemod ("dotted");
      pl_fmove (XV (xval), YP(YSQ(0.0)));
      pl_fcont (XV (xval), YP(YSQ(1.0)));
      pl_linemod ("solid");
      /* fall through */
    case AXES_AND_BOX:
      if (!y_axis.switch_axis_end)
	{
	  pl_fmove (XV (xval), 
		 YN (y_axis.alt_other_axis_loc));
	  pl_fcont (XV (xval),
		 YN (y_axis.alt_other_axis_loc)
		 - (subsubtick_size
		    + (subsubtick_size > 0.0
		       ? plotter.half_line_width
		       : -plotter.half_line_width)));
	}
      else
	{
	  pl_fmove (XV (xval), 
		 YN (y_axis.other_axis_loc));
	  pl_fcont (XV (xval),
		 YN (y_axis.other_axis_loc)
		 + (subsubtick_size
		    + (subsubtick_size > 0.0
		       ? plotter.half_line_width
		       : -plotter.half_line_width)));
	}
      /* fall through */
    case AXES:
    case AXES_AT_ORIGIN:
      if (!y_axis.switch_axis_end)
	{
	  pl_fmove (XV (xval), 
		 YN (y_axis.other_axis_loc));
	  pl_fcont (XV (xval), 
		 YN (y_axis.other_axis_loc)
		 + (subsubtick_size
		    + (subsubtick_size > 0.0 
		       ? plotter.half_line_width
		       : -plotter.half_line_width)));
	}
      else
	{
	  pl_fmove (XV (xval), 
		 YN (y_axis.alt_other_axis_loc));
	  pl_fcont (XV (xval), 
		 YN (y_axis.alt_other_axis_loc)
		 - (subsubtick_size
		    + (subsubtick_size > 0.0 
		       ? plotter.half_line_width
		       : -plotter.half_line_width)));
	}
      break;
    default:			/* shouldn't happen */
      break;
    }
}

static void
#ifdef _HAVE_PROTOS
plot_ordinate_log_subsubtick (double yval)
#else
plot_ordinate_log_subsubtick (yval)
     double yval;		/* log of location */
#endif
{
  double yrange = y_trans.input_max - y_trans.input_min;
  /* there is no way you could use longer labels on tick marks! */
  char labelbuf[2048];
  double tick_size = SS(plotter.tick_size); /* for positioning labels */
  double subsubtick_size = SS(plotter.subtick_size);
    
  /* switch to our font for drawing y axis label and tick labels */
  pl_fontname (y_axis.font_name);
  pl_ffontsize (SS(y_axis.font_size));
  
  /* discard subsubtick locations outside plotting area */
  if (yval < y_trans.input_min - FUZZ * yrange
      || yval > y_trans.input_max + FUZZ * yrange)
    return;
  
  /* label subsubtick if it seems appropriate */
  if (y_axis.user_specified_subsubticks)		 
    {
      double new_width;
      
      print_tick_label (labelbuf, y_axis, y_trans, 
			pow (10.0, yval));
      if (!x_axis.switch_axis_end)
	{
	  pl_fmove (XN(x_axis.other_axis_loc)
		 - ((tick_size >= 0 ? 0.75 : 1.75) 
		    * fabs((double)tick_size)
		    + plotter.half_line_width),
		 YV (yval));
	  new_width = pl_flabelwidth (labelbuf);
	  pl_alabel ('r', 'c', labelbuf);
	  y_axis.max_label_width = DMAX(y_axis.max_label_width, new_width);
	  y_axis.labelled_ticks++;
	}
      else
	{
	  pl_fmove (XN(x_axis.alt_other_axis_loc)
		 + ((tick_size >= 0 ? 0.75 : 1.75) 
		    * fabs((double)tick_size)
		    + plotter.half_line_width),
		YV (yval));
	  new_width = pl_flabelwidth (labelbuf);
	  pl_alabel ('l', 'c', labelbuf);
	  y_axis.max_label_width = DMAX(y_axis.max_label_width, new_width);
	  y_axis.labelled_ticks++;
	}
    }
  
  /* draw subsubtick */
  switch (plotter.grid_spec)
    {
    case AXES_AND_BOX_AND_GRID:
      pl_linemod ("dotted");
      pl_fmove (XP(XSQ(0.0)), YV (yval));
      pl_fcont (XP(XSQ(1.0)), YV (yval));
      pl_linemod ("solid");
      /* fall through */
    case AXES_AND_BOX:
      if (!x_axis.switch_axis_end)		      
	{
	  pl_fmove (XN (x_axis.alt_other_axis_loc),
		 YV (yval));
	  pl_fcont (XN (x_axis.alt_other_axis_loc)
		 - (subsubtick_size
		    + (subsubtick_size > 0.0 
		       ? plotter.half_line_width
		       : -plotter.half_line_width)),
		 YV (yval));
	}
      else
	{
	  pl_fmove (XN (x_axis.other_axis_loc),
		 YV (yval));
	  pl_fcont (XN (x_axis.other_axis_loc)
		 + (subsubtick_size
		    + (subsubtick_size > 0.0 
		       ? plotter.half_line_width
		       : -plotter.half_line_width)),
		 YV (yval));
	}
      /* fall through */
    case AXES:
    case AXES_AT_ORIGIN:
      if (!x_axis.switch_axis_end)
	{
	  pl_fmove (XN (x_axis.other_axis_loc),
		 YV (yval));
	  pl_fcont (XN (x_axis.other_axis_loc)
		 + (subsubtick_size
		    + (subsubtick_size > 0.0 
		       ? plotter.half_line_width
		       : -plotter.half_line_width)),
		 YV (yval));
	}
      else
	{
	  pl_fmove (XN (x_axis.alt_other_axis_loc),
		 YV (yval));
	  pl_fcont (XN (x_axis.alt_other_axis_loc)
		 - (subsubtick_size
		   + (plotter.tick_size > 0.0 
		      ? plotter.half_line_width
		      : -plotter.half_line_width)),
		 YV (yval));
	}
      break;
    default:	/* shouldn't happen */
      break;
    }
}


/* set_line_style() maps from pl_linemodes to physical pl_linemodes.  See
 * explanation at head of file. */

static void
#ifdef _HAVE_PROTOS
set_line_style (int style, bool use_color)
#else
set_line_style (style, use_color)
     int style;
     bool use_color;
#endif
{
  if (!use_color)		/* monochrome */
    {
      if (style > 0)
	/* don't issue pl_linemod if style<=0, since no polyline will be drawn */
	{
	  int i;

	  i = (style - 1) % NO_OF_LINEMODES;
	  pl_linemod (linemodes[i]);      
	}
      
      /* use same color as used for plot frame */
      pl_colorname (plotter.frame_color);
    }
  else				/* color */
    {
      int i, j;

      if (style > 0)		/* solid lines, various colors */
	{
	  i = ((style - 1) / NO_OF_LINEMODES) % NO_OF_LINEMODES;
	  j = (style - 1) % NO_OF_LINEMODES;
	  pl_linemod (linemodes[i]);            
	}

      else if (style == 0)	/* use first color, as if -m 1 was spec'd */
				/* (no line will be drawn) */
	j = 0;

      else			/* neg. pl_linemode (no line will be drawn) */
	j = (-style - 1) % (NO_OF_LINEMODES - 1);

      pl_colorname (colorstyle[j]);
    }
}


/* plot_point_array() calls plot_point() on each point in an array of
 * points.
 */

void
#ifdef _HAVE_PROTOS
plot_point_array (const Point *p, int length)
#else
plot_point_array (p, length)
     const Point *p;
     int length;
#endif
{
  int index;

  for (index = 0; index < length; index++)
    plot_point (&(p[index]));
}

/* plot_point() plots a single point, including the appropriate symbol and
 * errorbar(s) if any.  It may call either pl_fcont() or pl_fmove(), depending on
 * whether the pendown flag is set or not.  Gnuplot-style clipping (clip
 * mode = 0,1,2) is supported.  Of the plotter's global variables, it makes
 * heavy use of the x_trans and y_trans structures, which specify the
 * linear transformation from user coordinates to device coordinates.  It
 * also updates the plotter's internal state variables.
 */

void
#ifdef _HAVE_PROTOS
plot_point (const Point *point)
#else
plot_point (point)
     const Point *point;
#endif
{
  double local_x0, local_y0, local_x1, local_y1;
  int clipval;

  /* If new polyline is beginning, take its line style, color/monochrome
     attribute, and line width and fill fraction attributes from the first
     point of the polyline.  We assume all such attribute fields are the
     same for all points in the polyline (our point reader arranges this
     for us). */
  if (!(point->pendown) || plotter.first_point_of_plot)
    {
      int intfill;
      
      set_line_style (point->linemode, point->use_color);

      /* N.B. linewidth < 0.0 means use libplot default */
      pl_flinewidth (point->line_width * (double)PLOT_SIZE);
      
      if (point->fill_fraction < 0.0)
	intfill = 0;		/* transparent */
      else			/* guaranteed to be <= 1.0 */
	intfill = 1 + IROUND((1.0 - point->fill_fraction) * 0xfffe);
      pl_filltype (intfill);
    }

  /* have endpoints of new line segment */
  local_x0 = plotter.oldpoint_x;
  local_y0 = plotter.oldpoint_y;
  local_x1 = point->x;
  local_y1 = point->y;

  /* save current point for use as endpoint of next line segment */
  plotter.oldpoint_x = point->x;
  plotter.oldpoint_y = point->y;

  /* apply Cohen-Sutherland clipper to new line segment */
  clipval = clip_line (&local_x0, &local_y0, &local_x1, &local_y1);

  if (!(clipval & ACCEPTED))	/* rejected in toto */
    {
      pl_fmove (XV (point->x), YV (point->y)); /* move with pen up */      
      return;
    }

  /* not rejected, ideally move with pen down */
  if (point->pendown && (point->linemode > 0))
    {
      switch (plotter.clip_mode) /* do gnuplot style clipping (0, 1, or 2) */
	{
	case 0:
	  if ((clipval & CLIPPED_FIRST) || (clipval & CLIPPED_SECOND))
	    /* clipped on at least one end, so move with pen up */
	    pl_fmove (XV (point->x), YV (point->y));
	  else
	    /* line segment within box, so move with pen down */
	    {
	      if (!plotter.first_point_of_plot)
		pl_fcont (XV (point->x), YV (point->y));
	      else
		pl_fmove (XV (point->x), YV (point->y));
	    }
	  break;
	case 1:
	default:
	  if ((clipval & CLIPPED_FIRST) && (clipval & CLIPPED_SECOND))
	    /* both OOB, so move with pen up */
	    pl_fmove (XV (point->x), YV (point->y));
	  else			
	    /* at most one point is OOB */
	    {
	      if (clipval & CLIPPED_FIRST) /*current pt. OOB, new pt. not OOB*/
		{
		  if (!plotter.first_point_of_plot)
		    {
		      /* move to clipped current point, draw line segment */
		      pl_fmove (XV (local_x0), YV (local_y0));
		      pl_fcont (XV (point->x), YV (point->y));
		    }
		  else
		    pl_fmove (XV (point->x), YV (point->y));
		}
	      else		/* current point not OOB, new point OOB */
		{
		  if (!plotter.first_point_of_plot)
		    {
		      /* draw line segment to clipped new point */
		      pl_fcont (XV (local_x1), YV (local_y1));
		      /* N.B. lib's notion of position now differs from ours */
		    }
		  else
		    pl_fmove (XV (point->x), YV (point->y));
		}
	    }
	  break;
	case 2:
	  if ((clipval & CLIPPED_FIRST) || plotter.first_point_of_plot)
	    /* move to clipped current point if necc. */
	    pl_fmove (XV (local_x0), YV (local_y0));
	  
	  /* draw line segment to clipped new point */
	  pl_fcont (XV (local_x1), YV (local_y1));
	  
	  if (clipval & CLIPPED_SECOND)
	    /* new point OOB, so move to new point, breaking polyline */
	    pl_fmove (XV (point->x), YV (point->y)); 
	  break;
	}
    }
  else				/* move with pen up */
    pl_fmove (XV (point->x), YV (point->y)); 

  plotter.first_point_of_plot = false;
  
  /* if target point is OOB, return without plotting symbol or errorbar */
  if (clipval & CLIPPED_SECOND)
    return;

  /* plot symbol and errorbar, doing a pl_savestate()--pl_restorestate() to keep
     from breaking the polyline under construction (if any) */
  if (point->symbol >= 32)	/* yow, a character */
    {
      /* will do a font change, so save & restore state */
      pl_savestate();
      plot_errorbar (point);
      pl_fontname (point->symbol_font_name);
      pl_fmarker (XV(point->x), YV(point->y), 
	       point->symbol, SS(point->symbol_size));
      pl_restorestate();
    }

  else if (point->symbol > 0)	/* a marker symbol */
    {
      if (point->linemode > 0)
	/* drawing a line, so (to keep from breaking it) save & restore state*/
	{
	  pl_savestate();
	  plot_errorbar (point); /* may or may not have an errorbar */
	  pl_fmarker (XV(point->x), YV(point->y), 
		  point->symbol, SS(point->symbol_size));
	  pl_restorestate();
	}
      else
	/* not drawing a line, so just place the marker */
	{
	  plot_errorbar (point);
	  pl_fmarker (XV(point->x), YV(point->y), 
		   point->symbol, SS(point->symbol_size));
	}
    }
  
  else if (point->symbol == 0 && point->linemode == 0)
    /* backward compatibility: -m 0 (even with -S 0) plots a dot */
    {
      plot_errorbar (point);
      pl_fmarker (XV(point->x), YV(point->y), M_DOT, SS(point->symbol_size));
    }

  else				/* no symbol */
    {
      if (point->linemode > 0)
	/* drawing a line, so (to keep from breaking it) save & restore state*/
	{
	  pl_savestate ();
	  plot_errorbar (point);      
	  pl_restorestate ();
	}
      else
	/* no polyline, no symbol either; just draw errorbar */
	plot_errorbar(point);
    }

  return;
}


/* clip_line() takes two points, the endpoints of a line segment, and
 * destructively passes back two points: the endpoints of the line segment
 * clipped by Cohen-Sutherland to the rectangular plotting area.  Return
 * value contains bitfields ACCEPTED, CLIPPED_FIRST, and CLIPPED_SECOND.
 */

static int
#ifdef _HAVE_PROTOS
clip_line (double *x0_p, double *y0_p, double *x1_p, double *y1_p)
#else
clip_line (x0_p, y0_p, x1_p, y1_p)
     double *x0_p, *y0_p, *x1_p, *y1_p;
#endif
{
  double x0 = *x0_p;
  double y0 = *y0_p;
  double x1 = *x1_p;
  double y1 = *y1_p;
  outcode outcode0 = compute_outcode (x0, y0, true);
  outcode outcode1 = compute_outcode (x1, y1, true);  
  bool accepted;
  int clipval = 0;
  
  for ( ; ; )
    {
      if (!(outcode0 | outcode1)) /* accept */
	{
	  accepted = true;
	  break;
	}
      else if (outcode0 & outcode1) /* reject */
	{
	  accepted = false;
	  break;
	}
      else
	{
	  /* at least one endpoint is outside; choose one that is */
	  outcode outcode_out = (outcode0 ? outcode0 : outcode1);
	  double x, y;		/* intersection with clip edge */
	  
	  if (outcode_out & RIGHT)	  
	    {
	      x = x_trans.input_max;
	      y = y0 + (y1 - y0) * (x_trans.input_max - x0) / (x1 - x0);
	    }
	  else if (outcode_out & LEFT)
	    {
	      x = x_trans.input_min;
	      y = y0 + (y1 - y0) * (x_trans.input_min - x0) / (x1 - x0);
	    }
	  else if (outcode_out & TOP)
	    {
	      x = x0 + (x1 - x0) * (y_trans.input_max - y0) / (y1 - y0);
	      y = y_trans.input_max;
	    }
	  else
	    {
	      x = x0 + (x1 - x0) * (y_trans.input_min - y0) / (y1 - y0);
	      y = y_trans.input_min;
	    }
	  
	  if (outcode_out == outcode0)
	    {
	      x0 = x;
	      y0 = y;
	      outcode0 = compute_outcode (x0, y0, true);
	    }
	  else
	    {
	      x1 = x; 
	      y1 = y;
	      outcode1 = compute_outcode (x1, y1, true);
	    }
	}
    }

  if (accepted)
    {
      clipval |= ACCEPTED;
      if ((x0 != *x0_p) || (y0 != *y0_p))
	clipval |= CLIPPED_FIRST;
      if ((x1 != *x1_p) || (y1 != *y1_p))
	clipval |= CLIPPED_SECOND;
      *x0_p = x0;
      *y0_p = y0;
      *x1_p = x1;
      *y1_p = y1;
    }

  return clipval;
}

/* Compute usual Cohen-Sutherland outcode, containing bitfields LEFT,
   RIGHT, BOTTOM, TOP.  Nine possibilities: 
   {LEFT, interior, RIGHT} x {BOTTOM, interior, TOP}.
   The `tolerant' flag specifies how we handle points on the boundary. */
static outcode
#ifdef _HAVE_PROTOS
compute_outcode (double x, double y, bool tolerant)
#else
compute_outcode (x, y, tolerant)
     double x, y;
     bool tolerant;
#endif
{
  outcode code = 0;
  double xfuzz = FUZZ * x_trans.input_range;
  double yfuzz = FUZZ * y_trans.input_range;  
  int sign = (tolerant == true ? 1 : -1);
  
  if (x > x_trans.input_max + sign * xfuzz)
    code |= RIGHT;
  else if (x < x_trans.input_min - sign * xfuzz)
    code |= LEFT;
  if (y > y_trans.input_max + sign * yfuzz)
    code |= TOP;
  else if (y < y_trans.input_min - sign * yfuzz)
    code |= BOTTOM;
  
  return code;
}

static void
#ifdef _HAVE_PROTOS
transpose_portmanteau (int *val)
#else
transpose_portmanteau (val)
     int *val;
#endif
{
  bool xtrue, ytrue;
  int newval;
  
  xtrue = ((*val & X_AXIS) ? true : false);
  ytrue = ((*val & Y_AXIS) ? true : false);
  
  newval = (xtrue ? Y_AXIS : 0) | (ytrue ? X_AXIS : 0);
  *val = newval;
}

static void 
#ifdef _HAVE_PROTOS
plot_errorbar (const Point *p)
#else
plot_errorbar (p)
     const Point *p;
#endif
{
  if (p->have_x_errorbar || p->have_y_errorbar)
    /* save & restore state, since error bars are solid */
    {
      pl_savestate();
      pl_linemod ("solid");
	
      if (p->have_x_errorbar)
	{
	  pl_fline (XV(p->xmin), YV(p->y) - 0.5 * SS(p->symbol_size),
		 XV(p->xmin), YV(p->y) + 0.5 * SS(p->symbol_size));
	  pl_fline (XV(p->xmin), YV(p->y), XV(p->xmax), YV(p->y));
	  pl_fline (XV(p->xmax), YV(p->y) - 0.5 * SS(p->symbol_size),
		 XV(p->xmax), YV(p->y) + 0.5 * SS(p->symbol_size));
	}
      if (p->have_y_errorbar)
	{
	  pl_fline (XV(p->x) - 0.5 * SS(p->symbol_size), YV(p->ymin),
		 XV(p->x) + 0.5 * SS(p->symbol_size), YV(p->ymin));
	  pl_fline (XV(p->x), YV(p->ymin), XV(p->x), YV(p->ymax));
	  pl_fline (XV(p->x) - 0.5 * SS(p->symbol_size), YV(p->ymax),
		 XV(p->x) + 0.5 * SS(p->symbol_size), YV(p->ymax));
	}

      pl_restorestate();
    }
}
