Annotation of researchv10no/cmd/troff/Old/dcan.c, revision 1.1

1.1     ! root        1: /*
        !             2:  *     dcan
        !             3:  *     driver for impress/imagen canon laser printer
        !             4:  */
        !             5: 
        !             6: /*
        !             7: output language from troff:
        !             8: all numbers are character strings
        !             9: 
        !            10: sn     size in points
        !            11: fn     font as number from 1-n
        !            12: cx     ascii character x
        !            13: Cxyz   funny char xyz. terminated by white space
        !            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 x y r        arc counter-clockwise by x,y of radius r
        !            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:        b = space before line, a = after
        !            29: p      new page begins -- set v to 0
        !            30: #...\n comment
        !            31: x ...\n        device control functions:
        !            32:        x i     init
        !            33:        x T s   name of device is s
        !            34:        x r n h v       resolution is n/inch
        !            35:                h = min horizontal motion, v = min vert
        !            36:        x p     pause (can restart)
        !            37:        x s     stop -- done for ever
        !            38:        x t     generate trailer
        !            39:        x f n s font position n contains font s
        !            40:        x H n   set character height to n
        !            41:        x S n   set slant to N
        !            42: 
        !            43:        Subcommands like "i" are often spelled out like "init".
        !            44: */
        !            45: 
        !            46: #include       <stdio.h>
        !            47: #include       <signal.h>
        !            48: #include       <math.h>
        !            49: #include       <ctype.h>
        !            50: #include       "dev.h"
        !            51: #include       "/usr/bwk/imagen/src/imPcodes.h"
        !            52: #include       "glyph.h"
        !            53: 
        !            54: #define abs(n)  ((n) >= 0 ? (n) : -(n))
        !            55: 
        !            56: #define        NFONT   10
        !            57: 
        !            58: int    xx, yy;
        !            59: int    xoffset = 0;    /* shift output right by this amount */
        !            60: int    yoffset = -120; /* shift down (!) by this amount */
        !            61:                        /* -120 is 1/2 inch on canon */
        !            62:                        /* allows for 10 inch page -- */
        !            63:                        /* toss away 1/2 inch on top and bottom */
        !            64: float  xfac    = 1.0;  /* scaling factor for x */
        !            65: float  yfac    = 1.0;  /* scaling factor for y */
        !            66: int    inputarea       = 3;    /* input area = 3 * 8k bytes */
        !            67: int    rotate  = 0;    /* 0 => portrait, 1 => landscape */
        !            68: int    output  = 0;    /* do we do output at all? */
        !            69: int    pageno  = -1;   /* output page number */
        !            70: int    nolist  = 0;    /* output page list if > 0 */
        !            71: int    olist[20];      /* pairs of page numbers */
        !            72: 
        !            73: int    erase   = 1;
        !            74: float  aspect  = 1;    /* default aspect ratio */
        !            75: 
        !            76: struct dev dev;
        !            77: struct font *fontbase[NFONT+1];
        !            78: short  *pstab;
        !            79: int    nsizes  = 1;
        !            80: int    nfonts;
        !            81: int    smnt;   /* index of first special font */
        !            82: int    nchtab;
        !            83: char   *chname;
        !            84: short  *chtab;
        !            85: char   *fitab[NFONT+1];
        !            86: char   *widthtab[NFONT+1];     /* widtab would be a better name */
        !            87: char   *codetab[NFONT+1];      /* device codes */
        !            88: 
        !            89: #define        FATAL   1
        !            90: #define        BMASK   0377
        !            91: int    dbg     = 0;
        !            92: long   lineno  = 0;
        !            93: int    res     = 972;          /* input assumed computed according to this resolution */
        !            94:                                /* initial value to avoid 0 divide */
        !            95: FILE   *tf     = NULL;         /* output file pointer */
        !            96: char   *tempfile;
        !            97: char   *fontdir        = "/usr/lib/font";
        !            98: char   *bitdir         = "/usr/lib/font/devcan";
        !            99: char   *acctfile       = "/usr/adm/dcanacct";
        !           100: int    acctpages       = 0;
        !           101: int    copies          = 1;
        !           102: char   *username       = "???";
        !           103: char   *getlogin();
        !           104: extern char devname[];
        !           105: 
        !           106: FILE *fp       = stdin;        /* input file pointer */
        !           107: 
        !           108: main(argc, argv)
        !           109: char *argv[];
        !           110: {
        !           111:        char buf[BUFSIZ];
        !           112:        char *mktemp();
        !           113:        int done();
        !           114:        extern int DX, DY;
        !           115: 
        !           116:        username = getlogin();
        !           117:        while (argc > 1 && argv[1][0] == '-') {
        !           118:                switch (argv[1][1]) {
        !           119:                case 'c':
        !           120:                        copies = atoi(&argv[1][2]);
        !           121:                        break;
        !           122:                case 'r':
        !           123:                        rotate = !rotate;
        !           124:                        break;
        !           125:                case 't':
        !           126:                        tf = stdout;
        !           127:                        break;
        !           128:                case 'x':
        !           129:                        xoffset = atoi(&argv[1][2]);
        !           130:                        break;
        !           131:                case 'y':
        !           132:                        yoffset = atoi(&argv[1][2]);
        !           133:                        break;
        !           134:                case 'f':
        !           135:                        bitdir = argv[2];
        !           136:                        argv++;
        !           137:                        argc--;
        !           138:                        break;
        !           139:                case 'a':
        !           140:                        aspect = atof(&argv[1][2]);
        !           141:                        break;
        !           142:                case 'e':
        !           143:                        erase = 0;
        !           144:                        break;
        !           145:                case 'o':
        !           146:                        outlist(&argv[1][2]);
        !           147:                        break;
        !           148:                case 'i':       /* set input area parameter */
        !           149:                        inputarea = atoi(&argv[1][2]);
        !           150:                        if (inputarea < 1)
        !           151:                                inputarea = 1;
        !           152:                        else if (inputarea > 5)
        !           153:                                inputarea = 5;
        !           154:                        break;
        !           155:                case 'p':       /* pixels of resolution */
        !           156:                        DX = DY = atoi(&argv[1][2]);
        !           157:                        if (DX == 0)
        !           158:                                DX = DY = 1;
        !           159:                        break;
        !           160:                case 'd':
        !           161:                        dbg = atoi(&argv[1][2]);
        !           162:                        if (dbg == 0) dbg = 1;
        !           163:                        tf = stdout;
        !           164:                        break;
        !           165:                }
        !           166:                argc--;
        !           167:                argv++;
        !           168:        }
        !           169: 
        !           170:        signal(SIGINT, done);
        !           171:        signal(SIGHUP, done);
        !           172:        signal(SIGQUIT, done);
        !           173: 
        !           174:        tempfile = mktemp("/tmp/dcanXXXXX");
        !           175:        if (tf != stdout)
        !           176:                if ((tf = fopen(tempfile, "w")) == NULL)
        !           177:                        error(FATAL, "can't open temporary file %s", tempfile);
        !           178: 
        !           179:        if (argc <= 1)
        !           180:                conv(stdin);
        !           181:        else
        !           182:                while (--argc > 0) {
        !           183:                        if (strcmp(*++argv, "-") == 0)
        !           184:                                fp = stdin;
        !           185:                        else if ((fp = fopen(*argv, "r")) == NULL)
        !           186:                                error(FATAL, "can't open %s", *argv);
        !           187:                        conv(fp);
        !           188:                        fclose(fp);
        !           189:                }
        !           190:        fclose(tf);
        !           191:        sprintf(buf, "ipr -p %d -c %d %s", acctpages, copies, tempfile);
        !           192:        if(dbg){fprintf(stderr, "executing %s\n", buf); done();}
        !           193:        if (tf != stdout) {
        !           194:                system(buf);
        !           195:                account();
        !           196:        }
        !           197:        done();
        !           198: }
        !           199: 
        !           200: outlist(s)     /* process list of page numbers to be printed */
        !           201: char *s;
        !           202: {
        !           203:        int n1, n2, i;
        !           204: 
        !           205:        nolist = 0;
        !           206:        while (*s) {
        !           207:                n1 = 0;
        !           208:                if (isdigit(*s))
        !           209:                        do
        !           210:                                n1 = 10 * n1 + *s++ - '0';
        !           211:                        while (isdigit(*s));
        !           212:                else
        !           213:                        n1 = -9999;
        !           214:                n2 = n1;
        !           215:                if (*s == '-') {
        !           216:                        s++;
        !           217:                        n2 = 0;
        !           218:                        if (isdigit(*s))
        !           219:                                do
        !           220:                                        n2 = 10 * n2 + *s++ - '0';
        !           221:                                while (isdigit(*s));
        !           222:                        else
        !           223:                                n2 = 9999;
        !           224:                }
        !           225:                olist[nolist++] = n1;
        !           226:                olist[nolist++] = n2;
        !           227:                if (*s != '\0')
        !           228:                        s++;
        !           229:        }
        !           230:        olist[nolist] = 0;
        !           231:        if (dbg)
        !           232:                for (i=0; i<nolist; i += 2)
        !           233:                        printf("%3d %3d\n", olist[i], olist[i+1]);
        !           234: }
        !           235: 
        !           236: in_olist(n)    /* is n in olist? */
        !           237: int n;
        !           238: {
        !           239:        int i;
        !           240: 
        !           241:        if (nolist == 0)
        !           242:                return(1);      /* everything is included */
        !           243:        for (i = 0; i < nolist; i += 2)
        !           244:                if (n >= olist[i] && n <= olist[i+1])
        !           245:                        return(1);
        !           246:        return(0);
        !           247: }
        !           248: 
        !           249: conv(fp)
        !           250: register FILE *fp;
        !           251: {
        !           252:        register int c, k;
        !           253:        int m, n, i, n1, m1;
        !           254:        char str[100], buf[300];
        !           255: 
        !           256:        while ((c = getc(fp)) != EOF) {
        !           257:                switch (c) {
        !           258:                case '\n':      /* when input is text */
        !           259:                case ' ':
        !           260:                case 0:         /* occasional noise creeps in */
        !           261:                        break;
        !           262:                case '0': case '1': case '2': case '3': case '4':
        !           263:                case '5': case '6': case '7': case '8': case '9':
        !           264:                        /* two motion digits plus a character */
        !           265:                        hmot((c-'0')*10 + getc(fp)-'0');
        !           266:                        put1(getc(fp));
        !           267:                        break;
        !           268:                case 'c':       /* single ascii character */
        !           269:                        put1(getc(fp));
        !           270:                        break;
        !           271:                case 'C':
        !           272:                        fscanf(fp, "%s", str);
        !           273:                        put1s(str);
        !           274:                        break;
        !           275:                case 'D':       /* draw function */
        !           276:                        fgets(buf, sizeof(buf), fp);
        !           277:                        switch (buf[0]) {
        !           278:                        case 'l':       /* draw a line */
        !           279:                                sscanf(buf+1, "%d %d", &n, &m);
        !           280:                                drawline(n, m, ".");
        !           281:                                break;
        !           282:                        case 'c':       /* circle */
        !           283:                                sscanf(buf+1, "%d", &n);
        !           284:                                drawcirc(n);
        !           285:                                break;
        !           286:                        case 'e':       /* ellipse */
        !           287:                                sscanf(buf+1, "%d %d", &m, &n);
        !           288:                                drawellip(m, n);
        !           289:                                break;
        !           290:                        case 'a':       /* arc */
        !           291:                                sscanf(buf+1, "%d %d %d %d", &n, &m, &n1, &m1);
        !           292:                                drawarc(n, m, n1, m1);
        !           293:                                break;
        !           294:                        case '~':       /* wiggly line */
        !           295:                                drawwig(buf+1);
        !           296:                                break;
        !           297:                        default:
        !           298:                                error(FATAL, "unknown drawing function %s\n", buf);
        !           299:                                break;
        !           300:                        }
        !           301:                        break;
        !           302:                case 's':
        !           303:                        fscanf(fp, "%d", &n);   /* ignore fractional sizes */
        !           304:                        setsize(t_size(n));
        !           305:                        break;
        !           306:                case 'f':
        !           307:                        fscanf(fp, "%s", str);
        !           308:                        setfont(t_font(str));
        !           309:                        break;
        !           310:                case 'H':       /* absolute horizontal motion */
        !           311:                        /* fscanf(fp, "%d", &n); */
        !           312:                        while ((c = getc(fp)) == ' ')
        !           313:                                ;
        !           314:                        k = 0;
        !           315:                        do {
        !           316:                                k = 10 * k + c - '0';
        !           317:                        } while (isdigit(c = getc(fp)));
        !           318:                        ungetc(c, fp);
        !           319:                        hgoto(k);
        !           320:                        break;
        !           321:                case 'h':       /* relative horizontal motion */
        !           322:                        /* fscanf(fp, "%d", &n); */
        !           323:                        while ((c = getc(fp)) == ' ')
        !           324:                                ;
        !           325:                        k = 0;
        !           326:                        do {
        !           327:                                k = 10 * k + c - '0';
        !           328:                        } while (isdigit(c = getc(fp)));
        !           329:                        ungetc(c, fp);
        !           330:                        hmot(k);
        !           331:                        break;
        !           332:                case 'w':       /* word space */
        !           333:                        break;
        !           334:                case 'V':
        !           335:                        fscanf(fp, "%d", &n);
        !           336:                        vgoto(n);
        !           337:                        break;
        !           338:                case 'v':
        !           339:                        fscanf(fp, "%d", &n);
        !           340:                        vmot(n);
        !           341:                        break;
        !           342:                case 'p':       /* new page */
        !           343:                        fscanf(fp, "%d", &n);
        !           344:                        t_page(n);
        !           345:                        break;
        !           346:                case 'n':       /* end of line */
        !           347:                        while (getc(fp) != '\n')
        !           348:                                ;
        !           349:                        t_newline();
        !           350:                        break;
        !           351:                case '#':       /* comment */
        !           352:                        while (getc(fp) != '\n')
        !           353:                                ;
        !           354:                        break;
        !           355:                case 'x':       /* device control */
        !           356:                        devcntrl(fp);
        !           357:                        break;
        !           358:                default:
        !           359:                        error(!FATAL, "unknown input character %o %c\n", c, c);
        !           360:                        done();
        !           361:                }
        !           362:        }
        !           363: }
        !           364: 
        !           365: devcntrl(fp)   /* interpret device control functions */
        !           366: FILE *fp;
        !           367: {
        !           368:         char str[20], str1[50], buf[50];
        !           369:        int c, n;
        !           370: 
        !           371:        fscanf(fp, "%s", str);
        !           372:        switch (str[0]) {       /* crude for now */
        !           373:        case 'i':       /* initialize */
        !           374:                fileinit();
        !           375:                t_init(0);
        !           376:                break;
        !           377:        case 'T':       /* device name */
        !           378:                fscanf(fp, "%s", devname);
        !           379:                break;
        !           380:        case 't':       /* trailer */
        !           381:                t_trailer();
        !           382:                break;
        !           383:        case 'p':       /* pause -- can restart */
        !           384:                t_reset('p');
        !           385:                break;
        !           386:        case 's':       /* stop */
        !           387:                t_reset('s');
        !           388:                break;
        !           389:        case 'r':       /* resolution assumed when prepared */
        !           390:                fscanf(fp, "%d", &res);
        !           391:                break;
        !           392:        case 'f':       /* font used */
        !           393:                fscanf(fp, "%d %s", &n, str);
        !           394:                fgets(buf, sizeof buf, fp);     /* in case there's a filename */
        !           395:                ungetc('\n', fp);       /* fgets goes too far */
        !           396:                str1[0] = 0;    /* in case there's nothing to come in */
        !           397:                sscanf(buf, "%s", str1);
        !           398:                loadfont(n, str, str1);
        !           399:                break;
        !           400:        /* these don't belong here... */
        !           401:        case 'H':       /* char height */
        !           402:                fscanf(fp, "%d", &n);
        !           403:                t_charht(n);
        !           404:                break;
        !           405:        case 'S':       /* slant */
        !           406:                fscanf(fp, "%d", &n);
        !           407:                t_slant(n);
        !           408:                break;
        !           409:        }
        !           410:        while ((c = getc(fp)) != '\n')  /* skip rest of input line */
        !           411:                if (c == EOF)
        !           412:                        break;
        !           413: }
        !           414: 
        !           415: fileinit()     /* read in font and code files, etc. */
        !           416: {
        !           417:        int i, fin, nw;
        !           418:        char *malloc(), *filebase, *p;
        !           419:        char temp[60];
        !           420: 
        !           421:        /* open table for device,
        !           422:        /* read in resolution, size info, font info, etc.
        !           423:        /* and set params
        !           424:        */
        !           425:        strcpy(devname, "202"); /* this is the only char set we have */
        !           426:                                /* the resolution, etc., is already in */
        !           427:        sprintf(temp, "%s/dev%s/DESC.out", fontdir, devname);
        !           428:        if ((fin = open(temp, 0)) < 0)
        !           429:                error(FATAL, "can't open tables for %s\n", temp);
        !           430:        read(fin, &dev, sizeof(struct dev));
        !           431:        nfonts = dev.nfonts;
        !           432:        nsizes = dev.nsizes;
        !           433:        nchtab = dev.nchtab;
        !           434:        filebase = malloc(dev.filesize);        /* enough room for whole file */
        !           435:        read(fin, filebase, dev.filesize);      /* all at once */
        !           436:        pstab = (short *) filebase;
        !           437:        chtab = pstab + nsizes + 1;
        !           438:        chname = (char *) (chtab + dev.nchtab);
        !           439:        p = chname + dev.lchname;
        !           440:        for (i = 0; i <= nfonts; i++) {
        !           441:                fontbase[i] = NULL;
        !           442:                widthtab[i] = codetab[i] = fitab[i] = NULL;
        !           443:        }
        !           444:        close(fin);
        !           445: }
        !           446: 
        !           447: fontprint(i)   /* debugging print of font i (0,...) */
        !           448: {
        !           449:        int j, k, n;
        !           450:        char *p;
        !           451: 
        !           452:        printf("font %d:\n", i);
        !           453:        p = (char *) fontbase[i];
        !           454:        n = fontbase[i]->nwfont & BMASK;
        !           455:        printf("base=0%o, nchars=%d, spec=%d, name=%s, widtab=0%o, fitab=0%o\n",
        !           456:                p, n, fontbase[i]->specfont, fontbase[i]->namefont, widthtab[i], fitab[i]);
        !           457:        printf("widths:\n");
        !           458:        for (j=0; j <= n; j++) {
        !           459:                printf(" %2d", widthtab[i][j] & BMASK);
        !           460:                if (j % 20 == 19) printf("\n");
        !           461:        }
        !           462:        printf("\ncodetab:\n");
        !           463:        for (j=0; j <= n; j++) {
        !           464:                printf(" %2d", codetab[i][j] & BMASK);
        !           465:                if (j % 20 == 19) printf("\n");
        !           466:        }
        !           467:        printf("\nfitab:\n");
        !           468:        for (j=0; j <= dev.nchtab + 128-32; j++) {
        !           469:                printf(" %2d", fitab[i][j] & BMASK);
        !           470:                if (j % 20 == 19) printf("\n");
        !           471:        }
        !           472:        printf("\n");
        !           473: }
        !           474: 
        !           475: loadfont(n, s, s1)     /* load font info for font s on position n (0...) */
        !           476: int n;
        !           477: char *s, *s1;
        !           478: {
        !           479:        char temp[60];
        !           480:        int fin, nw, norig;
        !           481: 
        !           482:        if (n < 0 || n > NFONT)
        !           483:                error(FATAL, "illegal fp command %d %s", n, s);
        !           484:        if (fontbase[n] != NULL && strcmp(s, fontbase[n]->namefont) == 0)
        !           485:                return;
        !           486:        if (s1 == NULL || s1[0] == '\0')
        !           487:                sprintf(temp, "%s/dev%s/%s.out", fontdir, devname, s);
        !           488:        else
        !           489:                sprintf(temp, "%s/%s.out", s1, s);
        !           490:        if ((fin = open(temp, 0)) < 0) {
        !           491:                error(!FATAL, "can't open font table %s", temp);
        !           492:                return;
        !           493:        }
        !           494:        if (fontbase[n] != NULL)
        !           495:                free(fontbase[n]);
        !           496:        fontbase[n] = (struct font *) malloc(3*255 + dev.nchtab +
        !           497:                                (128-32) + sizeof(struct font));
        !           498:        if (fontbase[n] == NULL)
        !           499:                error(FATAL, "Out of space in loadfont %s\n", s);
        !           500:        read(fin, fontbase[n], 3*255 + nchtab+128-32 + sizeof(struct font));
        !           501:        close(fin);
        !           502:        if (smnt == 0 && fontbase[n]->specfont == 1)
        !           503:                smnt = n;
        !           504:        nw = fontbase[n]->nwfont & BMASK;
        !           505:        widthtab[n] = (char *) fontbase[n] + sizeof(struct font);
        !           506:        codetab[n] = (char *) widthtab[n] + 2 * nw;
        !           507:        fitab[n] = (char *) widthtab[n] + 3 * nw;
        !           508:        t_fp(n, fontbase[n]->namefont, fontbase[n]->intname);
        !           509:        if (dbg > 1) fontprint(n);
        !           510: }
        !           511: 
        !           512: 
        !           513: error(f, s, a1, a2, a3, a4, a5, a6, a7) {
        !           514:        fprintf(stderr, "dcan: ");
        !           515:        fprintf(stderr, s, a1, a2, a3, a4, a5, a6, a7);
        !           516:        fprintf(stderr, " near line %ld\n", lineno);
        !           517:        if (f)
        !           518:                done();
        !           519: }
        !           520: 
        !           521: 
        !           522: /*
        !           523:        Here beginneth all the stuff that really depends
        !           524:        on the canon (we hope).
        !           525: */
        !           526: 
        !           527: #define        RES     240             /* resolution of canon */
        !           528: #define        MAXX    (8*RES + RES/2)         /* 8-1/2 inches? */
        !           529: #define        MAXY    (11 * RES)
        !           530: #define        WIDTH   8
        !           531: #define        LOGWID  3
        !           532: #define        K       * 1024  /* clever, so watch out */
        !           533: 
        !           534: char   devname[20]     = "can";
        !           535: 
        !           536: int    nglyph          = 0;    /* number of glyphs loaded */
        !           537: int    totglyph        = 0;    /* total space used by glyphs sent down */
        !           538: int    maxglyph        = 16 K; /* maximum space for glyphs */
        !           539: 
        !           540: #define        oput(c) if (output) xychar(c); else;
        !           541: 
        !           542: /* input coordinate system: */
        !           543: 
        !           544: int    size    = 1;
        !           545: int    font    = 1;            /* current font */
        !           546: int    hpos;           /* horizontal position where we are supposed to be next (left = 0) */
        !           547: int    vpos;           /* current vertical position (down positive) */
        !           548: int    DX      = 10;   /* step size in x for drawing */
        !           549: int    DY      = 10;   /* step size in y for drawing */
        !           550: 
        !           551: /* canon coordinate system: */
        !           552: 
        !           553: int    lastsize        = -1;
        !           554: int    lastfont        = -1;
        !           555: int    lastx           = -1;
        !           556: int    lasty           = -1;
        !           557: int    lastfam         = -1;
        !           558: 
        !           559: int    drawdot = '.';  /* draw with this character */
        !           560: int    drawsize = 1;   /* shrink by this factor when drawing */
        !           561: 
        !           562: t_init(reinit) /* initialize device */
        !           563: int reinit;
        !           564: {
        !           565:        int i;
        !           566: 
        !           567:        if (! reinit) {
        !           568:                for (i = 0; i < nchtab; i++)
        !           569:                        if (strcmp(&chname[chtab[i]], "l.") == 0)
        !           570:                                break;
        !           571:                if (i < nchtab) {
        !           572:                        drawdot = i + 128;
        !           573:                        drawsize = 1;
        !           574:                } else {
        !           575:                        drawdot = '.';
        !           576:                        drawsize = 2;   /* half size */
        !           577:                }
        !           578: 
        !           579:                /* some Imagen-specific junk: */
        !           580:                fprintf(tf, "%1d", inputarea);  /* their kludge for setting */
        !           581:                                                /* input area to x * 8k */
        !           582:                maxglyph = 52 K - inputarea K - 4 K;
        !           583:                                                /* glyph area = 52K - input */
        !           584:                fprintf(tf, "  %s\n", username);
        !           585:                putc(0, tf);            /* terminate this [so they say] */
        !           586:                fprintf(tf, "%8.8s", "dcan1/24");       /* padding 8 bytes */
        !           587:                                                /* ignored but needed */
        !           588:                xfac = (float) RES / res * aspect;
        !           589:                yfac = (float) RES / res;
        !           590:        }
        !           591:        fflush(tf);
        !           592:        hpos = vpos = 0;
        !           593:        setsize(t_size(10));    /* start somewhere */
        !           594: }
        !           595: 
        !           596: t_page(pg)     /* do whatever new page functions */
        !           597: {
        !           598:        register int i, j, n;
        !           599:        register unsigned char *p;
        !           600:        static int firstpage = 1;
        !           601: 
        !           602:        pageno = pg;
        !           603:        if(dbg)fprintf(stderr, "t_page %d, output=%d\n", pg, output);
        !           604:        if (output != 0) {
        !           605:                /* beginning of first page, or */
        !           606:                /* have just printed something, and seen p<n> for next one */
        !           607:                /* ought to read in entire page, select needed glyphs */
        !           608:                putc(AEND, tf);
        !           609:                firstpage = 0;
        !           610:                fflush(tf);
        !           611:        }
        !           612:        output = in_olist(pg);
        !           613:        if (output) {
        !           614:                if (totglyph >= maxglyph) {
        !           615:                        clearglyphs();
        !           616:                        totglyph = 0;
        !           617:                }
        !           618:                putc(APAGE, tf);
        !           619:                acctpages++;
        !           620:        }
        !           621:        lastx = lasty = -1;
        !           622:        t_init(1);
        !           623: }
        !           624: 
        !           625: t_newline()    /* do whatever for the end of a line */
        !           626: {
        !           627:        hpos = 0;
        !           628: }
        !           629: 
        !           630: t_size(n)      /* convert integer to internal size number*/
        !           631: int n;
        !           632: {
        !           633:        int i;
        !           634: 
        !           635:        if (n <= pstab[0])
        !           636:                return(1);
        !           637:        else if (n >= pstab[nsizes-1])
        !           638:                return(nsizes);
        !           639:        for (i = 0; n > pstab[i]; i++)
        !           640:                ;
        !           641:        return(i+1);
        !           642: }
        !           643: 
        !           644: t_charht(n)    /* set character height to n */
        !           645: int n;
        !           646: {
        !           647:        /* punt for now */
        !           648: }
        !           649: 
        !           650: t_slant(n)     /* set slant to n */
        !           651: int n;
        !           652: {
        !           653:        /* punt for now */
        !           654: }
        !           655: 
        !           656: t_font(s)      /* convert string to internal font number */
        !           657: char *s;
        !           658: {
        !           659:        int n;
        !           660: 
        !           661:        n = atoi(s);
        !           662:        if (n < 1 || n > nfonts)
        !           663:                n = 1;
        !           664:        return(n);
        !           665: }
        !           666: 
        !           667: t_reset(c)
        !           668: {
        !           669:        int n;
        !           670: 
        !           671:        if (output)
        !           672:                acctpages++;
        !           673:        if (c == 's') {
        !           674:                putc(AEND, tf);
        !           675:                putc(AEOF, tf);
        !           676:        }
        !           677: }
        !           678: 
        !           679: account()      /* record paper use */
        !           680: {
        !           681:        FILE *f = NULL;
        !           682: 
        !           683:        if (tf == stdout)
        !           684:                return;
        !           685:        f = fopen(acctfile, "a");
        !           686:        if (f != NULL) {
        !           687:                if (username == NULL)
        !           688:                        username = "???";
        !           689:                fprintf(f, "%4d %s\n", acctpages, username);
        !           690:        }
        !           691: }
        !           692: 
        !           693: 
        !           694: t_trailer()
        !           695: {
        !           696: }
        !           697: 
        !           698: hgoto(n)
        !           699: {
        !           700:        hpos = n;       /* this is where we want to be */
        !           701:                        /* before printing a character, */
        !           702:                        /* have to make sure it's true */
        !           703: }
        !           704: 
        !           705: hmot(n)        /* generate n units of horizontal motion */
        !           706: int n;
        !           707: {
        !           708:        hpos += n;
        !           709: }
        !           710: 
        !           711: vgoto(n)
        !           712: {
        !           713:        vpos = n;
        !           714: }
        !           715: 
        !           716: vmot(n)        /* generate n units of vertical motion */
        !           717: int n;
        !           718: {
        !           719:        vgoto(vpos + n);        /* ignores rounding */
        !           720: }
        !           721: 
        !           722: put1s(s)       /* s is a funny char name */
        !           723:        register char *s;
        !           724: {
        !           725:        static int i = 0;
        !           726: 
        !           727:        if (!output)
        !           728:                return;
        !           729:        if (dbg) printf("%s ", s);
        !           730:        if (strcmp(s, &chname[chtab[i]]) != 0)
        !           731:                for (i = 0; i < nchtab; i++)
        !           732:                        if (strcmp(&chname[chtab[i]], s) == 0)
        !           733:                                break;
        !           734:        if (i < nchtab)
        !           735:                put1(i + 128);
        !           736:        else
        !           737:                i = 0;
        !           738: }
        !           739: 
        !           740: put1(c)        /* output char c */
        !           741:        register int c;
        !           742: {
        !           743:        char *pw;
        !           744:        register char *p;
        !           745:        register int i, j, k;
        !           746:        int ofont, code, w;
        !           747: 
        !           748:        if (!output)
        !           749:                return;
        !           750:        c -= 32;
        !           751:        if (c <= 0) {
        !           752:                if (dbg) printf("non-exist 0%o\n", c+32);
        !           753:                return;
        !           754:        }
        !           755:        k = ofont = font;
        !           756:        i = fitab[font][c] & BMASK;
        !           757:        if (i != 0) {   /* it's on this font */
        !           758:                p = codetab[font];
        !           759:                pw = widthtab[font];
        !           760:        } else if (smnt > 0) {          /* on special (we hope) */
        !           761:                for (k=smnt, j=0; j <= nfonts; j++, k = (k+1) % (nfonts+1))
        !           762:                        if ((i = fitab[k][c] & BMASK) != 0) {
        !           763:                                p = codetab[k];
        !           764:                                pw = widthtab[k];
        !           765:                                setfont(k);
        !           766:                                break;
        !           767:                        }
        !           768:        }
        !           769:        if (i == 0 || (code = p[i] & BMASK) == 0 || k > nfonts) {
        !           770:                if (dbg) printf("not found 0%o\n", c+32);
        !           771:                return;
        !           772:        }
        !           773:        if (dbg) {
        !           774:                if (isprint(c+32))
        !           775:                        printf("%c %d\n", c+32, code);
        !           776:                else
        !           777:                        printf("%03o %d\n", c+32, code);
        !           778:        } else
        !           779:                oput(code);
        !           780:        if (font != ofont)
        !           781:                setfont(ofont);
        !           782: }
        !           783: 
        !           784: setsize(n)     /* set point size to n (internal) */
        !           785: int n;
        !           786: {
        !           787:        size = n;
        !           788: }
        !           789: 
        !           790: /* font position info: */
        !           791: 
        !           792: struct {
        !           793:        char *name;
        !           794:        int number;
        !           795: } fontname[NFONT+1];
        !           796: 
        !           797: t_fp(n, s, si) /* font position n now contains font s, intname si */
        !           798: int n;
        !           799: char *s, *si;
        !           800: {
        !           801:        fontname[n].name = s;
        !           802:        fontname[n].number = atoi(si);
        !           803: }
        !           804: 
        !           805: setfont(n)     /* set font to n */
        !           806: int n;
        !           807: {
        !           808:        if (!output)
        !           809:                return;
        !           810:        if (n < 0 || n > NFONT)
        !           811:                error(FATAL, "illegal font %d\n", n);
        !           812:        font = n;
        !           813: }
        !           814: 
        !           815: done()
        !           816: {
        !           817:        exit(0);
        !           818: }
        !           819: 
        !           820: /*
        !           821:        The following things manage raster font information.
        !           822:        The big problem is mapping desired font + size into
        !           823:        available font + size.  For now, a file RASTERLIST
        !           824:        contains entries like
        !           825:                R 6 8 10 14 999
        !           826:                I 8 10 12 999
        !           827:                ...
        !           828:        This data is used to create an array "fontdata" that
        !           829:        describes legal fonts and sizes, and pointers to any
        !           830:        data from files that has actually been loaded.
        !           831: */
        !           832: 
        !           833: struct fontdata {
        !           834:        char    name[4];        /* e.g., "R" or "PA" */
        !           835:        int     size[10];       /* e.g., 6 8 10 14 0 */
        !           836:        struct  fontset *fsp[10];       /* either NULL or block of data */
        !           837: };
        !           838: 
        !           839: #define        MAXFONT 20      /* no more than this many fonts forever */
        !           840: 
        !           841: struct fontdata        fontdata[MAXFONT];
        !           842: int    maxfonts        = 0;    /* how many actually used; set in initfontdata() */
        !           843: 
        !           844: struct Fontheader      fh;
        !           845: struct fontset {
        !           846:        int     size;
        !           847:        int     family;
        !           848:        struct  Charparam *chp;
        !           849:        unsigned char   *cdp;   /* char data pointer */
        !           850:        unsigned char   *chused;        /* bit-indexed; 1 if char downloaded */
        !           851: };
        !           852: 
        !           853: /* A global variable for the current font+size */
        !           854: struct fontset *fs;
        !           855: int    nfamily         = 0;    /* number of "families" (font+size) */
        !           856: 
        !           857: initfontdata() /* read RASTERLIST information */
        !           858: {
        !           859:        char name[100];
        !           860:        FILE *fp;
        !           861:        int i, j, n;
        !           862: 
        !           863:        sprintf(name, "%s/RASTERLIST", bitdir);
        !           864:        if ((fp = fopen(name, "r")) == NULL)
        !           865:                error(FATAL, "can't open %s\n", name);
        !           866:        maxfonts = 0;
        !           867:        while (fscanf(fp, "%s", fontdata[maxfonts].name) != EOF) {
        !           868:                i = 0;
        !           869:                while (fscanf(fp, "%d", &n) != EOF && n < 100) {
        !           870:                        fontdata[maxfonts].size[i] = n;
        !           871:                        fontdata[maxfonts].fsp[i] = NULL;
        !           872:                        i++;
        !           873:                }
        !           874:                fontdata[maxfonts].size[i] = 999;
        !           875:                if (++maxfonts > MAXFONT)
        !           876:                        error(FATAL, "Too many fonts in RASTERLIST");
        !           877:        }
        !           878:        fclose(fp);
        !           879:        if (dbg) {
        !           880:                fprintf(stderr, "initfontdata():  maxfonts=%d\n", maxfonts);
        !           881:                for (i = 0; i < maxfonts; i++) {
        !           882:                        fprintf(stderr, "%.4s ", fontdata[i].name);
        !           883:                        for (j = 0; fontdata[i].size[j] < 100; j++)
        !           884:                                fprintf(stderr, " %3d", fontdata[i].size[j]);
        !           885:                        fprintf(stderr, "\n");
        !           886:                }
        !           887:        }
        !           888: }
        !           889: 
        !           890: getfontdata(f, s)      /* causes loading of font information if needed */
        !           891:        char *f;
        !           892:        int s;
        !           893: {
        !           894:        int fd, n, i, j;
        !           895:        char name[100];
        !           896:        static int first = 1;
        !           897: 
        !           898:        if (first) {
        !           899:                initfontdata();
        !           900:                first = 0;
        !           901:        }
        !           902: 
        !           903:        for (i = 0; i < maxfonts; i++)
        !           904:                if (strcmp(f, fontdata[i].name) == 0)
        !           905:                        break;
        !           906:        if (i >= maxfonts)      /* the requested font wasn't there */
        !           907:                i = 0;          /* use the first one (probably R) */
        !           908: 
        !           909:        /* find the best approximation to size s */
        !           910:        for (j = 1; s >= fontdata[i].size[j]; j++)
        !           911:                ;
        !           912:        j--;
        !           913: 
        !           914:        /* open file if necessary */
        !           915:        if (fontdata[i].fsp[j] == NULL) {
        !           916:                fs = (struct fontset *) malloc(sizeof(struct fontset));
        !           917:                fontdata[i].fsp[j] = fs;
        !           918:                fs->chp = (struct Charparam *) malloc(256*sizeof(struct Charparam));
        !           919:                sprintf(name, "%s/%s.%d%s", bitdir,
        !           920:                        f, fontdata[i].size[j], rotate? "r" : "");
        !           921:                fd = open(name, 0);
        !           922:                if (fd == -1)
        !           923:                        error(FATAL, "can't open %s\n", name);
        !           924:                read(fd, &fh, sizeof(struct Fontheader));
        !           925:                read(fd, fs->chp, 256*sizeof(struct Charparam));
        !           926:                fs->size = fontdata[i].size[j];
        !           927:                fs->family = nfamily;
        !           928:                nfamily += 2;   /* even-odd leaves room for big fonts */
        !           929:                fs->cdp = (unsigned char *) malloc(fh.f_size);
        !           930:                fs->chused = (unsigned char *) malloc(256/8);
        !           931:                read(fd, fs->cdp, fh.f_size);
        !           932:                close(fd);
        !           933:        }
        !           934:        fs = fontdata[i].fsp[j];
        !           935: }
        !           936: 
        !           937: xychar(c)
        !           938:        register int c;
        !           939: {
        !           940:        register unsigned char *p;
        !           941:        register struct Charparam *par;
        !           942:        register int x, y;
        !           943:        int i, n, rwid, ht, fam;
        !           944: 
        !           945:        x = hpos * xfac + 0.5;
        !           946:        x += xoffset;
        !           947:        y = vpos * yfac + 0.5;
        !           948:        y += yoffset;
        !           949: 
        !           950:        if (font != lastfont || size != lastsize) {
        !           951:                getfontdata(fontname[font].name, pstab[size-1]);
        !           952:                lastsize = size;
        !           953:                lastfont = font;
        !           954:        }
        !           955:        par = fs->chp + c;
        !           956:        p = fs->cdp + par->c_addr;
        !           957: 
        !           958:        fam = fs->family;
        !           959:        if (c > 127)
        !           960:                fam++;
        !           961:        if (fam != lastfam) {
        !           962:                putc(AF, tf);
        !           963:                putc(lastfam = fam, tf);
        !           964:        }
        !           965: 
        !           966:        /* first cut:  ship each glyph as needed. */
        !           967:        /* ignore memory use, efficiency, etc. */
        !           968: 
        !           969:        if ( !bit(fs->chused, c) ) {    /* 1st use of this character */
        !           970:                nglyph++;
        !           971:                totglyph += glspace(par);
        !           972:                setbit(fs->chused, c);
        !           973:                putc(ASGLY, tf);
        !           974:                putint((fam << 7) | c, tf);
        !           975: /*             putc(par->c_width, tf); /* character width */
        !           976:                putc(0, tf);    /* zap nominal width because it's wrong */
        !           977:                putc(par->c_left + par->c_right + 1, tf);
        !           978:                putc(par->c_left, tf);
        !           979:   /* this nonsense fixes a bug in output produced by rec.c: */
        !           980:   /* when up is < 0 (and = 0?) size is one too big */
        !           981:                rwid = (1 + par->c_left + par->c_right + WIDTH-1) / WIDTH;
        !           982:                ht = par->c_size / rwid;
        !           983:                par->c_down = ht - par->c_up;
        !           984:                putc(par->c_down + par->c_up, tf);
        !           985:                putc(par->c_up, tf);
        !           986:                for (i = par->c_size; i--; )
        !           987:                        putc(*p++, tf);
        !           988:        }
        !           989: 
        !           990:        if (y != lasty) {
        !           991:                putc(AV, tf);
        !           992:                putint(y<<1, tf);
        !           993:                lasty = y;
        !           994:        }
        !           995:        if (x != lastx) {
        !           996:                if (abs(x-lastx) > 127) {
        !           997:                        putc(AH, tf);
        !           998:                        putint(x<<1, tf);
        !           999:                } else {
        !          1000:                        putc(AM, tf);
        !          1001:                        putc(x-lastx, tf);
        !          1002:                        putc(AM, tf);
        !          1003:                }
        !          1004:        }
        !          1005: 
        !          1006:        if (c <= 127)
        !          1007:                putc(c, tf);    /* fails if c > 127, probably disastrously */
        !          1008:        else
        !          1009:                putc(c-128, tf);
        !          1010: /*     lastx = x + par->c_width; */
        !          1011:        lastx = x;
        !          1012: }
        !          1013: 
        !          1014: glspace(par)
        !          1015:        struct Charparam *par;
        !          1016: {
        !          1017:        int n;
        !          1018: 
        !          1019:        /* works only for small glyphs right now */
        !          1020: 
        !          1021:        n = 12
        !          1022:          + ((par->c_left+par->c_right+1+15)/16 ) * (par->c_up+par->c_down)
        !          1023:          + 2;
        !          1024:        return n;
        !          1025: }
        !          1026: 
        !          1027: clearglyphs()  /* remove "used" bits from all glyphs */
        !          1028:                /* delete all families */
        !          1029:                /* very conservative policy */
        !          1030: {
        !          1031:        int i, j, k;
        !          1032:        struct fontset *f;
        !          1033: 
        !          1034:        if (tf == stdout) fprintf(stderr, "clear %d glyphs (%d/%d) on page %d\n",
        !          1035:                nglyph, totglyph, maxglyph, pageno);
        !          1036:        for (i = 0; i < maxfonts; i++)
        !          1037:                for (j = 0; fontdata[i].size[j] < 999; j++) {
        !          1038:                        f = fontdata[i].fsp[j];
        !          1039:                        if (f != NULL) {
        !          1040:                                putc(ADELF, tf);
        !          1041:                                putc(f->family, tf);
        !          1042:                                for (k = 0; k < 256/8; k++)
        !          1043:                                        f->chused[k] = 0;
        !          1044:                        }
        !          1045:                }
        !          1046: }
        !          1047: 
        !          1048: bit(p, n)      /* return n-th bit of p[] */
        !          1049:        char *p;
        !          1050:        int n;
        !          1051: {
        !          1052:        return (p[n/8] >> (7 - n%8)) & 01;
        !          1053: }
        !          1054: 
        !          1055: setbit(p, n)   /* set bit n of p[] */
        !          1056:        char *p;
        !          1057:        int n;
        !          1058: {
        !          1059:        p[n/8] |= 01 << (7 - n%8);
        !          1060: }
        !          1061: 
        !          1062: putint(n, f)
        !          1063:        int n;
        !          1064:        FILE *f;
        !          1065: {
        !          1066:        putc(n >> 8, f);
        !          1067:        putc(n & 0377, f);
        !          1068: }

unix.superglobalmegacorp.com

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