root/usr/src/cmd/troff/troff.d/t10.c
/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License, Version 1.0 only
 * (the "License").  You may not use this file except in compliance
 * with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright 1989 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*      Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
/*        All Rights Reserved   */

/*
 * University Copyright- Copyright (c) 1982, 1986, 1988
 * The Regents of the University of California
 * All Rights Reserved
 *
 * University Acknowledgment- Portions of this document are derived from
 * software developed by the University of California, Berkeley, and its
 * contributors.
 */

#include "tdef.h"
#include <ctype.h>
#include "ext.h"
/*
 * troff10.c
 *
 * typesetter interface
 */

int     vpos     = 0;   /* absolute vertical position on page */
int     hpos     = 0;   /* ditto horizontal */

short   *chtab;
char    *chname;
char    *fontab[NFONT+1];
char    *kerntab[NFONT+1];
char    *fitab[NFONT+1];
char    *codetab[NFONT+1];

int     Inch;
int     Hor;
int     Vert;
int     Unitwidth;
int     nfonts;
int     nsizes;
int     nchtab;

/* these characters are used as various signals or values
 * in miscellaneous places.
 * values are set in specnames in t10.c
 */

int     c_hyphen;
int     c_emdash;
int     c_rule;
int     c_minus;
int     c_fi;
int     c_fl;
int     c_ff;
int     c_ffi;
int     c_ffl;
int     c_acute;
int     c_grave;
int     c_under;
int     c_rooten;
int     c_boxrule;
int     c_lefthand;
int     c_dagger;

#include "dev.h"
struct dev dev;
struct Font *fontbase[NFONT+1];

tchar *ptout0();


int
ptinit()
{
        int     i, fin, nw;
        char    *setbrk(), *filebase, *p;

        /* open table for device,
         * read in resolution, size info, font info, etc.
         * and set params
         */
        strcat(termtab, "/dev");
        strcat(termtab, devname);
        strcat(termtab, "/DESC.out");   /* makes "..../devXXX/DESC.out" */
        if ((fin = open(termtab, 0)) < 0) {
                errprint(gettext("can't open tables for %s"), termtab);
                done3(1);
        }
        read(fin, (char *) &dev, sizeof(struct dev ));
        Inch = dev.res;
        Hor = dev.hor;
        Vert = dev.vert;
        Unitwidth = dev.unitwidth;
        nfonts = dev.nfonts;
        nsizes = dev.nsizes;
        nchtab = dev.nchtab;
        if (nchtab >= NCHARS - 128) {
                errprint(gettext("too many special characters in file %s"),
                        termtab);
                done3(1);
        }
        filebase = setbrk(dev.filesize + 2*EXTRAFONT);  /* enough room for whole file */
        read(fin, filebase, dev.filesize);      /* all at once */
        pstab = (short *) filebase;
        chtab = pstab + nsizes + 1;
        chname = (char *) (chtab + dev.nchtab);
        p = chname + dev.lchname;
        for (i = 1; i <= nfonts; i++) {
                fontbase[i] = (struct Font *) p;
                nw = *p & BYTEMASK;     /* 1st thing is width count */
                fontlab[i] = PAIR(fontbase[i]->namefont[0], fontbase[i]->namefont[1]);
                /* for now, still 2 char names */
                if (smnt == 0 && fontbase[i]->specfont == 1)
                        smnt = i;       /* first special font */
                p += sizeof(struct Font);       /* that's what's on the beginning */
                fontab[i] = p;
                kerntab[i] = p + nw;
                codetab[i] = p + 2 * nw;
                fitab[i] = p + 3 * nw;  /* skip width, kern, code */
                p += 3 * nw + dev.nchtab + 128 - 32;
        }
        fontbase[0] = (struct Font *) p;        /* the last shall be first */
        fontbase[0]->nwfont = EXTRAFONT - dev.nchtab - (128-32) - sizeof (struct Font);
        fontab[0] = p + sizeof (struct Font);
        close(fin);
        /* there are a lot of things that used to be constant
         * that now require code to be executed.
         */
        sps = SPS;
        ics = ICS;
        for (i = 0; i < 16; i++)
                tabtab[i] = DTAB * (i + 1);
        pl = 11 * INCH;
        po = PO;
        spacesz = SS;
        lss = lss1 = VS;
        ll = ll1 = lt = lt1 = LL;
        specnames();    /* install names like "hyphen", etc. */
        if (ascii)
                return (0);
        fdprintf(ptid, "x T %s\n", devname);
        fdprintf(ptid, "x res %d %d %d\n", Inch, Hor, Vert);
        fdprintf(ptid, "x init\n");     /* do initialization for particular device */
  /*
        for (i = 1; i <= nfonts; i++)
                fdprintf(ptid, "x font %d %s\n", i, fontbase[i]->namefont);
        fdprintf(ptid, "x xxx fonts=%d sizes=%d unit=%d\n", nfonts, nsizes, Unitwidth);
        fdprintf(ptid, "x xxx nchtab=%d lchname=%d nfitab=%d\n",
                dev.nchtab, dev.lchname, dev.nchtab+128-32);
        fdprintf(ptid, "x xxx sizes:\nx xxx ");
        for (i = 0; i < nsizes; i++)
                fdprintf(ptid, " %d", pstab[i]);
        fdprintf(ptid, "\nx xxx chars:\nx xxx ");
        for (i = 0; i < dev.nchtab; i++)
                fdprintf(ptid, " %s", &chname[chtab[i]]);
        fdprintf(ptid, "\nx xxx\n");
  */

        return (0);
}

