/* -*-C-*- dvitos.c */
/*-->dvitos*/
/**********************************************************************/
/******************************* dvitos *******************************/
/**********************************************************************/
/* <BEEBE.TEX.DVI>DVITOS.C.37,  5-Jun-85 00:34:50, Edit by BEEBE */
/* (1) output FF at end-of-page before printer reset, and then output */
/* tab stop settings (because the printer has no default ones!) */
/* (2) make run-length encoding a run-time option '-r' */
/* (3) Preloading of fonts always enabled when selected page lists specified */
/* (4) Add code to remove adjacent bits on graphics lines */
/* <BEEBE.TEX.DVI>DVITOS.C.30,  4-Jun-85 17:17:21, Edit by BEEBE */
/* Adapted from DVIJET */

/***********************************************************************

                Performance remarks on the Toshiba P1351
                              [04-Jun-85]

Statistics were collected on  tests run from a  dedicated IBM PC with  a
9600-baud serial interface to the  printer, using a stand-alone  printer
driver program supporting XON/XOFF flow control.

------------------------------------------------------------------------
Test File		File Size	Time		Char/sec
			 (bytes)	(sec)		(limit is 960)
------------------------------------------------------------------------
TeX page (-m1000)	122983		485		253
run-length coded

TeX page (-m1000)	187362		212		883
full bitmap

TeX page (-m579)	 50476		152		327
run-length coded

TeX page (-m579)	 64933		 78		832
full bitmap

80 50-char lines of	  4174		 35		119
X's

C program		  3949		100		 39
(12 cpi draft mode)

C program		  3949		117		 33
(12 cpi Elite mode)

------------------------------------------------------------------------

The Toshiba P-1351 manual claims  a 12 cpi draft  mode speed of 192  cps
and a  12  cpi  Elite  speed  of 93  cps.   The  9600-baud  serial  port
limitation of 960 cps is far above these, but the figures above indicate
that the performance  on actual  manuscripts is  highly overrated.   The
printer driver program was  designed to echo a  character to the  screen
every time an XOFF was received from the printer, and from several dozen
to several hundred (for the run-length coded data) were received.   This
indicates that (1) the input buffer is probably a bottleneck, except for
full bitmap graphics,  and (2) horizontal  spacing and carriage  returns
(the C  programs)  drastically  reduce  the  throughput.   It  would  be
interesting to see how the printer performs with a parallel interface.

The TeX output quality  was marginal with full  bitmap graphics, due  to
bad alignment of  successive 24-bit-column  strips of  the bitmap.   The
output with run-length encoding was  terrible due to bad positioning  of
horizontal skips.   The  correctness  of  the  run-length  encoding  was
verified by an independent program which unpacks the output bitmap  into
another file which  compared identical  to the full  bitmap output.   In
addition, the bitmap itself was  verified by calling PRXBitMap()  inside
prtbmap() to format it for the Printronix dot matrix printer.

***********************************************************************/

#include "dvihead.h"

/**********************************************************************/
/************************  Device Definitions  ************************/
/**********************************************************************/

/* All output-device-specific definitions go here.  This section must
be changed when modifying a dvi driver for use on a new device */

#undef TOSHIBAP1351
#define  TOSHIBAP1351       1		/* conditional compilation flag */

#define VERSION_NO	"2.10"		/* DVI driver version number */

#define  DEVICE_ID	"Toshiba P-1351 180h x 180v dpi dot matrix printer"
				/* this string is printed at runtime */

#define OUTFILE_EXT	"tos"

#define  BYTE_SIZE        7		/* output file byte size */

#undef STDRES
#define STDRES  0			/* for low-resolution devices */

#define  XDPI		180		/* horizontal dots/inch */
#if    OS_PCDOS
#define  XPSIZE		7		/* horizontal paper size in inches */
#else /* NOT OS_PCDOS */
#define  XPSIZE		11		/* horizontal paper size in inches */
#endif /* OS_PCDOS */

#define  XSIZE		(((XDPI*XPSIZE+2*HOST_WORD_SIZE-1)/\
				(2*HOST_WORD_SIZE))*(2*HOST_WORD_SIZE))
					/* number of horizontal dots; */
					/* MUST BE multiple of */
					/* 2*HOST_WORD_SIZE */
#define  XWORDS		((XSIZE + HOST_WORD_SIZE - 1)/HOST_WORD_SIZE)
					/* number of words in rows  */
					/* of bitmap array */

