/* I just got myself a brand new HP Desk Jet Plus printer to use with TeX. I and a number of other people asked the net for sources to one of the many DeskJet drivers which must be out there, but there was no response. I searched all the archives I could find, with similarly unsuccessful results. So I took the dvijet driver from the Beebe release and modified it for the desk jet plus. It might be called (as by some) a "vanilla" driver because it doesn't use downloadable fonts, but downloadable fonts requires an extra memory cartridge -- for that I would have just as soon bought the IIp. I did include some memory compaction, though, which reduces file size and transfer time by 50 - 70 % So it's not quite vanilla. The various compaction schemes can be switched off at compile time if the old Desk Jet doesn't support them (I have no idea) or if any of them prove buggy (possible). I include the modified source below, as I am guessing there are others out there on the net who would find it useful. Perhaps some kind soul will even archive it. It requires the rest of the Beebe driver code, which is commonly available -- eg, at ymir.claremont.edu & utah ... whatever. Hope this is helpful. Sorry if this isn't the place I'm supposed to post source, but it's the best way I know to reach people who might find this useful. Paul Kirkaas kirkaas@cs.ucla.edu ---------------------------------------------------------- /* -*-C-*- dvidjp.c */ /*-->dvidjp*/ /**********************************************************************/ /******************************* dvidjp *******************************/ /**********************************************************************/ /* Driver for HP DeskJet Plus printer -- modified from * Beebe Laser Jet driver * by Paul Kirkaas (kirkaas@cs.ucla.edu) 22 May 1990 * patch 1 -- bugs in YOFF & COMPACTION fixed 23 May 1990 * * Employs 3 types of data compaction for greater efficiency -- * up to 60 - 70% reduction in output file size. Each compaction * scheme can be switched on or off independently by defining any * or all of: * * XOFF -- X offset -- instead of a string of leading zeros, offset * by appropriate amount. Minimal savings when used with * COMPACTION mode set. * * YOFF -- Y offset -- compress multiple blank lines into a single * y skip command. * * COMPACTION -- Uses mode 2 compaction on Desk Jet Plus -- this is * the trickiest but most effective compression modification. * This is the likliest part to have a bug. * Lots of cleaning up can be done here. * * If you have an old Desk Jet (not a Plus) it may not have * all these abilities -- try undfining some of these switches. * * It should run straight with the Beebe driver set; you might * have to make some adjustments. * * Please let me know if you make any improvements or repairs. * Thanks * * Good Luck. */ #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 HPLASERJET #define HPLASERJET 1 /* conditional compilation flag */ #define HPDESKJET 1 /* conditional compilation flag */ /* Include the following line for Y offsets*/ #define YOFF 1 /* Include the following line for using temporary X offsets*/ #define XOFF 1 /* Include the following line for Mode 2 Compaction */ #define COMPACTION 1 /* #define COMPACTION 0 */ #define VERSION_NO "2.10" /* DVI driver version number */ #define DEVICE_ID (comp ? \ "Hewlett-Packard Desk Jet plus (from LaserJet)\n\ WITH Data Compaction for faster I/O" : \ "Hewlett-Packard Desk Jet plus (from LaserJet)\n\ with-OUT Data Compaction I/O" ) /* this string is printed at runtime */ #define OUTFILE_EXT (comp ? "djc" : "dj") #define DEFAULT_RESOLUTION 300 /* default dots/inch on HP Desk Jet */ #define BYTE_SIZE 8 /* output file byte size */ #undef STDRES #define STDRES 1 /* 0 for low-resolution devices */ #define XDPI 300 /* HP Laser Jet horizontal dots/inch */ #define XPSIZE 8 /* horizontal paper size in inches */ #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 300 /* HP Laser Jet vertical dots/inch */ #define YPSIZE 11 /* vertical paper size in inches */ #define YSIZE (YDPI*YPSIZE) /* number of vertical dots */ /* 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 int comp = COMPACTION; /* compaction flag -- reset in option.h */ #include "bitmap.h" #include "main.h" #undef STDMAG #undef RESOLUTION #define STDMAG 1500 #define RESOLUTION (((float)STDMAG)/5.0) /* dots per inch */ #include "abortrun.h" #include "actfact.h" #include "alldone.h" #include "chargf.h" #include "charpk.h" #include "charpxl.h" #include "clrbmap.h" #include "clrrow.h" #include "dbgopen.h" /*-->devinit*/ /**********************************************************************/ /****************************** devinit *******************************/ /**********************************************************************/ void devinit(argc,argv) /* initialize device */ int argc; char *argv[]; { (void)getbmap(); OUTS("\033E"); /* printer reset */ OUTS("\033&l0L"); /* Disable perforation skip */ /* OUTS("\033*rB"); /* End graphics */ OUTS("\033*t300R"); /* printer resolution */ if (comp) OUTS("\033*b2M"); /* Select Compacted Graphics Mode Two */ else OUTS("\033*b0M"); /* Select Full Graphics Mode */ OUTS("\033*r0A"); /* Start graphics, at leftmost position */ } /*-->devterm*/ /**********************************************************************/ /****************************** devterm *******************************/ /**********************************************************************/ void devterm() /* terminate device */ { OUTS("\033E"); /* printer reset*/ } #include "dvifile.h" #include "dviinit.h" #include "dviterm.h" #include "dispchar.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 "initglob.h" #include "inch.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(pbit) UNSIGN32 *pbit; /* pointer to raster line */ /************************************************************************* Use machine-specific coding here for efficiency. For TOPS-20, we encode 9 bytes from every pair of 36-bit words. For each raster line on the paper, the Laser Jet expects a binary 8-bit byte stream of the form <ESC>*bnnnWxxxxxxx ... xxxxxxx <--- nnn bytes ---> where each byte contains, in order from high to low bit, a left-to-right bit pattern. No end-of-line marker is required; the escape sequence automatically causes a new raster line to be started. *************************************************************************/ { register UNSIGN32 w_even,w_odd; register UNSIGN32 *p; register BYTE *pbuf; BYTE buf[1+(XSIZE+7)/8]; /* space for EOS + n 8-bit bytes */ register INT16 i,last_word; int ic,jc; /* just counters */ unsigned char * endb; /* pointer to end of buf */ unsigned char greystr[129]; /* Max length of 127 in compacted mode 2 */ unsigned char * greyp; /* Pointer to greystr[] */ unsigned char outstr[999]; /* Compacted mode 2 output string */ unsigned char * outp; /* Pointer to outstr[] */ unsigned char * endob; /* pointer to end of outstr */ unsigned char cmp; /* Comparison variable */ int outsz; /* Size of outstr -- since includes NULLs */ int count; int gcnt; /* Length of grey runs */ #if IBM_PC_MICROSOFT for (last_word = XBIT - 1; (last_word >= 1) && (*(UNSIGN32*)normaddr(pbit,last_word) == 0); --last_word) ; /* trim white space a word at a time */ #else p = pbit + XBIT - 1; /* point to last word on line */ for (last_word = XBIT - 1; (last_word >= 1) && (*p == 0); --last_word) --p; /* trim white space a word at a time */ #endif p = pbit; pbuf = &buf[0]; for (i = 0; i <= last_word; i += 2) /* loop over trimmed raster */ { w_even = (*p++); w_odd = (*p++); #if (HOST_WORD_SIZE == 36) *pbuf++ = (BYTE)( (w_even >> 28) & 0xff); *pbuf++ = (BYTE)( (w_even >> 20) & 0xff); *pbuf++ = (BYTE)( (w_even >> 12) & 0xff); *pbuf++ = (BYTE)( (w_even >> 4) & 0xff); *pbuf++ = (BYTE)( ((w_even << 4) | (w_odd >> 32)) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 24) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 16) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 8) & 0xff); *pbuf++ = (BYTE)( (w_odd ) & 0xff); #else /* HOST_WORD_SIZE == 32 */ /* encode 8 bytes at a time on 32-bit machines */ *pbuf++ = (BYTE)( (w_even >> 24) & 0xff); *pbuf++ = (BYTE)( (w_even >> 16) & 0xff); *pbuf++ = (BYTE)( (w_even >> 8) & 0xff); *pbuf++ = (BYTE)( (w_even ) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 24) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 16) & 0xff); *pbuf++ = (BYTE)( (w_odd >> 8) & 0xff); *pbuf++ = (BYTE)( (w_odd ) & 0xff); #endif } *pbuf = '\0'; /* trailing EOS marker */ last_word |= 1; /* make last_word ODD */ for (i = ((last_word+1)*HOST_WORD_SIZE)/8; (*(--pbuf) == '\0') && (i > 1); --i) ; /* trim trailing zero bytes, leaving at least one */ last_word = i; #ifdef XOFF /* Trim leading zero bytes & do appropriate Temporary X offset */ OUTS("\033*b"); /* Set up for raster transfer */ pbuf=buf; for (ic=0; !buf[ic] && ic<last_word ; ic++, pbuf++) ; if (ic>10) /* Do an X-shift */ { OUTF("%dx",8*ic); /* Shift by 8*ic # of blank pixels */ } else /* don't bother */ { pbuf = buf; ic = 0; } endb = &buf[last_word]; if (comp) { greyp = greystr; outp = outstr; outsz = gcnt = 0; endb = &buf[last_word]; while (pbuf <= endb) { cmp = *pbuf; for (count = 0; count<127 && pbuf<=endb && cmp==*pbuf; count ++, pbuf++) ; if (count==0){printf("DANGER DANGER ***COUNT IS ZERO***!!!\n");exit(0);} if (count <= 3) /* No run yet -- just put them in greystr */ { gcnt += count; for (jc = 0; jc < count ; jc ++) { *greyp = cmp; greyp++; } if (gcnt >= 124) /* Can't have more than 127, so ... */ { outstr[outsz] = gcnt-1; outsz++; for (jc = 0; jc < gcnt; jc ++) { outstr[outsz + jc] = greystr[jc]; } outsz += gcnt; gcnt = 0; greyp = greystr; continue; } } else /* We have a run of 4 or more */ { if (gcnt) /* Flush our accumilated grey */ { outstr[outsz] = gcnt-1; outsz++; for (jc = 0; jc < gcnt; jc ++) { outstr[outsz + jc] = greystr[jc]; } outsz += gcnt; gcnt = 0; greyp = greystr; } outstr[outsz] = (unsigned char) (1-count); /* Is this cast right? */ outsz ++; outstr[outsz] = cmp; outsz++; } } if (gcnt) /* Flush our accumilated grey */ { outstr[outsz] = gcnt-1; outsz++; for (jc = 0; jc < gcnt; jc ++) { outstr[outsz + jc] = greystr[jc]; } outsz += gcnt; gcnt = 0; greyp = greystr; } OUTF("%dW",outsz); /* How much is coming ... */ endob = outstr + outsz -1; /* End of the outstr */ for (outp = outstr; outp <= endob; outp++ ) OUTC(*outp); } else { OUTF("%dW",(int)last_word-ic); /* How much is coming ... */ /* cannot use fprintf with %s format because of NUL's in string, and it is slow anyway */ for (i = ic; i < last_word; ++pbuf,++i) OUTC(*pbuf); } #else OUTF("\033*b%dW",(int)last_word); pbuf = &buf[0]; /* cannot use fprintf with %s format because of NUL's in string, and it is slow anyway */ for (i = 0; i < last_word; ++pbuf,++i) OUTC(*pbuf); #endif XOFF } /*-->prtbmap*/ /**********************************************************************/ /****************************** prtbmap *******************************/ /**********************************************************************/ void prtbmap() { register UNSIGN32 *p; register INT16 j,k,ybottom,ytop; #ifdef YOFF int ycnt = 0; #endif YOFF /* OUTS("\033E"); /* printer reset */ /* OUTS("\033&l0L"); /* Disable perforation skip */ /* OUTS("\033*rB"); /* End graphics */ /* OUTS("\033*t300R"); /* printer resolution */ if (comp) OUTS("\033*b2M"); /* Select Compacted Graphics Mode Two */ else OUTS("\033*b0M"); /* Select Full Graphics Mode */ OUTS("\033*r0A"); /* Start graphics, at leftmost position */ if (DBGOPT(DBG_PAGE_DUMP)) { INT16 k1,k2,k3; for (k3 = 0; k3 < XBIT; (k3 += 7, ++p)) { /* print bitmap 7 words at a pass */ k1 = k3; k2 = MIN(XBIT,k1+7); (void)printf("prtbmap()...bitmap words %d..%d",k1,k2-1); NEWLINE(stdout); (void)printf(" "); for (k = k1; k < k2; ++k) (void)printf("%10d",k*HOST_WORD_SIZE); NEWLINE(stdout); for (j = YBIT-1; j >= 0; --j) { p = BITMAP(j,0); for (k = 0; k < XBIT; (++k,++p)) { if (*p) /* print non-blank raster line */ { p = BITMAP(j,k1); (void)printf("%5d:",j); for (k = k1; k < k2; (++k,++p)) (void)printf(" %09lx",*p); NEWLINE(stdout); break; /* exit loop over k */ } } } } } (void)clearerr(plotfp); #if ZAPTHISOUT k = -1; /* find top non-zero raster */ for (j = YBIT-1; (j > 0) && (k < 0); --j) /* loop over raster lines */ { p = BITMAP(j,XBIT-1); for (k = XBIT - 1; ((k >= 0) && (*p == 0)); --k) --p; /* trim white space */ } ytop = j; #else ytop = YBIT-1; #endif 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,j-1); #if ZAPTHISOUT for (j = ytop; (j >= ybottom); --j) { OUTF("%5d:",(int)j); for (k = 0; k < XBIT; ++k) OUTF(" %9x",*BITMAP(j,k)); NEWLINE(plotfp); } #endif #ifdef YOFF ycnt = 0; for (j = ytop; (j >= ybottom) ; --j) /* loop over raster lines */ { if (isNul (BITMAP(j,0))) ycnt ++; else { if (ycnt) OUTF("\033*p+%dY",ycnt); outline(BITMAP(j,0)); ycnt = 0; } } #else for (j = ytop; (j >= ybottom) ; --j) /* loop over raster lines */ outline(BITMAP(j,0)); #endif YOFF OUTS("\033*rB\f"); /* end raster graphics, eject page */ (void)fflush(plotfp); if (DISKFULL(plotfp)) (void)fatal("Output error -- disk storage probably full"); } #ifdef YOFF isNul (pbit) UNSIGN32 *pbit; { UNSIGN32 * p; #if IBM_PC_MICROSOFT int last_word; for (last_word = XBIT - 1; (last_word >= 1) && (*(UNSIGN32*)normaddr(pbit,last_word) == 0); --last_word) ; /* trim white space a word at a time */ if (last_word <=1) return 1; return 0; #else p = pbit + XBIT - 1; /* point to last word on line */ while ( !*p && p>pbit ) p--; if (p==pbit && !*p) return 1; return 0; #endif IBM_PC_MICROSOFT } #endif YOFF #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"