int
specnames()
{
        static struct {
                int     *n;
                char    *v;
        } spnames[] = {
                &c_hyphen, "hy",
                &c_emdash, "em",
                &c_rule, "ru",
                &c_minus, "\\-",
                &c_fi, "fi",
                &c_fl, "fl",
                &c_ff, "ff",
                &c_ffi, "Fi",
                &c_ffl, "Fl",
                &c_acute, "aa",
                &c_grave, "ga",
                &c_under, "ul",
                &c_rooten, "rn",
                &c_boxrule, "br",
                &c_lefthand, "lh",
                &c_dagger, "dg",
                0, 0
        };
        int     i;

        for (i = 0; spnames[i].n; i++)
                *spnames[i].n = findch(spnames[i].v);

        return (0);
}

int
findch(s)       /* find char s in chname */
char    *s;
{
        int     i;

        for (i = 0; i < nchtab; i++)
                if (strcmp(s, &chname[chtab[i]]) == 0)
                        return(i + 128);
        return(0);
}

int
ptout(i)
tchar   i;
{
        int     dv;
        tchar   *k;
        int temp, a, b;

        if (cbits(i) != '\n') {
                *olinep++ = i;
                return (0);
        }
        if (olinep == oline) {
                lead += lss;
                return (0);
        }

        hpos = po;      /* ??? */
        esc = 0;        /* ??? */
        ptesc();        /* the problem is to get back to the left end of the line */
        dv = 0;
        for (k = oline; k < olinep; k++) {
                if (ismot(*k) && isvmot(*k)) {
                        temp = absmot(*k);
                        if (isnmot(*k))
                                temp = -temp;
                        dv += temp;
                }
        }
        if (dv) {
                vflag++;
                *olinep++ = makem(-dv);
                vflag = 0;
        }

        b = dip->blss + lss;
        lead += dip->blss + lss;
        dip->blss = 0;
        for (k = oline; k < olinep; )
                k = ptout0(k);  /* now passing a pointer! */
        olinep = oline;
        lead += dip->alss;
        a = dip->alss;
        dip->alss = 0;
        /*
        fdprintf(ptid, "x xxx end of line: hpos=%d, vpos=%d\n", hpos, vpos);
*/
        fdprintf(ptid, "n%d %d\n", b, a);       /* be nice to chuck */

        return (0);
}