#define  YDPI		180		/* vertical dots/inch */

#if    OS_PCDOS
#define  YPSIZE		9		/* vertical paper size in inches */
#else /* NOT OS_PCDOS */
#define  YPSIZE		11		/* vertical paper size in inches */
#endif /* OS_PCDOS */

#define  YSIZE		(YDPI*YPSIZE)	/* number of vertical dots */

#define  V24SIZE	(4*XWORDS*HOST_WORD_SIZE)
					/* 4 chars encode 24 bits in each  */
					/* bit position of each word */

/* The printer bit map (must have an even number of columns). */

#define XBIT ((1+2*XWORDS)/2)
#define YBIT YSIZE

#if    (IBM_PC_LATTICE | IBM_PC_MICROSOFT | IBM_PC_WIZARD)
#undef SEGMEM
#define SEGMEM 1 /* ( ((long)XBIT * (long)YBIT) > 65536L ) */
#endif

#include "bitmap.h"

#undef MAXOPEN				/* default too big with large bitmap */
#define  MAXOPEN	 3		/* reduce number of open PXL files */


#include "main.h"
#include "abortrun.h"
#include "actfact.h"
#include "alldone.h"

/*-->devinit*/
/**********************************************************************/
/****************************** devinit *******************************/
/**********************************************************************/

void
devinit(argc,argv)		/* initialize device */
int argc;
char *argv[];
{
    (void)getbmap();
    if (runlengthcode && !quiet)
    {
	(void)fprintf(stderr,"[Run-length encoding of output file]");
	NEWLINE(stderr);
    }
}

/*-->devterm*/
/**********************************************************************/
/****************************** devterm *******************************/
/**********************************************************************/

void
devterm()			/* terminate device */
{
}

#include "chargf.h"
#include "charpk.h"
#include "charpxl.h"
#include "clrbmap.h"
#include "clrrow.h"
#include "dbgopen.h"
#include "dispchar.h"
#include "dvifile.h"
#include "dviinit.h"
#include "dviterm.h"
#include "f20open.h"
#include "fatal.h"
#include "fillrect.h"
#include "findpost.h"
#include "fixpos.h"
#include "fontfile.h"
#include "fontsub.h"
#include "getbmap.h"
#include "getbytes.h"
#include "getfntdf.h"
#include "getpgtab.h"
#include "inch.h"
#include "initglob.h"
#include "loadchar.h"
#include "movedown.h"
#include "moveover.h"
#include "moveto.h"
#include "nosignex.h"
#include "openfont.h"
#include "option.h"

/*-->outline*/
/**********************************************************************/
/****************************** outline *******************************/
/**********************************************************************/

void
outline(pline)
char *pline;				/* pointer to raster line */

