|
|
1.1 root 1: THIS VERSION IS NOT UP TO DATE;
2: INCLUDED FOR HISTORICAL REASONS.
3: /*
4: * drive C/A/T typesetter
5: * currently stdout only
6: */
7:
8: /*
9: output language from troff:
10: all numbers are character strings
11:
12: sn size in points
13: fn font as number from 1-n
14: cx ascii character x
15: Cxyz funny char xyz. terminated by white space
16: Hn go to absolute horizontal position n
17: Vn go to absolute vertical position n (down is positive)
18: hn go n units horizontally (relative)
19: vn ditto vertically
20: nnc move right nn, then print c (exactly 2 digits!)
21: (this wart is an optimization that shrinks output file size
22: about 35% and run-time about 15% while preserving ascii-ness)
23: Dt ...\n draw operation 't':
24: Dl x y line from here by x,y
25: Dc d circle of diameter d with left side here
26: De x y ellipse of axes x,y with left side here
27: Da x y r arc counter-clockwise by x,y of radius r
28: D~ x y x y ... wiggly line by x,y then x,y ...
29: nb a end of line (information only -- no action needed)
30: b = space before line, a = after
31: p new page begins -- set v to 0
32: #...\n comment
33: x ...\n device control functions:
34: x i init
35: x T s name of device is s
36: x r n h v resolution is n/inch
37: h = min horizontal motion, v = min vert
38: x p pause (can restart)
39: x s stop -- done for ever
40: x t generate trailer
41: x f n s font position n contains font s
42: x H n set character height to n
43: x S n set slant to N
44:
45: Subcommands like "i" are often spelled out like "init".
46: */
47:
48: #include <stdio.h>
49: #include <ctype.h>
50: #include <signal.h>
51:
52: #include "dev.h"
53: #define NFONT 8
54:
55: int output = 0; /* do we do output at all? */
56: int nolist = 0; /* output page list if > 0 */
57: int olist[20]; /* pairs of page numbers */
58: int spage = 9999; /* stop every spage pages */
59: int scount = 0;
60:
61: struct dev dev;
62: struct font *fontbase[NFONT+1];
63: int *pstab;
64: int nsizes;
65: int nfonts;
66: int smnt; /* index of first special font */
67: int nchtab;
68: char *chname;
69: int *chtab;
70: char *fitab[NFONT+1];
71: char *widthtab[NFONT+1]; /* widtab would be a better name */
72: char *codetab[NFONT+1]; /* device codes */
73:
74: #define FATAL 1
75: #define BMASK 0377
76: int dbg = 0;
77: int res; /* input assumed computed according to this resolution */
78:
79: #define HERE 0
80: #define THERE 1
81: #define HOST "r70\n"
82: char *typesetter[] = {
83: "/dev/cat", /* typesetter device */
84: "r70!/dev/202"
85: };
86: FILE *tf = NULL; /* output file */
87: char *fontdir = "/usr/bwk/troff/devcat";
88: extern char devname[];
89:
90: #define hmot(n) hpos += n
91: #define hgoto(n) hpos = n
92:
93: int size = -1; /* this is invalid */
94: int font = -1; /* current font */
95: int hpos; /* horizontal position where we are supposed to be next (left = 0) */
96: int oldh; /* previous H position */
97: int vpos; /* current vertical position (down positive) */
98: int oldv; /* current pos in 1/4 point units */
99: int horig; /* h origin of current block; hpos rel to this */
100: int vorig;
101: int DX = 1; /* step size in x for drawing */
102: int DY = 3; /* step size in y for drawing */
103: int drawdot = '.'; /* draw with this character */
104: int drawsize = 1; /* shrink by this factor when drawing */
105:
106: main(argc, argv)
107: char *argv[];
108: {
109: FILE *fp;
110: int i;
111: int busyflag = 0;
112: int waitflag = 0;
113: int where = HERE;
114: char wherebuf[100];
115: int done();
116:
117: while (argc > 1 && argv[1][0] == '-') {
118: switch (argv[1][1]) {
119: case 't':
120: tf = stdout;
121: break;
122: case 'o':
123: outlist(&argv[1][2]);
124: break;
125: case 'd':
126: dbg = atoi(&argv[1][2]);
127: if (dbg == 0) dbg = 1;
128: break;
129: case 'b':
130: busyflag = 1;
131: break;
132: case 'w':
133: waitflag = 1;
134: break;
135: case 's':
136: spage = atoi(&argv[1][2]);
137: if (spage <= 0)
138: spage = 9999;
139: break;
140: }
141: argc--;
142: argv++;
143: }
144: tryagain:
145: if ((fp = fopen("/etc/where", "r")) != NULL)
146: if(fgets(wherebuf, sizeof(wherebuf), fp) != NULL)
147: if(strcmp(wherebuf, HOST) != 0)
148: where = THERE;
149: fclose(fp);
150: if (tf != stdout)
151: tf = fopen(typesetter[where], "w");
152: if (busyflag) {
153: printf(tf==NULL? "Busy.\n": "Available.\n");
154: exit(0);
155: }
156: if (tf == NULL) {
157: if (waitflag == 0) {
158: error(!FATAL, "can't open typesetter\n");
159: exit(1);
160: } else {
161: sleep(60);
162: goto tryagain;
163: }
164: }
165:
166: if (signal(SIGINT, done) == SIG_IGN) {
167: signal(SIGINT, SIG_IGN);
168: signal(SIGQUIT, SIG_IGN);
169: signal(SIGHUP, SIG_IGN);
170: } else {
171: signal(SIGQUIT, done);
172: signal(SIGHUP, done);
173: }
174: signal(SIGTERM, done);
175:
176: if (argc <= 1)
177: conv(stdin);
178: else
179: while (--argc > 0) {
180: if (strcmp(*++argv, "-") == 0)
181: fp = stdin;
182: else if ((fp = fopen(*argv, "r")) == NULL)
183: error(FATAL, "can't open %s", *argv);
184: conv(fp);
185: fclose(fp);
186: }
187: account();
188: done();
189: }
190:
191: outlist(s) /* process list of page numbers to be printed */
192: char *s;
193: {
194: int n1, n2, i;
195:
196: nolist = 0;
197: while (*s) {
198: n1 = 0;
199: if (isdigit(*s))
200: do
201: n1 = 10 * n1 + *s++ - '0';
202: while (isdigit(*s));
203: else
204: n1 = -9999;
205: n2 = n1;
206: if (*s == '-') {
207: s++;
208: n2 = 0;
209: if (isdigit(*s))
210: do
211: n2 = 10 * n2 + *s++ - '0';
212: while (isdigit(*s));
213: else
214: n2 = 9999;
215: }
216: olist[nolist++] = n1;
217: olist[nolist++] = n2;
218: if (*s != '\0')
219: s++;
220: }
221: olist[nolist] = 0;
222: if (dbg)
223: for (i=0; i<nolist; i += 2)
224: printf("%3d %3d\n", olist[i], olist[i+1]);
225: }
226:
227: conv(fp)
228: register FILE *fp;
229: {
230: register int c, k;
231: int m, n, i, n1, m1;
232: char str[100], buf[300];
233:
234: while ((c = getc(fp)) != EOF) {
235: switch (c) {
236: case '\n': /* when input is text */
237: case ' ':
238: case 0: /* occasional noise creeps in */
239: break;
240: case '{': /* push down current environment */
241: t_push();
242: break;
243: case '}':
244: t_pop();
245: break;
246: case '0': case '1': case '2': case '3': case '4':
247: case '5': case '6': case '7': case '8': case '9':
248: /* two motion digits plus a character */
249: hmot((c-'0')*10 + getc(fp)-'0');
250: put1(getc(fp));
251: break;
252: case 'c': /* single ascii character */
253: put1(getc(fp));
254: break;
255: case 'C':
256: fscanf(fp, "%s", str);
257: put1s(str);
258: break;
259: case 't': /* straight text */
260: fgets(buf, sizeof(buf), fp);
261: t_text(buf);
262: break;
263: case 'D': /* draw function */
264: fgets(buf, sizeof(buf), fp);
265: switch (buf[0]) {
266: case 'l': /* draw a line */
267: sscanf(buf+1, "%d %d", &n, &m);
268: drawline(n, m, ".");
269: break;
270: case 'c': /* circle */
271: sscanf(buf+1, "%d", &n);
272: drawcirc(n);
273: break;
274: case 'e': /* ellipse */
275: sscanf(buf+1, "%d %d", &m, &n);
276: drawellip(m, n);
277: break;
278: case 'a': /* arc */
279: sscanf(buf+1, "%d %d %d %d", &n, &m, &n1, &m1);
280: drawarc(n, m, n1, m1);
281: break;
282: case '~': /* wiggly line */
283: drawwig(buf+1);
284: break;
285: default:
286: error(FATAL, "unknown drawing function %s\n", buf);
287: break;
288: }
289: break;
290: case 's':
291: fscanf(fp, "%d", &n); /* ignore fractional sizes */
292: setsize(t_size(n));
293: break;
294: case 'f':
295: fscanf(fp, "%s", str);
296: setfont(t_font(str));
297: break;
298: case 'H': /* absolute horizontal motion */
299: /* fscanf(fp, "%d", &n); */
300: while ((c = getc(fp)) == ' ')
301: ;
302: k = 0;
303: do {
304: k = 10 * k + c - '0';
305: } while (isdigit(c = getc(fp)));
306: ungetc(c, fp);
307: hgoto(k);
308: break;
309: case 'h': /* relative horizontal motion */
310: /* fscanf(fp, "%d", &n); */
311: while ((c = getc(fp)) == ' ')
312: ;
313: k = 0;
314: do {
315: k = 10 * k + c - '0';
316: } while (isdigit(c = getc(fp)));
317: ungetc(c, fp);
318: hmot(k);
319: break;
320: case 'w': /* word space */
321: break;
322: case 'V':
323: fscanf(fp, "%d", &n);
324: vgoto(n);
325: break;
326: case 'v':
327: fscanf(fp, "%d", &n);
328: vmot(n);
329: break;
330: case 'p': /* new page */
331: fscanf(fp, "%d", &n);
332: t_page(n);
333: break;
334: case 'n': /* end of line */
335: while (getc(fp) != '\n')
336: ;
337: t_newline();
338: break;
339: case '#': /* comment */
340: while (getc(fp) != '\n')
341: ;
342: break;
343: case 'x': /* device control */
344: devcntrl(fp);
345: break;
346: default:
347: error(!FATAL, "unknown input character %o %c\n", c, c);
348: done();
349: }
350: }
351: }
352:
353: devcntrl(fp) /* interpret device control functions */
354: FILE *fp;
355: {
356: char str[20], str1[50], buf[50];
357: int c, n;
358:
359: fscanf(fp, "%s", str);
360: switch (str[0]) { /* crude for now */
361: case 'i': /* initialize */
362: fileinit();
363: t_init(0);
364: break;
365: case 'T': /* device name */
366: fscanf(fp, "%s", devname);
367: break;
368: case 't': /* trailer */
369: t_trailer();
370: break;
371: case 'p': /* pause -- can restart */
372: t_reset('p');
373: break;
374: case 's': /* stop */
375: t_reset('s');
376: break;
377: case 'r': /* resolution assumed when prepared */
378: fscanf(fp, "%d", &res);
379: break;
380: case 'f': /* font used */
381: fscanf(fp, "%d %s", &n, str);
382: fgets(buf, sizeof buf, fp); /* in case there's a filename */
383: ungetc('\n', fp); /* fgets goes too far */
384: str1[0] = 0; /* in case there's nothing to come in */
385: sscanf(buf, "%s", str1);
386: loadfont(n, str, str1);
387: break;
388: /* these don't belong here... */
389: case 'H': /* char height */
390: fscanf(fp, "%d", &n);
391: t_charht(n);
392: break;
393: case 'S': /* slant */
394: fscanf(fp, "%d", &n);
395: t_slant(n);
396: break;
397: }
398: while ((c = getc(fp)) != '\n') /* skip rest of input line */
399: if (c == EOF)
400: break;
401: }
402:
403: fileinit() /* read in font and code files, etc. */
404: {
405: int i, fin, nw;
406: char *malloc(), *filebase, *p;
407: char temp[60];
408:
409: /* open table for device,
410: /* read in resolution, size info, font info, etc.
411: /* and set params
412: */
413: sprintf(temp, "%s/dev%s/desc.out", fontdir, devname);
414: if ((fin = open(temp, 0)) < 0) {
415: error(FATAL, "can't open tables for %s\n", temp);
416: }
417: read(fin, &dev, sizeof(struct dev));
418: nfonts = dev.nfonts;
419: nsizes = dev.nsizes;
420: nchtab = dev.nchtab;
421: filebase = malloc(dev.filesize); /* enough room for whole file */
422: read(fin, filebase, dev.filesize); /* all at once */
423: pstab = (int *) filebase;
424: chtab = pstab + nsizes + 1;
425: chname = (char *) (chtab + dev.nchtab);
426: p = chname + dev.lchname;
427: for (i = 1; i <= nfonts; i++) {
428: fontbase[i] = (struct font *) p;
429: nw = *p & BMASK; /* 1st thing is width count */
430: if (smnt == 0 && fontbase[i]->specfont == 1)
431: smnt = i; /* first special font */
432: p += sizeof(struct font); /* that's what's on the beginning */
433: widthtab[i] = p;
434: fitab[i] = p + nw;
435: p += nw + dev.nchtab + 128 - 32;
436: loadcode(i, nw, NULL);
437: t_fp(i, fontbase[i]->namefont, fontbase[i]->intname);
438: if(dbg > 1) fontprint(i);
439: }
440: fontbase[0] = (struct font *) malloc(255 + dev.nchtab + (128-32) + sizeof (struct font));
441: widthtab[0] = (char *) fontbase[0] + sizeof (struct font);
442: fontbase[0]->nwfont = 255;
443: close(fin);
444: }
445:
446: fontprint(i) /* debugging print of font i (0,...) */
447: {
448: int j, k, n;
449: char *p;
450:
451: printf("font %d:\n", i);
452: p = (char *) fontbase[i];
453: n = fontbase[i]->nwfont & BMASK;
454: printf("base=0%o, nchars=%d, spec=%d, name=%s, widtab=0%o, fitab=0%o\n",
455: p, n, fontbase[i]->specfont, fontbase[i]->namefont, widthtab[i], fitab[i]);
456: printf("widths:\n");
457: for (j=0; j <= n; j++) {
458: printf(" %2d", widthtab[i][j] & BMASK);
459: if (j % 20 == 19) printf("\n");
460: }
461: printf("\ncodetab:\n");
462: for (j=0; j <= n; j++) {
463: printf(" %2d", codetab[i][j] & BMASK);
464: if (j % 20 == 19) printf("\n");
465: }
466: printf("\nfitab:\n");
467: for (j=0; j <= dev.nchtab + 128-32; j++) {
468: printf(" %2d", fitab[i][j] & BMASK);
469: if (j % 20 == 19) printf("\n");
470: }
471: printf("\n");
472: }
473:
474: loadcode(n, nw, s1) /* load codetab on position n (0...); #chars is nw */
475: int n, nw;
476: char *s1;
477: {
478: int cin;
479: char temp[60];
480: char *malloc();
481:
482: if (codetab[n] != NULL)
483: free(codetab[n]);
484: codetab[n] = malloc(nw); /* read in device codes */
485: if (s1 == NULL || *s1 == 0)
486: sprintf(temp, "%s/dev%s/%s.code", fontdir, devname, fontbase[n]->namefont);
487: else
488: sprintf(temp, "%s/%s.code", s1, fontbase[n]->namefont);
489: if ((cin = open(temp, 0)) < 0)
490: error(FATAL, "can't open code tables %s", temp);
491: read(cin, codetab[n], nw);
492: close(cin);
493: }
494:
495: loadfont(n, s, s1) /* load font info for font s on position n (0...) */
496: int n;
497: char *s, *s1;
498: {
499: char temp[60];
500: int fin, nw, norig;
501:
502: if (n < 0 || n > NFONT)
503: error(FATAL, "illegal fp command %d %s", n, s);
504: if (strcmp(s, fontbase[n]->namefont) == 0)
505: return;
506: if (s1 == NULL || s1[0] == '\0')
507: sprintf(temp, "%s/dev%s/%s.out", fontdir, devname, s);
508: else
509: sprintf(temp, "%s/%s.out", s1, s);
510: if ((fin = open(temp, 0)) < 0)
511: error(FATAL, "can't open font table %s", temp);
512: norig = fontbase[n]->nwfont & BMASK;
513: read(fin, fontbase[n], norig + nchtab+128-32 + sizeof(struct font));
514: if ((fontbase[n]->nwfont & BMASK) > norig)
515: error(FATAL, "Font %s too big for position %d\n", s, n);
516: close(fin);
517: nw = fontbase[n]->nwfont & BMASK;
518: widthtab[n] = (char *) fontbase[n] + sizeof(struct font);
519: fitab[n] = (char *) widthtab[n] + nw;
520: loadcode(n, nw, s1);
521: t_fp(n, fontbase[n]->namefont, fontbase[n]->intname);
522: fontbase[n]->nwfont = norig; /* so can later use full original size */
523: if (dbg > 1) fontprint(n);
524: }
525:
526: done()
527: {
528: if (tf == NULL)
529: exit(1);
530: t_reset('s');
531: exit(0);
532: }
533:
534: error(f, s, a1, a2, a3, a4, a5, a6, a7) {
535: fprintf(stderr, "dcat: ");
536: fprintf(stderr, s, a1, a2, a3, a4, a5, a6, a7);
537: fprintf(stderr, "\n");
538: if (f)
539: done();
540: }
541:
542:
543: /*
544: Here beginneth all the stuff that really depends
545: on the Graphic Systems CAT.
546: */
547:
548:
549: char devname[20] = "cat";
550:
551: #define RES 432 /* CAT is 432 goobies per inch */
552: /* this is the way the cat actually works.
553: /* the input resolution might be different,
554: /* but this version of the program doesn't use it yet.
555: */
556: #define TRAILER (14 * res) /* trailer for last page */
557: #define ROUND3(n) (((n+1)/3)*3)
558: #define IMOT 16 /* mandatory initial motion for typesetter */
559: #define LMARGIN 0 /* left margin offset */
560: #define HMAX (46 * (res/6)) /* maximum horizontal size = 46 picas */
561: #define VMAX (15 * res) /* 15 inch page */
562:
563: #define DOUBLE 0200 /* double lens indicated by this */
564:
565: #define INIT 0100 /* initialize typesetter */
566: /* supposedly the same as */
567: /* DOWN, FWD, LRAIL, LMAG, LCASE */
568: #define STOP 0111 /* all done */
569: #define PAD 0101 /* lower rail padding */
570: #define FWD 0107 /* horizontal forward (right) */
571: #define BACK 0110 /* horizontal backward */
572: #define UP 0114 /* vertical up (back up page) */
573: #define DOWN 0112 /* down page */
574: #define HMOT 0200 /* horizontal motion bit */
575: #define VMOT 0140 /* vertical motion bits */
576: #define SIZE 0120 /* size change */
577:
578: /* font 1 = lower rail, lower mag */
579: /* font 2 = upper rail, lower mag */
580: /* font 3 = lower rail, upper mag */
581: /* font 4 = upper rail, upper mag */
582: /* god only knows how tilt factors into this */
583: #define URAIL 0102
584: #define LRAIL 0101
585: #define UMAG 0103
586: #define LMAG 0104
587: #define LTILT 0117 /* for CAT 8 */
588: #define UTILT 0116 /* copied from Scribe data base, not tested */
589: #define UCASE 0106 /* upper case mode */
590: #define LCASE 0105
591:
592: #define MAXFWD 127 /* max number of horizontal units */
593: #define MAXDOWN 31 /* max vertical units */
594:
595: long paper; /* paper used */
596:
597: static int hdir = FWD; /* current horizontal dir */
598: static int vdir = DOWN; /* current vertical dir */
599:
600: static int lastw = 0; /* last width, used by t_text */
601: static int rail = -1; /* LRAIL or URAIL */
602: static int mag = -1; /* LMAG or UMAG */
603: static int tilt = -1;
604: static int curcase = -1; /* either LCASE or UCASE */
605: static int hpos; /* current horizontal position (left = 0) */
606: static int vpos; /* current vertical position (down positive) */
607:
608: t_init(reinit) /* initialize device */
609: int reinit;
610: {
611: int i;
612:
613: hpos = vpos = oldv = oldh = 0;
614: drawdot = '.';
615: drawsize = 2; /* half size */
616: setsize(t_size(10)); /* start somewhere */
617: setfont(1);
618: if (!reinit) {
619: putc(INIT, tf);
620: hmot(IMOT);
621: vmot(3 * MAXDOWN);
622: hpos = vpos = 0;
623: movep(LMARGIN, 0);
624: } else {
625: putc(DOWN, tf);
626: putc(FWD, tf);
627: putc(LRAIL, tf);
628: putc(LMAG, tf);
629: putc(LCASE, tf);
630: }
631: }
632:
633: #define MAXSTATE 5
634:
635: struct state {
636: int ssize;
637: int sfont;
638: int shpos;
639: int svpos;
640: int shorig;
641: int svorig;
642: };
643: struct state state[MAXSTATE];
644: struct state *statep = state;
645:
646: t_push() /* begin a new block */
647: {
648: hflush();
649: statep->ssize = size;
650: statep->sfont = font;
651: statep->shorig = horig;
652: statep->svorig = vorig;
653: statep->shpos = hpos;
654: statep->svpos = vpos;
655: horig = hpos;
656: vorig = vpos;
657: hpos = vpos = 0;
658: if (statep++ >= state+MAXSTATE)
659: error(FATAL, "{ nested too deep");
660: hpos = vpos = 0;
661: }
662:
663: t_pop() /* pop to previous state */
664: {
665: if (--statep < state)
666: error(FATAL, "extra }");
667: size = statep->ssize;
668: font = statep->sfont;
669: hpos = statep->shpos;
670: vpos = statep->svpos;
671: horig = statep->shorig;
672: vorig = statep->svorig;
673: }
674:
675: t_page(n) /* do whatever new page functions */
676: {
677: int i;
678:
679: if (output) {
680: if (tf != stdout)
681: paper += vpos;
682: if (++scount >= spage) {
683: t_reset('p');
684: scount = 0;
685: }
686: }
687: vpos = 0;
688: oldv = 0;
689: output = 1;
690: /* somehow get to left margin? */
691: if (nolist == 0)
692: return; /* no -o specified */
693: output = 0;
694: for (i = 0; i < nolist; i += 2)
695: if (n >= olist[i] && n <= olist[i+1]) {
696: output = 1;
697: break;
698: }
699: }
700:
701: t_newline() /* do whatever for the end of a line */
702: {
703: hpos = 0; /* because we want to be back at the left margin */
704: }
705:
706: t_size(n) /* convert integer to internal size number*/
707: int n;
708: {
709: int i;
710:
711: if (n <= pstab[0])
712: return(1);
713: else if (n >= pstab[nsizes-1])
714: return(nsizes);
715: for (i = 0; n > pstab[i]; i++)
716: ;
717: return(i+1);
718: }
719:
720: t_charht(n) /* set character height to n */
721: int n;
722: {
723: }
724:
725: t_slant(n) /* set slant to n */
726: int n;
727: {
728: }
729:
730: t_font(s) /* convert string to internal font number */
731: char *s;
732: {
733: int n;
734:
735: n = atoi(s);
736: if (n < 0 || n > nfonts)
737: n = 1;
738: return(n);
739: }
740:
741: t_text(s) /* print string s as text */
742: char *s;
743: {
744: int c, w;
745: char str[100];
746:
747: if (!output)
748: return;
749: while (c = *s++) {
750: if (c == '\\') {
751: switch (c = *s++) {
752: case '\\':
753: case 'e':
754: put1('\\');
755: break;
756: case '(':
757: str[0] = *s++;
758: str[1] = *s++;
759: str[2] = '\0';
760: put1s(str);
761: break;
762: }
763: } else {
764: put1(c);
765: }
766: hmot(lastw);
767: if (dbg) printf("width = %d\n", lastw);
768: }
769: }
770:
771: t_reset(c)
772: {
773: int n;
774:
775: if (output)
776: paper += vpos;
777: output = 1; /* by God */
778: putc(INIT, tf);
779: putc(STOP, tf);
780: for (n = 0; n < 8; n++) /* flush out CAT internal buffer */
781: putc(PAD, tf);
782: fflush(tf);
783: }
784:
785: char *tracct[] = { /* accounting file */
786: "/usr/adm/tracct",
787: "r70!/usr/adm/tracct"
788: };
789:
790: account() /* record paper use */
791: {
792: FILE *f = NULL;
793: char *name, *getlogin();
794: int i;
795:
796: return; /* for now */
797: if (tf == stdout)
798: return;
799: for(i=0; i<sizeof(tracct)/sizeof(*tracct) && f== NULL; i++)
800: f = fopen(tracct[i], "a");
801: if(f != NULL) {
802: name = getlogin();
803: if (name == NULL)
804: name = "???";
805: fprintf(f, "%4d %s\n", (int)(paper/res + 20+6)/12, name);
806: }
807: }
808:
809: t_trailer()
810: {
811: return;
812: vpos = oldv = 0;
813: vgoto(TRAILER);
814: vpos = oldv = 0;
815: }
816:
817: movep(x, y) /* go to internal position x, y */
818: int x, y;
819: {
820: hmot(x - hpos);
821: vmot(ROUND3(y) - vpos);
822: }
823:
824: hflush() /* actual horizontal output occurs here */
825: {
826: int htrue;
827: int k, dir, n;
828:
829: if (!output)
830: return;
831: htrue = hpos + horig;
832: n = htrue - oldh;
833: if (htrue+n > HMAX)
834: n = HMAX - htrue;
835: if (htrue+n < 0)
836: n = -htrue;
837: htrue += n;
838: dir = (n >= 0) ? FWD : BACK;
839: if (dir != hdir)
840: putc(hdir = dir, tf);
841: if (n < 0)
842: n = -n;
843: while (n > 0) {
844: k = (n > MAXFWD) ? MAXFWD : n;
845: putc(HMOT | ~k, tf);
846: n -= k;
847: }
848: oldh = htrue + horig;
849: }
850:
851: vgoto(n)
852: {
853: vmot(n - vpos);
854: }
855:
856: vmot(n) /* generate n units of vertical motion */
857: int n;
858: {
859: int dir, k;
860: if (vpos+n > VMAX)
861: n = VMAX - vpos;
862: if (vpos+n < 0)
863: n = -vpos;
864: vpos += n;
865: vpos = ROUND3(vpos);
866: dir = (n >= 0) ? DOWN : UP;
867: if (dir != vdir)
868: putc(vdir = dir, tf);
869: if (n < 0)
870: n = -n;
871: n = ((n+1) / 3); /* round to multiple of 3 */
872: while (n > 0) {
873: k = (n > MAXDOWN) ? MAXDOWN : n;
874: putc(VMOT | (~k & MAXDOWN), tf);
875: n -= k;
876: }
877: if (vpos % 3 != 0)
878: fprintf(stderr, "vpos = %d\n", vpos);
879: }
880:
881:
882: put1s(s) /* s is a funny char name */
883: char *s;
884: {
885: int i;
886:
887: if (!output)
888: return;
889: if (dbg) printf("%s ", s);
890: for (i = 0; i < nchtab; i++)
891: if (strcmp(&chname[chtab[i]], s) == 0)
892: break;
893: if (i < nchtab)
894: put1(i + 128);
895: }
896:
897: put1(c) /* output char c */
898: int c;
899: {
900: char *pw;
901: register char *p;
902: register int i, k;
903: int j, ofont, code, w;
904:
905: if (!output)
906: return;
907: c -= 32;
908: if (c <= 0) {
909: if (dbg) printf("non-exist 0%o\n", c+32);
910: lastw = widthtab[font][0] * pstab[size-1] / dev.unitwidth;
911: return;
912: }
913: k = ofont = font;
914: i = fitab[font][c] & BMASK;
915: if (i != 0) { /* it's on this font */
916: p = codetab[font];
917: pw = widthtab[font];
918: } else if (smnt > 0) { /* on special (we hope) */
919: for (k=smnt, j=0; j <= nfonts; j++, k = (k+1) % (nfonts+1))
920: if ((i = fitab[k][c] & BMASK) != 0) {
921: p = codetab[k];
922: pw = widthtab[k];
923: setfont(k);
924: break;
925: }
926: }
927: if (i == 0 || (code = p[i] & BMASK) == 0 || k > nfonts) {
928: if (dbg) printf("not found 0%o\n", c+32);
929: return;
930: }
931: if (code < 64) { /* it's lower case */
932: if (curcase != LCASE)
933: putc(curcase = LCASE, tf);
934: } else {
935: if (curcase != UCASE)
936: putc(curcase = UCASE, tf);
937: code -= 64;
938: }
939: hflush();
940: if (dbg) {
941: if (isprint(c+32))
942: printf("%c %d\n", c+32, code);
943: else
944: printf("%03o %d\n", c+32, code);
945: } else
946: putc(code, tf); /* character is < 254 */
947: if (font != ofont)
948: setfont(ofont);
949: lastw = pw[i] & 077;
950: lastw = (lastw * pstab[size-1] + dev.unitwidth/2) / dev.unitwidth;
951: }
952:
953: setsize(n) /* set point size to n (internal) */
954: int n;
955: {
956: static int stab[] ={ 0,010,0,01,07,02,03,04,05,0211,06,0212,0213,0214,0215,0216,0217 };
957: int spos;
958:
959: if (!output)
960: return;
961: if (n == size)
962: return; /* already there */
963: putc(stab[n] & ~DOUBLE | SIZE, tf);
964: if (!(stab[size] & DOUBLE) && stab[n] & DOUBLE) {
965: spos = hpos;
966: hmot(55); /* single-double transition */
967: hpos = spos; /* lens-compensation motion doesn't affect absolute position */
968: } else if (stab[size] & DOUBLE && !(stab[n] & DOUBLE)) {
969: spos = hpos;
970: hmot(-55); /* double-single transition */
971: hpos = spos;
972: }
973: size = n;
974: }
975:
976: /* font position info: */
977:
978: struct {
979: char *name;
980: int number;
981: } fontname[NFONT+1];
982:
983: t_fp(n, s, si) /* font position n now contains font s, intname si */
984: int n;
985: char *s, *si;
986: {
987: fontname[n].name = s;
988: fontname[n].number = atoi(si);
989: }
990:
991: setfont(n) /* set font to n */
992: int n;
993: {
994: if (!output)
995: return;
996: /* needs stuff added for 8 font version */
997: /* easiest to loop over 3 font bits */
998: /* masking them off */
999: switch (n) {
1000: case 1:
1001: if (rail != LRAIL)
1002: putc(rail = LRAIL, tf);
1003: if (mag != LMAG)
1004: putc(mag = LMAG, tf);
1005: break;
1006: case 2:
1007: if (rail != URAIL)
1008: putc(rail = URAIL, tf);
1009: if (mag != LMAG)
1010: putc(mag = LMAG, tf);
1011: break;
1012: case 3:
1013: if (rail != LRAIL)
1014: putc(rail = LRAIL, tf);
1015: if (mag != UMAG)
1016: putc(mag = UMAG, tf);
1017: break;
1018: case 4:
1019: if (rail != URAIL)
1020: putc(rail = URAIL, tf);
1021: if (mag != UMAG)
1022: putc(mag = UMAG, tf);
1023: break;
1024: default:
1025: error(FATAL, "illegal font %d\n", n);
1026: }
1027: font = n;
1028: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.