tchar *
ptout0(pi)
tchar   *pi;
{
        short j, k, w;
        short   z, dx, dy, dx2, dy2, n;
        tchar   i;
        int outsize;    /* size of object being printed */

        outsize = 1;    /* default */
        i = *pi;
        k = cbits(i);
        if (ismot(i)) {
                j = absmot(i);
                if (isnmot(i))
                        j = -j;
                if (isvmot(i))
                        lead += j;
                else
                        esc += j;
                return(pi+outsize);
        }
        if (k == XON) {
                if (xfont != mfont)
                        ptfont();
                if (xpts != mpts)
                        ptps();
                if (lead)
                        ptlead();
                fdprintf(ptid, "x X ");
                /*
             * not guaranteed of finding a XOFF if a word overflow
                 * error occured, so also bound this loop by olinep
                 */
                pi++;
                while( cbits(*pi) != XOFF && pi < olinep )
                        outascii(*pi++);
                oput('\n');
                if ( cbits(*pi) == XOFF )
                        pi++;
                return pi;
        }
                        ;
        if (k == CHARHT) {
                if (xpts != mpts)
                        ptps();
                fdprintf(ptid, "x H %d\n", sbits(i));
                return(pi+outsize);
        }
        if (k == SLANT) {
                fdprintf(ptid, "x S %d\n", sfbits(i)-180);
                return(pi+outsize);
        }
        if (k == WORDSP) {
                oput('w');
                return(pi+outsize);
        }
        if (k == FONTPOS) {
                char temp[3];
                n = i >> 16;
                temp[0] = n & BYTEMASK;
                temp[1] = n >> BYTE;
                temp[2] = 0;
                ptfpcmd(0, temp);
                return(pi+outsize);
        }
        if (sfbits(i) == oldbits) {
                xfont = pfont;
                xpts = ppts;
        } else
                xbits(i, 2);
        if (k < 040 && k != DRAWFCN)
                return(pi+outsize);
        if (k >= 32) {
                if (widcache[k-32].fontpts == (xfont<<8) + xpts  && !setwdf) {
                        w = widcache[k-32].width;
                        bd = 0;
                        cs = 0;
                } else
                        w = getcw(k-32);
        }
        j = z = 0;
        if (k != DRAWFCN) {
                if (cs) {
                        if (bd)
                                w += (bd - 1) * HOR;
                        j = (cs - w) / 2;
                        w = cs - j;
                        if (bd)
                                w -= (bd - 1) * HOR;
                }
                if (iszbit(i)) {
                        if (cs)
                                w = -j;
                        else
                                w = 0;
                        z = 1;
                }
        }
        esc += j;
        if (xfont != mfont)
                ptfont();
        if (xpts != mpts)
                ptps();
        if (lead)
                ptlead();
        /* put out the real character here */
        if (k == DRAWFCN) {
                if (esc)
                        ptesc();
                dx = absmot(pi[3]);
                if (isnmot(pi[3]))
                        dx = -dx;
                dy = absmot(pi[4]);
                if (isnmot(pi[4]))
                        dy = -dy;
                switch (cbits(pi[1])) {
                case DRAWCIRCLE:        /* circle */
                        fdprintf(ptid, "D%c %d\n", DRAWCIRCLE, dx);     /* dx is diameter */
                        w = 0;
                        hpos += dx;
                        break;
                case DRAWELLIPSE:
                        fdprintf(ptid, "D%c %d %d\n", DRAWELLIPSE, dx, dy);
                        w = 0;
                        hpos += dx;
                        break;
                case DRAWLINE:  /* line */
                        k = cbits(pi[2]);
                        fdprintf(ptid, "D%c %d %d ", DRAWLINE, dx, dy);
                        if (k < 128)
                                fdprintf(ptid, "%c\n", k);
                        else
                                fdprintf(ptid, "%s\n", &chname[chtab[k - 128]]);
                        w = 0;
                        hpos += dx;
                        vpos += dy;
                        break;
                case DRAWARC:   /* arc */
                        dx2 = absmot(pi[5]);
                        if (isnmot(pi[5]))
                                dx2 = -dx2;
                        dy2 = absmot(pi[6]);
                        if (isnmot(pi[6]))
                                dy2 = -dy2;
                        fdprintf(ptid, "D%c %d %d %d %d\n", DRAWARC,
                                dx, dy, dx2, dy2);
                        w = 0;
                        hpos += dx + dx2;
                        vpos += dy + dy2;
                        break;
                case DRAWSPLINE:        /* spline */
                default:        /* something else; copy it like spline */
                        fdprintf(ptid, "D%c %d %d", cbits(pi[1]), dx, dy);
                        w = 0;
                        hpos += dx;
                        vpos += dy;
                        if (cbits(pi[3]) == DRAWFCN || cbits(pi[4]) == DRAWFCN) {
                                /* it was somehow defective */
                                fdprintf(ptid, "\n");
                                break;
                        }
                        for (n = 5; cbits(pi[n]) != DRAWFCN; n += 2) {
                                dx = absmot(pi[n]);
                                if (isnmot(pi[n]))
                                        dx = -dx;
                                dy = absmot(pi[n+1]);
                                if (isnmot(pi[n+1]))
                                        dy = -dy;
                                fdprintf(ptid, " %d %d", dx, dy);
                                hpos += dx;
                                vpos += dy;
                        }
                        fdprintf(ptid, "\n");
                        break;
                }
                for (n = 3; cbits(pi[n]) != DRAWFCN; n++)
                        ;
                outsize = n + 1;
        } else if (k < 128) {
                /* try to go faster and compress output */
                /* by printing nnc for small positive motion followed by c */
                /* kludgery; have to make sure set all the vars too */
                if (esc > 0 && esc < 100) {
                        oput(esc / 10 + '0');
                        oput(esc % 10 + '0');
                        oput(k);
                        hpos += esc;
                        esc = 0;
                } else {
                        if (esc)
                                ptesc();
                        oput('c');
                        oput(k);
                        oput('\n');
                }
        } else {
                if (esc)
                        ptesc();
                if (k >= nchtab + 128)
                        fdprintf(ptid, "N%d\n", k - (nchtab+128));
                else
                        fdprintf(ptid, "C%s\n", &chname[chtab[k - 128]]);
        }
        if (bd) {
                bd -= HOR;
                if (esc += bd)
                        ptesc();
                if (k < 128) {
                        fdprintf(ptid, "c%c\n", k);
                } else if (k >= nchtab + 128) {
                        fdprintf(ptid, "N%d\n", k - (nchtab+128));
                } else
                        fdprintf(ptid, "C%s\n", &chname[chtab[k - 128]]);
                if (z)
                        esc -= bd;
        }
        esc += w;
        return(pi+outsize);
}

