Annotation of researchv10no/cmd/troff/d202.c, revision 1.1

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

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.