{
    register INT16 len;
    register char *a,*b,*c;
    INT16 k,rem;

    for ((len = V24SIZE, c = pline + V24SIZE - 1);
	(*c == '\100') && (len > 1);
	(--len,--c))			/* trim white space to 1 or more */
	;
    len = (len + 3) & ~03;		/* but round up to multiple of 4 */
    pline[len] = '\0';			/* store string terminator */

    /*******************************************************************
    The Toshiba P-1351 has the requirement in raster graphics mode  that
    no two horizontally adjacent dots can be on.  We must therefore scan
    the character quadruples in the line and turn off matching bits.
    *******************************************************************/
    for ((a = pline, k = 4); k < len; k += 4)
    {
        *(a+4) = (*(a+4) & ~(*a)) | '\100';
	++a;
        *(a+4) = (*(a+4) & ~(*a)) | '\100';
	++a;
        *(a+4) = (*(a+4) & ~(*a)) | '\100';
	++a;
        *(a+4) = (*(a+4) & ~(*a)) | '\100';
	++a;
    }

    /*******************************************************************
    We search for runs as follows.  "b" marks the beginning of unwritten
    data, and "a" anchors the beginning of a run which continues to just
    before "c".   The  run length  is  "c"-"a".  Since  the  run  length
    encoding carries a penalty of additional output equal to the  length
    of the  run length  prefix (<ESC>Hnnn)  plus a  new graphics  prefix
    (<ESC>;nnnn), short runs are not so encoded.

    If a long  enough run  is found,  then the  string "b"  .. "a"-1  is
    output since its length is now  known, followed by the run  encoding
    for the character at "a".  Then "a" and "b" are advanced to "c", and
    the scan continues.

    If the  run  beginning at  "a"  is too  short,  then "a"  is  simply
    advanced.  The scan terminates at the NUL end-of-string marker.

    ?????????????xxxxxxxxxxxxxxxx????????????????????'\0'
        ^	     ^		     ^			  ^
        |	     |		     |			  |
        b	     a		     c			 EOS

    	     <-----run------>

    The Toshiba P-1351 makes no provision for runs of black space,  only
    horizontal relative moves (in units if 1/180 inch) over white space,
    so we search only for quadruples (24-bit columns in 6 low-order bits
    of 4 characters) of the latter, which appear as strings "@@@@".
    *******************************************************************/

    if (runlengthcode)
    {
	if (len <= 12)			/* too short to run-length scan */
	    OUTF2("\033;%04d%s",len >> 2,pline); /* quadruple count ESC;nnnn */
	else
	{
	    for (a = b = c = pline; *a; )
	    {
		if (*a == '\100')		/* allowable run character */
		    for (c = a; (*a == *++c); )
			;			/* advance over run */
		if (((INT16)(c-a)) > 11)	/* long enough run */
		{
		    if (rem = ((INT16)(a - b)) & 03)
			    a += 4 - rem;	/* must keep (a-b) = 4n */
		    c -= ((INT16)(c - a)) % 12; /* and (c-a) = 12n */
		}
		if (((INT16)(c-a)) > 11)	/* output long run */
		{
		    if (a > b)			/* output previous string */
		    {
			OUTF("\033;%04d",((INT16)(a - b)) >> 2);
			for ( ; b < a; b++)
			    OUTC(*b);
		    }
		    len = (INT16)(c-a) / 4;

		    /* NB: manual says in one place the count is of
		    1/120 inch, and in another, 1/180 inch.  Experiment
		    shows it to be 1/180 inch. */

		    OUTC('\033');
		    OUTC('H');
		    OUTC(0100|(len >> 8));
		    OUTC(0100|((len >> 4) & 017));
		    OUTC(0100|(len & 017));
		    a = b = c;
		}
		else				/* ignore short run */
		    ++a;
	    }

	    if (rem = ((INT16)(a - b)) % 4)	/* should never be necessary */
		a += 4 - rem;			/* but must keep (a-b) = 4n */

	    if (a > b)				/* output remaining string */
	    {
		OUTF("\033;%04d",((INT16)(a - b)) >> 2);
		for ( ; b < a; b++)
		     OUTC(*b);
	    }
	}
    }
    else				/* no runlength coding */
    {
	OUTF("\033;%04d",len >> 2);	/* quadruple count ESC;nnnn */
	OUTS(pline);
    }
    OUTC('\n');			/* end-of-graphics line */
}

/*-->prtbmap*/
/**********************************************************************/
/****************************** prtbmap *******************************/
/**********************************************************************/