int
ptps()
{
        int     i, j, k;

        i = xpts;
        for (j = 0; i > (k = pstab[j]); j++)
                if (!k) {
                        k = pstab[--j];
                        break;
                }
        fdprintf(ptid, "s%d\n", k);     /* really should put out string rep of size */
        mpts = i;

        return (0);
}

int
ptfont()
{
        mfont = xfont;
        fdprintf(ptid, "f%d\n", xfont);

        return (0);
}

int
ptfpcmd(f, s)
int     f;
char    *s;
{
        if (ascii)
                return (0);
        fdprintf(ptid, "x font %d %s\n", f, s);
        ptfont();       /* make sure that it gets noticed */

        return (0);
}

int
ptlead()
{
        vpos += lead;
        if (!ascii)
                fdprintf(ptid, "V%d\n", vpos);
        lead = 0;

        return (0);
}


int
ptesc(void)
{
        hpos += esc;
        if (esc > 0) {
                oput('h');
                if (esc>=10 && esc<100) {
                        oput(esc/10 + '0');
                        oput(esc%10 + '0');
                } else
                        fdprintf(ptid, "%d", esc);
        } else
                fdprintf(ptid, "H%d\n", hpos);
        esc = 0;

        return (0);
}


int
newpage(int n)  /* called at end of each output page (we hope) */
{
        int i;

        ptlead();
        vpos = 0;
        if (ascii)
                return (0);
        fdprintf(ptid, "p%d\n", n);     /* new page */
        for (i = 0; i <= nfonts; i++)
                if (fontbase[i]->namefont[0] != '\0')
                        fdprintf(ptid, "x font %d %s\n", i,
                            fontbase[i]->namefont);
        ptps();
        ptfont();

        return (0);
}

int
pttrailer(void)
{
        fdprintf(ptid, "x trailer\n");

        return (0);
}

int
ptstop(void)
{
        fdprintf(ptid, "x stop\n");

        return (0);
}

int
dostop(void)
{
        if (ascii)
                return (0);
        ptlead();
        vpos = 0;
        /* fdprintf(ptid, "x xxx end of page\n");*/
        if (!nofeed)
                pttrailer();
        ptlead();
        fdprintf(ptid, "x pause\n");
        flusho();
        mpts = mfont = 0;
        ptesc();
        esc = po;
        hpos = vpos = 0;        /* probably in wrong place */

        return (0);
}