t10.c revision 7c478bd95313f5f23a4c958a745db2134aa03244
/*
* 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 */
#pragma ident "%Z%%M% %I% %E% SMI"
/*
* 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();
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;
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");
*/
}
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);
}
findch(s) /* find char s in chname */
register char *s;
{
register int i;
for (i = 0; i < nchtab; i++)
if (strcmp(s, &chname[chtab[i]]) == 0)
return(i + 128);
return(0);
}
ptout(i)
register tchar i;
{
register dv;
register tchar *k;
int temp, a, b;
if (cbits(i) != '\n') {
*olinep++ = i;
return;
}
if (olinep == oline) {
lead += lss;
return;
}
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 */
}
tchar *
ptout0(pi)
tchar *pi;
{
register short j, k, w;
short z, dx, dy, dx2, dy2, n;
register 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);
}
ptps()
{
register 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;
}
ptfont()
{
mfont = xfont;
fdprintf(ptid, "f%d\n", xfont);
}
ptfpcmd(f, s)
int f;
char *s;
{
if (ascii)
return;
fdprintf(ptid, "x font %d %s\n", f, s);
ptfont(); /* make sure that it gets noticed */
}
ptlead()
{
vpos += lead;
if (!ascii)
fdprintf(ptid, "V%d\n", vpos);
lead = 0;
}
ptesc()
{
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;
}
newpage(n) /* called at end of each output page (we hope) */
{
int i;
ptlead();
vpos = 0;
if (ascii)
return;
fdprintf(ptid, "p%d\n", n); /* new page */
for (i = 0; i <= nfonts; i++)
if (fontbase[i]->namefont && fontbase[i]->namefont[0])
fdprintf(ptid, "x font %d %s\n", i, fontbase[i]->namefont);
ptps();
ptfont();
}
pttrailer()
{
fdprintf(ptid, "x trailer\n");
}
ptstop()
{
fdprintf(ptid, "x stop\n");
}
dostop()
{
if (ascii)
return;
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 */
}