void
prtbmap()
{
    register char *c;			/* pointer into v24[] */
    UNSIGN32 *p;			/* pointer into bitmap[][] */
    register UNSIGN32 mask;		/* mask for single bit selection */
    register INT16 k;			/* innermost loop index */
    INT16 i,j,ybottom,ytop;
    char v24[V24SIZE+1];		/* vertical 24-bit raster encodings */
					/* with space for string terminator */

#if    SEGMEM
    UNSIGN32* q[24];			/* pointers to 24 raster lines */
#define Q(n) q[n]			/* use precomputed pointers */
#else /* NOT SEGMEM */
#define Q(n) (p - (n)*XBIT)		/* compute pointers on the fly */
#endif /* SEGMEM */


    (void)clearerr(plotfp);

    v24[V24SIZE] = '\0';		/* NUL for EOS marker */

    ytop = YBIT-1;

    k = -1;	    /* find bottom non-zero raster */
    for (j = 0; (j < ytop) && (k < 0); ++j) /* loop over raster lines */
    {

#if    IBM_PC_MICROSOFT
	for (k = XBIT - 1; ((k >= 0) && (*BITMAP(j,k) == 0)); --k)
	    ;			/* trim white space */
#else
	p = BITMAP(j,XBIT-1);
	for (k = XBIT - 1; ((k >= 0) && (*p == 0)); --k)
	    --p;		/* trim white space */
#endif
    }
    ybottom = MAX(0,24*((j-1)/24));	/* need 24n raster lines */

    OUTS("\033\032I\033G\033L07");
	/* ESC CTL-Z I to reinitialize printer */
	/* ESC L 0 7 to select correct interline spacing */
	/* ESC G for 180 dots/inch spacing and graphics mode */

    for (j = ytop; (j >= ybottom) ; j -= 24)	/* loop over raster lines */
						/* in groups of 24 */
    {
#if    SEGMEM
#else
	p = BITMAP(j,0);	/* the j-th raster line */
#endif
	c = &v24[0];		/* vertical 24-bit encodings */
	for (i = 0; i < XBIT; ++i)	/* loop over raster words */
	{
	    /* PCC-20 compiled (1 << (HOST_WORD_SIZE-1) to 0, so do it the
	       hard way, arghh... */
	    mask = 1;
	    mask <<= (HOST_WORD_SIZE-1);	/* to select leftmost bit */

#if    SEGMEM
	    for (k = 0; k < 24; ++k)	/* compute pointers to 24 rasters */
	        q[k] = BITMAP(j-k,i);
#endif

	    for (k = 0; k < HOST_WORD_SIZE; (++c,++k))/* loop over word bits */
	    {
		/* examine bits in 24 adjacent rows and build 4 6-bit */
		/* character values */

		*c = '\100';		/* initial character template */
		if (*Q(0) & mask)
 		    *c |= '\040';
		if (*Q(1) & mask)
		    *c |= '\020';
		if (*Q(2) & mask)
		    *c |= '\010';
		if (*Q(3) & mask)
		    *c |= '\004';
		if (*Q(4) & mask)
		    *c |= '\002';
		if (*Q(5) & mask)
		    *c |= '\001';


		*(++c) = '\100';	/* initial character template */
		if (*Q( 6) & mask)
 		    *c |= '\040';
		if (*Q( 7) & mask)
		    *c |= '\020';
		if (*Q( 8) & mask)
		    *c |= '\010';
		if (*Q( 9) & mask)
		    *c |= '\004';
		if (*Q(10) & mask)
		    *c |= '\002';
		if (*Q(11) & mask)
		    *c |= '\001';


		*(++c) = '\100';	/* initial character template */
		if (*Q(12) & mask)
 		    *c |= '\040';
		if (*Q(13) & mask)
		    *c |= '\020';
		if (*Q(14) & mask)
		    *c |= '\010';
		if (*Q(15) & mask)
		    *c |= '\004';
		if (*Q(16) & mask)
		    *c |= '\002';
		if (*Q(17) & mask)
		    *c |= '\001';


		*(++c) = '\100';	/* initial character template */
		if (*Q(18) & mask)
 		    *c |= '\040';
		if (*Q(19) & mask)
		    *c |= '\020';
		if (*Q(20) & mask)
		    *c |= '\010';
		if (*Q(21) & mask)
		    *c |= '\004';
		if (*Q(22) & mask)
		    *c |= '\002';
		if (*Q(23) & mask)
		    *c |= '\001';

		mask >>= 1;		/* move masking bit right 1 position */
	    }				/* end loop over k */
#if    SEGMEM
#else
	    ++p;			/* point to next word on raster */
#endif
	}				/* end loop over i */
	outline(&v24[0]);
    }					/* end loop over j */
    OUTS("\033\064");	/* ESC 4 to end graphics mode */
    OUTC('\f');		/* page eject */
    OUTS("\033\032I\n");/* reset printer to initial state */
    OUTS(		/* reset tab stops -- dumb printer has no defaults */
    "\033\062\033(09,17,25,33,41,49,57,65,73,81,89,97,A5,B3,C1,C9,D7,E5,F3.");

    (void)fflush(plotfp);
    if (DISKFULL(plotfp))
	(void)fatal("prtbmap(): Output error -- disk storage probably full");
}

#include "outrow.h"
#include "prtpage.h"
#include "readfont.h"
#include "readgf.h"
#include "readpk.h"
#include "readpost.h"
#include "readpxl.h"
#include "reldfont.h"
#include "rulepxl.h"
#include "setchar.h"
#include "setfntnm.h"
#include "setrule.h"
#include "signex.h"
#include "skgfspec.h"
#include "skipfont.h"
#include "skpkspec.h"
#include "special.h"
#include "strchr.h"
#include "strcm2.h"
#include "strid2.h"
#include "strrchr.h"
#include "tctos.h"
#include "usage.h"
#include "warning.h"