xref: /plan9/sys/src/cmd/postscript/tr2post/chartab.c (revision 456a8764e4ea95d7aa2c2cf34e5112293070bc84)
1219b2ee8SDavid du Colombier /*    Unicode   |     PostScript
2219b2ee8SDavid du Colombier  *  start  end  | offset  font name
3219b2ee8SDavid du Colombier  * 0x0000 0x00ff  0x00   LucidaSansUnicode00
4219b2ee8SDavid du Colombier  */
5219b2ee8SDavid du Colombier #include <u.h>
6219b2ee8SDavid du Colombier #include <libc.h>
7219b2ee8SDavid du Colombier #include <bio.h>
8219b2ee8SDavid du Colombier #include "common.h"
9219b2ee8SDavid du Colombier #include "tr2post.h"
10219b2ee8SDavid du Colombier #include "comments.h"
11219b2ee8SDavid du Colombier #include "path.h"
12219b2ee8SDavid du Colombier 
13219b2ee8SDavid du Colombier /* Postscript font names, e.g., `LucidaSansUnicode00'
14219b2ee8SDavid du Colombier  * names may only be added because reference to the
15219b2ee8SDavid du Colombier  * names is made by indexing into this table.
16219b2ee8SDavid du Colombier  */
17219b2ee8SDavid du Colombier static struct pfnament *pfnafontmtab = 0;
18219b2ee8SDavid du Colombier static int pfnamcnt = 0;
19219b2ee8SDavid du Colombier int curpostfontid = -1;
20219b2ee8SDavid du Colombier int curfontsize = -1;
21219b2ee8SDavid du Colombier int curtrofffontid = -1;
22219b2ee8SDavid du Colombier static int curfontpos = -1;
237dd7cddfSDavid du Colombier static int fontheight = 0;
247dd7cddfSDavid du Colombier static int fontslant = 0;
25219b2ee8SDavid du Colombier 
26219b2ee8SDavid du Colombier /* This is troffs mounted font table.  It is an anachronism resulting
27219b2ee8SDavid du Colombier  * from the design of the APS typesetter.  fontmnt is the
28219b2ee8SDavid du Colombier  * number of positions available.  fontmnt is really 11, but
29219b2ee8SDavid du Colombier  * should not be limited.
30219b2ee8SDavid du Colombier  */
31219b2ee8SDavid du Colombier int fontmnt = 0;
32219b2ee8SDavid du Colombier char **fontmtab;
33219b2ee8SDavid du Colombier 
34219b2ee8SDavid du Colombier struct troffont *troffontab = 0;
35219b2ee8SDavid du Colombier 
36219b2ee8SDavid du Colombier int troffontcnt = 0;
37219b2ee8SDavid du Colombier 
38219b2ee8SDavid du Colombier void
mountfont(int pos,char * fontname)39219b2ee8SDavid du Colombier mountfont(int pos, char *fontname) {
40219b2ee8SDavid du Colombier 	int i;
41219b2ee8SDavid du Colombier 
42*456a8764SDavid du Colombier 	if (debug) fprint(2, "mountfont(%d, %s)\n", pos, fontname);
43219b2ee8SDavid du Colombier 	if (pos < 0 || pos >= fontmnt)
44219b2ee8SDavid du Colombier 		error(FATAL, "cannot mount a font at position %d,\n  can only mount into postions 0-%d\n",
45219b2ee8SDavid du Colombier 			pos, fontmnt-1);
46219b2ee8SDavid du Colombier 
47219b2ee8SDavid du Colombier 	i = strlen(fontname);
48219b2ee8SDavid du Colombier 	fontmtab[pos] = galloc(fontmtab[pos], i+1, "mountfont():fontmtab");
49219b2ee8SDavid du Colombier 	strcpy(fontmtab[pos], fontname);
50219b2ee8SDavid du Colombier 	if (curfontpos == pos)	curfontpos = -1;
51219b2ee8SDavid du Colombier }
52219b2ee8SDavid du Colombier 
53219b2ee8SDavid du Colombier void
settrfont(void)54219b2ee8SDavid du Colombier settrfont(void) {
55219b2ee8SDavid du Colombier 	if (curfontpos == fontpos) return;
56219b2ee8SDavid du Colombier 
57219b2ee8SDavid du Colombier 	if (fontmtab[fontpos] == 0)
58219b2ee8SDavid du Colombier 		error(FATAL, "Font at position %d was not initialized, botch!\n", fontpos);
59219b2ee8SDavid du Colombier 
60219b2ee8SDavid du Colombier 	curtrofffontid = findtfn(fontmtab[fontpos], 1);
61*456a8764SDavid du Colombier 	if (debug) fprint(2, "settrfont()-> curtrofffontid=%d\n", curtrofffontid);
62219b2ee8SDavid du Colombier 	curfontpos = fontpos;
63219b2ee8SDavid du Colombier 	if (curtrofffontid < 0) {
64219b2ee8SDavid du Colombier 		int i;
65219b2ee8SDavid du Colombier 
66219b2ee8SDavid du Colombier 		error(WARNING, "fontpos=%d\n", fontpos);
67219b2ee8SDavid du Colombier 		for (i=0; i<fontmnt; i++)
68219b2ee8SDavid du Colombier 			if (fontmtab[i] == 0)
69219b2ee8SDavid du Colombier 				error(WARNING, "fontmtab[%d]=0x0\n", i);
70219b2ee8SDavid du Colombier 			else
71219b2ee8SDavid du Colombier 				error(WARNING, "fontmtab[%d]=%s\n", i, fontmtab[i]);
72219b2ee8SDavid du Colombier 		exits("settrfont()");
73219b2ee8SDavid du Colombier 	}
74219b2ee8SDavid du Colombier }
75219b2ee8SDavid du Colombier 
76219b2ee8SDavid du Colombier void
setpsfont(int psftid,int fontsize)77219b2ee8SDavid du Colombier setpsfont(int psftid, int fontsize) {
78219b2ee8SDavid du Colombier 	if (psftid == curpostfontid && fontsize == curfontsize) return;
79219b2ee8SDavid du Colombier 	if (psftid >= pfnamcnt)
80219b2ee8SDavid du Colombier 		error(FATAL, "Postscript font index=%d used but not defined, there are only %d fonts\n",
81219b2ee8SDavid du Colombier 			psftid, pfnamcnt);
82219b2ee8SDavid du Colombier 
83219b2ee8SDavid du Colombier 	endstring();
84219b2ee8SDavid du Colombier 	if (pageon()) {
85219b2ee8SDavid du Colombier 		Bprint(Bstdout, "%d /%s f\n", fontsize, pfnafontmtab[psftid].str);
867dd7cddfSDavid du Colombier 		if ( fontheight != 0 || fontslant != 0 )
877dd7cddfSDavid du Colombier 			Bprint(Bstdout, "%d %d changefont\n", fontslant, (fontheight != 0) ? fontheight : fontsize);
88219b2ee8SDavid du Colombier 		pfnafontmtab[psftid].used = 1;
89219b2ee8SDavid du Colombier 		curpostfontid = psftid;
90219b2ee8SDavid du Colombier 		curfontsize = fontsize;
91219b2ee8SDavid du Colombier 	}
92219b2ee8SDavid du Colombier }
93219b2ee8SDavid du Colombier 
94219b2ee8SDavid du Colombier /* find index of PostScript font name in table
95219b2ee8SDavid du Colombier  * returns -1 if name is not in table
96219b2ee8SDavid du Colombier  * If insflg is not zero
97219b2ee8SDavid du Colombier  * and the name is not found in the table, insert it.
98219b2ee8SDavid du Colombier  */
99219b2ee8SDavid du Colombier int
findpfn(char * fontname,int insflg)100219b2ee8SDavid du Colombier findpfn(char *fontname, int insflg) {
101219b2ee8SDavid du Colombier 	char *tp;
102219b2ee8SDavid du Colombier 	int i;
103219b2ee8SDavid du Colombier 
104219b2ee8SDavid du Colombier 	for (i=0; i<pfnamcnt; i++) {
105219b2ee8SDavid du Colombier 		if (strcmp(pfnafontmtab[i].str, fontname) == 0)
106219b2ee8SDavid du Colombier 			return(i);
107219b2ee8SDavid du Colombier 	}
108219b2ee8SDavid du Colombier 	if (insflg) {
109219b2ee8SDavid du Colombier 		tp = galloc(pfnafontmtab, sizeof(struct pfnament)*(pfnamcnt+1), "findpfn():pfnafontmtab");
110219b2ee8SDavid du Colombier 		if (tp == 0)
111219b2ee8SDavid du Colombier 			return(-2);
112219b2ee8SDavid du Colombier 		pfnafontmtab = (struct pfnament *)tp;
113219b2ee8SDavid du Colombier 		i = strlen(fontname);
114219b2ee8SDavid du Colombier 		pfnafontmtab[pfnamcnt].str = galloc(0, i+1, "findpfn():pfnafontmtab[].str");
115219b2ee8SDavid du Colombier 		strncpy(pfnafontmtab[pfnamcnt].str, fontname, i);
116219b2ee8SDavid du Colombier 		pfnafontmtab[pfnamcnt].str[i] = '\0';
1177dd7cddfSDavid du Colombier 		pfnafontmtab[pfnamcnt].used = 0;
118219b2ee8SDavid du Colombier 		return(pfnamcnt++);
119219b2ee8SDavid du Colombier 	}
120219b2ee8SDavid du Colombier 	return(-1);
121219b2ee8SDavid du Colombier }
122219b2ee8SDavid du Colombier 
123219b2ee8SDavid du Colombier char postroffdirname[] = "/sys/lib/postscript/troff"; /* "/sys/lib/postscript/troff/"; */
124219b2ee8SDavid du Colombier char troffmetricdirname[] = "/sys/lib/troff/font"; /* "/sys/lib/troff/font/devutf/"; */
125219b2ee8SDavid du Colombier 
126219b2ee8SDavid du Colombier int
readpsfontdesc(char * fontname,int trindex)127219b2ee8SDavid du Colombier readpsfontdesc(char *fontname, int trindex) {
128*456a8764SDavid du Colombier 	int errorflg = 0, line = 1, rv, start, end, offset;
129219b2ee8SDavid du Colombier 	int startfont, endfont, startchar, endchar, i, pfid;
130219b2ee8SDavid du Colombier 	char psfontnam[128];
131219b2ee8SDavid du Colombier 	struct troffont *tp;
132*456a8764SDavid du Colombier 	Biobuf *bfd;
133*456a8764SDavid du Colombier 	Biobufhdr *Bfd;
134*456a8764SDavid du Colombier 	static char *filename = 0;
135219b2ee8SDavid du Colombier 
136*456a8764SDavid du Colombier 	if (debug)
137*456a8764SDavid du Colombier 		fprint(2, "readpsfontdesc(%s,%d)\n", fontname, trindex);
138*456a8764SDavid du Colombier 	filename = galloc(filename, strlen(postroffdirname)+1+strlen(fontname)+1,
139*456a8764SDavid du Colombier 		"readpsfontdesc: cannot allocate memory\n");
140219b2ee8SDavid du Colombier 	sprint(filename, "%s/%s", postroffdirname, fontname);
141219b2ee8SDavid du Colombier 
142219b2ee8SDavid du Colombier 	bfd = Bopen(filename, OREAD);
143219b2ee8SDavid du Colombier 	if (bfd == 0) {
144219b2ee8SDavid du Colombier 		error(WARNING, "cannot open file %s\n", filename);
145219b2ee8SDavid du Colombier 		return(0);
146219b2ee8SDavid du Colombier 	}
147*456a8764SDavid du Colombier 	Bfd = &bfd->Biobufhdr;
148219b2ee8SDavid du Colombier 
149219b2ee8SDavid du Colombier 	do {
150219b2ee8SDavid du Colombier 		offset = 0;
151219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'd', &start, 0)) == 0) {
152219b2ee8SDavid du Colombier 			errorflg = 1;
153219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal start value\n", filename, line);
154219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
155219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'd', &end, 0)) == 0) {
156219b2ee8SDavid du Colombier 			errorflg = 1;
157219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal end value\n", filename, line);
158219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
159*456a8764SDavid du Colombier 		if (Bgetfield(Bfd, 'd', &offset, 0) < 0) {
160219b2ee8SDavid du Colombier 			errorflg = 1;
161219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal offset value\n", filename, line);
162219b2ee8SDavid du Colombier 		}
163219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 's', psfontnam, 128)) == 0) {
164219b2ee8SDavid du Colombier 			errorflg = 1;
165219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal fontname value\n", filename, line);
166*456a8764SDavid du Colombier 		} else if (rv < 0)
167*456a8764SDavid du Colombier 			break;
168219b2ee8SDavid du Colombier 		Brdline(Bfd, '\n');
169219b2ee8SDavid du Colombier 		if (!errorflg) {
170219b2ee8SDavid du Colombier 			struct psfent *psfentp;
171219b2ee8SDavid du Colombier 			startfont = RUNEGETGROUP(start);
172219b2ee8SDavid du Colombier 			startchar = RUNEGETCHAR(start);
173219b2ee8SDavid du Colombier 			endfont = RUNEGETGROUP(end);
174219b2ee8SDavid du Colombier 			endchar = RUNEGETCHAR(end);
175219b2ee8SDavid du Colombier 			pfid = findpfn(psfontnam, 1);
176219b2ee8SDavid du Colombier 			if (startfont != endfont) {
177219b2ee8SDavid du Colombier 				error(WARNING, "font descriptions must not cross 256 glyph block boundary\n");
178219b2ee8SDavid du Colombier 				break;
179219b2ee8SDavid du Colombier 			}
180*456a8764SDavid du Colombier 			tp = &troffontab[trindex];
181*456a8764SDavid du Colombier 			tp->psfmap = galloc(tp->psfmap, ++tp->psfmapsize *
182*456a8764SDavid du Colombier 				sizeof(struct psfent),
183*456a8764SDavid du Colombier 				"readpsfontdesc():psfmap");
184219b2ee8SDavid du Colombier 			psfentp = &(tp->psfmap[tp->psfmapsize-1]);
185219b2ee8SDavid du Colombier 			psfentp->start = start;
186219b2ee8SDavid du Colombier 			psfentp->end = end;
187219b2ee8SDavid du Colombier 			psfentp->offset = offset;
188219b2ee8SDavid du Colombier 			psfentp->psftid = pfid;
1897dd7cddfSDavid du Colombier 			if (debug) {
190*456a8764SDavid du Colombier 				fprint(2, "\tpsfmap->start=0x%x\n", start);
191*456a8764SDavid du Colombier 				fprint(2, "\tpsfmap->end=0x%x\n", end);
192*456a8764SDavid du Colombier 				fprint(2, "\tpsfmap->offset=0x%x\n", offset);
193*456a8764SDavid du Colombier 				fprint(2, "\tpsfmap->pfid=0x%x\n", pfid);
1947dd7cddfSDavid du Colombier 			}
195*456a8764SDavid du Colombier 			SET(i);
196*456a8764SDavid du Colombier 			USED(i, startchar, endchar);
197219b2ee8SDavid du Colombier /*
198219b2ee8SDavid du Colombier 			for (i=startchar; i<=endchar; i++) {
199219b2ee8SDavid du Colombier 				tp->charent[startfont][i].postfontid = pfid;
200219b2ee8SDavid du Colombier 				tp->charent[startfont][i].postcharid = i + offset - startchar;
201219b2ee8SDavid du Colombier 			}
202219b2ee8SDavid du Colombier  */
203219b2ee8SDavid du Colombier 			if (debug) {
204*456a8764SDavid du Colombier 				fprint(2, "%x %x ", start, end);
205*456a8764SDavid du Colombier 				if (offset)
206*456a8764SDavid du Colombier 					fprint(2, "%x ", offset);
207*456a8764SDavid du Colombier 				fprint(2, "%s\n", psfontnam);
208219b2ee8SDavid du Colombier 			}
209219b2ee8SDavid du Colombier 			line++;
210219b2ee8SDavid du Colombier 		}
211219b2ee8SDavid du Colombier 	} while(errorflg != 1);
212219b2ee8SDavid du Colombier 	Bterm(Bfd);
213219b2ee8SDavid du Colombier 	return(1);
214219b2ee8SDavid du Colombier }
215219b2ee8SDavid du Colombier 
216219b2ee8SDavid du Colombier int
readtroffmetric(char * fontname,int trindex)217219b2ee8SDavid du Colombier readtroffmetric(char *fontname, int trindex) {
218*456a8764SDavid du Colombier 	int ntoken, errorflg = 0, line = 1, rv;
219*456a8764SDavid du Colombier 	int width, flag, charnum, thisfont, thischar;
220*456a8764SDavid du Colombier 	char stoken[128], *str;
221*456a8764SDavid du Colombier 	struct charent **cp;
222*456a8764SDavid du Colombier 	BOOLEAN specharflag;
223219b2ee8SDavid du Colombier 	Biobuf *bfd;
224219b2ee8SDavid du Colombier 	Biobufhdr *Bfd;
225219b2ee8SDavid du Colombier 	Rune troffchar, quote;
226*456a8764SDavid du Colombier 	static char *filename = 0;
227219b2ee8SDavid du Colombier 
228*456a8764SDavid du Colombier 	if (debug)
229*456a8764SDavid du Colombier 		fprint(2, "readtroffmetric(%s,%d)\n", fontname, trindex);
230*456a8764SDavid du Colombier 	filename = galloc(filename, strlen(troffmetricdirname) + 4 +
231*456a8764SDavid du Colombier 		strlen(devname) + 1 + strlen(fontname) + 1,
232*456a8764SDavid du Colombier 		"readtroffmetric():filename");
233219b2ee8SDavid du Colombier 	sprint(filename, "%s/dev%s/%s", troffmetricdirname, devname, fontname);
234219b2ee8SDavid du Colombier 
235219b2ee8SDavid du Colombier 	bfd = Bopen(filename, OREAD);
236219b2ee8SDavid du Colombier 	if (bfd == 0) {
237219b2ee8SDavid du Colombier 		error(WARNING, "cannot open file %s\n", filename);
238219b2ee8SDavid du Colombier 		return(0);
239219b2ee8SDavid du Colombier 	}
240*456a8764SDavid du Colombier 	Bfd = &bfd->Biobufhdr;
241219b2ee8SDavid du Colombier 	do {
242*456a8764SDavid du Colombier 		/*
243*456a8764SDavid du Colombier 		 * deal with the few lines at the beginning of the
244219b2ee8SDavid du Colombier 		 * troff font metric files.
245219b2ee8SDavid du Colombier 		 */
246219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 's', stoken, 128)) == 0) {
247219b2ee8SDavid du Colombier 			errorflg = 1;
248219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal token\n", filename, line);
249*456a8764SDavid du Colombier 		} else if (rv < 0)
250*456a8764SDavid du Colombier 			break;
251*456a8764SDavid du Colombier 		if (debug)
252*456a8764SDavid du Colombier 			fprint(2, "%s\n", stoken);
253219b2ee8SDavid du Colombier 
254219b2ee8SDavid du Colombier 		if (strcmp(stoken, "name") == 0) {
255219b2ee8SDavid du Colombier 			if ((rv=Bgetfield(Bfd, 's', stoken, 128)) == 0) {
256219b2ee8SDavid du Colombier 				errorflg = 1;
257219b2ee8SDavid du Colombier 				error(WARNING, "file %s:%d illegal token\n", filename, line);
258219b2ee8SDavid du Colombier 			} else if (rv < 0) break;
259219b2ee8SDavid du Colombier 		} else if (strcmp(stoken, "named") == 0) {
260219b2ee8SDavid du Colombier 			Brdline(Bfd, '\n');
261219b2ee8SDavid du Colombier 		} else if (strcmp(stoken, "fontname") == 0) {
262219b2ee8SDavid du Colombier 			if ((rv=Bgetfield(Bfd, 's', stoken, 128)) == 0) {
263219b2ee8SDavid du Colombier 				errorflg = 1;
264219b2ee8SDavid du Colombier 				error(WARNING, "file %s:%d illegal token\n", filename, line);
265219b2ee8SDavid du Colombier 			} else if (rv < 0) break;
266219b2ee8SDavid du Colombier 		} else if (strcmp(stoken, "spacewidth") == 0) {
267219b2ee8SDavid du Colombier 			if ((rv=Bgetfield(Bfd, 'd', &ntoken, 0)) == 0) {
268219b2ee8SDavid du Colombier 				errorflg = 1;
269219b2ee8SDavid du Colombier 				error(WARNING, "file %s:%d illegal token\n", filename, line);
270219b2ee8SDavid du Colombier 			} else if (rv < 0) break;
271219b2ee8SDavid du Colombier 			troffontab[trindex].spacewidth = ntoken;
272219b2ee8SDavid du Colombier 			thisfont = RUNEGETGROUP(' ');
273219b2ee8SDavid du Colombier 			thischar = RUNEGETCHAR(' ');
274219b2ee8SDavid du Colombier 			for (cp = &(troffontab[trindex].charent[thisfont][thischar]); *cp != 0; cp = &((*cp)->next))
275219b2ee8SDavid du Colombier 				if ((*cp)->name)
276219b2ee8SDavid du Colombier 					if  (strcmp((*cp)->name, " ") == 0)
277219b2ee8SDavid du Colombier 						break;
278219b2ee8SDavid du Colombier 
279219b2ee8SDavid du Colombier 			if (*cp == 0) *cp = galloc(0, sizeof(struct charent), "readtroffmetric:charent");
280219b2ee8SDavid du Colombier 			(*cp)->postfontid = thisfont;
281219b2ee8SDavid du Colombier 			(*cp)->postcharid = thischar;
282219b2ee8SDavid du Colombier 			(*cp)->troffcharwidth = ntoken;
283219b2ee8SDavid du Colombier 			(*cp)->name = galloc(0, 2, "readtroffmetric: char name");
284219b2ee8SDavid du Colombier 			(*cp)->next = 0;
285219b2ee8SDavid du Colombier 			strcpy((*cp)->name, " ");
286219b2ee8SDavid du Colombier 		} else if (strcmp(stoken, "special") == 0) {
287219b2ee8SDavid du Colombier 			troffontab[trindex].special = TRUE;
288219b2ee8SDavid du Colombier 		} else if (strcmp(stoken, "charset") == 0) {
289219b2ee8SDavid du Colombier 			line++;
290219b2ee8SDavid du Colombier 			break;
291219b2ee8SDavid du Colombier 		}
292219b2ee8SDavid du Colombier 		if (!errorflg) {
293219b2ee8SDavid du Colombier 			line++;
294219b2ee8SDavid du Colombier 		}
295219b2ee8SDavid du Colombier 	} while(!errorflg && rv>=0);
296219b2ee8SDavid du Colombier 	while(!errorflg && rv>=0) {
297219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 's', stoken, 128)) == 0) {
298219b2ee8SDavid du Colombier 			errorflg = 1;
299219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal rune token <0x%x> rv=%d\n", filename, line, troffchar, rv);
300219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
301219b2ee8SDavid du Colombier 		if (utflen(stoken) > 1) specharflag = TRUE;
302219b2ee8SDavid du Colombier 		else specharflag = FALSE;
303219b2ee8SDavid du Colombier 		/* if this character is a quote we have to use the previous characters info */
304219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'r', &quote, 0)) == 0) {
305219b2ee8SDavid du Colombier 			errorflg = 1;
306*456a8764SDavid du Colombier 			error(WARNING, "file %s:%d illegal width or quote token <0x%x> rv=%d\n",
307*456a8764SDavid du Colombier 				filename, line, quote, rv);
308219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
309219b2ee8SDavid du Colombier 		if (quote == '"') {
310219b2ee8SDavid du Colombier 			/* need some code here */
311219b2ee8SDavid du Colombier 			goto flush;
312*456a8764SDavid du Colombier 		} else
313219b2ee8SDavid du Colombier 			Bungetrune(Bfd);
314219b2ee8SDavid du Colombier 
315219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'd', &width, 0)) == 0) {
316219b2ee8SDavid du Colombier 			errorflg = 1;
317219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal width token <0x%x> rv=%d\n", filename, line, troffchar, rv);
318219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
319219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'd', &flag, 0)) == 0) {
320219b2ee8SDavid du Colombier 			errorflg = 1;
321219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal flag token <0x%x> rv=%d\n", filename, line, troffchar, rv);
322219b2ee8SDavid du Colombier 		} else if (rv < 0) break;
323219b2ee8SDavid du Colombier 		if ((rv=Bgetfield(Bfd, 'd', &charnum, 0)) == 0) {
324219b2ee8SDavid du Colombier 			errorflg = 1;
325219b2ee8SDavid du Colombier 			error(WARNING, "file %s:%d illegal character number token <0x%x> rv=%d\n", filename, line, troffchar, rv);
326*456a8764SDavid du Colombier 		} else if (rv < 0)
327*456a8764SDavid du Colombier 			break;
328219b2ee8SDavid du Colombier flush:
329219b2ee8SDavid du Colombier 		str = Brdline(Bfd, '\n');
330219b2ee8SDavid du Colombier 		/* stash the crap from the end of the line for debugging */
331219b2ee8SDavid du Colombier 		if (debug) {
332219b2ee8SDavid du Colombier 			if (str == 0) {
333*456a8764SDavid du Colombier 				fprint(2, "premature EOF\n");
334219b2ee8SDavid du Colombier 				return(0);
335219b2ee8SDavid du Colombier 			}
336219b2ee8SDavid du Colombier 			str[Blinelen(Bfd)-1] = '\0';
337219b2ee8SDavid du Colombier 		}
338219b2ee8SDavid du Colombier 		line++;
339219b2ee8SDavid du Colombier 		chartorune(&troffchar, stoken);
340*456a8764SDavid du Colombier 		if (specharflag)
341219b2ee8SDavid du Colombier 			if (debug)
342*456a8764SDavid du Colombier 				fprint(2, "%s %d  %d 0x%x %s # special\n",
343*456a8764SDavid du Colombier 					stoken, width, flag, charnum, str);
344219b2ee8SDavid du Colombier 		if (strcmp(stoken, "---") == 0) {
345219b2ee8SDavid du Colombier 			thisfont = RUNEGETGROUP(charnum);
346219b2ee8SDavid du Colombier 			thischar = RUNEGETCHAR(charnum);
347219b2ee8SDavid du Colombier 			stoken[0] = '\0';
348219b2ee8SDavid du Colombier 		} else {
349219b2ee8SDavid du Colombier 			thisfont = RUNEGETGROUP(troffchar);
350219b2ee8SDavid du Colombier 			thischar = RUNEGETCHAR(troffchar);
351219b2ee8SDavid du Colombier 		}
352*456a8764SDavid du Colombier 		for (cp = &troffontab[trindex].charent[thisfont][thischar];
353*456a8764SDavid du Colombier 		    *cp != 0; cp = &(*cp)->next)
354219b2ee8SDavid du Colombier 			if ((*cp)->name) {
355*456a8764SDavid du Colombier 				if (debug)
356*456a8764SDavid du Colombier 					fprint(2, "installing <%s>, found <%s>\n",
357*456a8764SDavid du Colombier 						stoken, (*cp)->name);
358219b2ee8SDavid du Colombier 				if  (strcmp((*cp)->name, stoken) == 0)
359219b2ee8SDavid du Colombier 					break;
360219b2ee8SDavid du Colombier 			}
361*456a8764SDavid du Colombier 		if (*cp == 0)
362*456a8764SDavid du Colombier 			*cp = galloc(0, sizeof(struct charent),
363*456a8764SDavid du Colombier 				"readtroffmetric:charent");
364219b2ee8SDavid du Colombier 		(*cp)->postfontid = RUNEGETGROUP(charnum);
365219b2ee8SDavid du Colombier 		(*cp)->postcharid = RUNEGETCHAR(charnum);
366219b2ee8SDavid du Colombier 		(*cp)->troffcharwidth = width;
367219b2ee8SDavid du Colombier 		(*cp)->name = galloc(0, strlen(stoken)+1, "readtroffmetric: char name");
368219b2ee8SDavid du Colombier 		(*cp)->next = 0;
369219b2ee8SDavid du Colombier 		strcpy((*cp)->name, stoken);
370219b2ee8SDavid du Colombier 		if (debug) {
371219b2ee8SDavid du Colombier 			if (specharflag)
372*456a8764SDavid du Colombier 				fprint(2, "%s", stoken);
373219b2ee8SDavid du Colombier 			else
374*456a8764SDavid du Colombier 				fprint(2, "%C", troffchar);
375*456a8764SDavid du Colombier 			fprint(2, " %d  %d 0x%x %s # psfontid=0x%x pscharid=0x%x thisfont=0x%x thischar=0x%x\n",
376219b2ee8SDavid du Colombier 				width, flag, charnum, str,
377219b2ee8SDavid du Colombier 				(*cp)->postfontid,
378219b2ee8SDavid du Colombier 				(*cp)->postcharid,
379219b2ee8SDavid du Colombier 				thisfont, thischar);
380219b2ee8SDavid du Colombier 		}
381219b2ee8SDavid du Colombier 	}
382219b2ee8SDavid du Colombier 	Bterm(Bfd);
383219b2ee8SDavid du Colombier 	Bflush(Bstderr);
384219b2ee8SDavid du Colombier 	return(1);
385219b2ee8SDavid du Colombier }
386219b2ee8SDavid du Colombier 
387219b2ee8SDavid du Colombier /* find index of troff font name in table
388219b2ee8SDavid du Colombier  * returns -1 if name is not in table
389219b2ee8SDavid du Colombier  * returns -2 if it cannot allocate memory
390219b2ee8SDavid du Colombier  * returns -3 if there is a font mapping problem
391219b2ee8SDavid du Colombier  * If insflg is not zero
392219b2ee8SDavid du Colombier  * and the name is not found in the table, insert it.
393219b2ee8SDavid du Colombier  */
394219b2ee8SDavid du Colombier int
findtfn(char * fontname,BOOLEAN insflg)395219b2ee8SDavid du Colombier findtfn(char *fontname, BOOLEAN insflg) {
396219b2ee8SDavid du Colombier 	struct troffont *tp;
397219b2ee8SDavid du Colombier 	int i, j;
398219b2ee8SDavid du Colombier 
399*456a8764SDavid du Colombier 	if (debug)
400*456a8764SDavid du Colombier 		fprint(2, "findtfn(%s,%d)\n", fontname, insflg);
401219b2ee8SDavid du Colombier 	for (i=0; i<troffontcnt; i++) {
402219b2ee8SDavid du Colombier 		if (troffontab[i].trfontid==0) {
403219b2ee8SDavid du Colombier 			error(WARNING, "findtfn:troffontab[%d].trfontid=0x%x, botch!\n",
404219b2ee8SDavid du Colombier 				i, troffontab[i].trfontid);
405219b2ee8SDavid du Colombier 			continue;
406219b2ee8SDavid du Colombier 		}
407219b2ee8SDavid du Colombier 		if (strcmp(troffontab[i].trfontid, fontname) == 0)
408219b2ee8SDavid du Colombier 			return(i);
409219b2ee8SDavid du Colombier 	}
410219b2ee8SDavid du Colombier 	if (insflg) {
411*456a8764SDavid du Colombier 		tp = (struct troffont *)galloc(troffontab,
412*456a8764SDavid du Colombier 			sizeof(struct troffont)*(troffontcnt+1),
413*456a8764SDavid du Colombier 			"findtfn: struct troffont:");
414219b2ee8SDavid du Colombier 		if (tp == 0)
415219b2ee8SDavid du Colombier 			return(-2);
416219b2ee8SDavid du Colombier 		troffontab = tp;
417219b2ee8SDavid du Colombier 		tp = &(troffontab[troffontcnt]);
418219b2ee8SDavid du Colombier 		i = strlen(fontname);
419219b2ee8SDavid du Colombier 		tp->trfontid = galloc(0, i+1, "findtfn: trfontid:");
420219b2ee8SDavid du Colombier 
421219b2ee8SDavid du Colombier 		/* initialize new troff font entry with name and numeric fields to 0 */
422219b2ee8SDavid du Colombier 		strncpy(tp->trfontid, fontname, i);
423219b2ee8SDavid du Colombier 		tp->trfontid[i] = '\0';
424219b2ee8SDavid du Colombier 		tp->special = FALSE;
425219b2ee8SDavid du Colombier 		tp->spacewidth = 0;
426219b2ee8SDavid du Colombier 		tp->psfmapsize = 0;
427219b2ee8SDavid du Colombier 		tp->psfmap = 0;
428219b2ee8SDavid du Colombier 		for (i=0; i<NUMOFONTS; i++)
429219b2ee8SDavid du Colombier 			for (j=0; j<FONTSIZE; j++)
430219b2ee8SDavid du Colombier 				tp->charent[i][j] = 0;
431219b2ee8SDavid du Colombier 		troffontcnt++;
432219b2ee8SDavid du Colombier 		if (!readtroffmetric(fontname, troffontcnt-1))
433219b2ee8SDavid du Colombier 			return(-3);
434219b2ee8SDavid du Colombier 		if (!readpsfontdesc(fontname, troffontcnt-1))
435219b2ee8SDavid du Colombier 			return(-3);
436219b2ee8SDavid du Colombier 		return(troffontcnt-1);
437219b2ee8SDavid du Colombier 	}
438219b2ee8SDavid du Colombier 	return(-1);
439219b2ee8SDavid du Colombier }
440219b2ee8SDavid du Colombier 
441219b2ee8SDavid du Colombier void
finish(void)442219b2ee8SDavid du Colombier finish(void) {
443219b2ee8SDavid du Colombier 	int i;
444219b2ee8SDavid du Colombier 
445219b2ee8SDavid du Colombier 	Bprint(Bstdout, "%s", TRAILER);
446219b2ee8SDavid du Colombier 	Bprint(Bstdout, "done\n");
447219b2ee8SDavid du Colombier 	Bprint(Bstdout, "%s", DOCUMENTFONTS);
448219b2ee8SDavid du Colombier 
449219b2ee8SDavid du Colombier 	for (i=0; i<pfnamcnt; i++)
450219b2ee8SDavid du Colombier 		if (pfnafontmtab[i].used)
451219b2ee8SDavid du Colombier 			Bprint(Bstdout, " %s", pfnafontmtab[i].str);
452219b2ee8SDavid du Colombier 	Bprint(Bstdout, "\n");
453219b2ee8SDavid du Colombier 
454219b2ee8SDavid du Colombier 	Bprint(Bstdout, "%s %d\n", PAGES, pages_printed);
455219b2ee8SDavid du Colombier 
456219b2ee8SDavid du Colombier }
4577dd7cddfSDavid du Colombier 
4587dd7cddfSDavid du Colombier /* Set slant to n degrees. Disable slanting if n is 0. */
4597dd7cddfSDavid du Colombier void
t_slant(int n)4607dd7cddfSDavid du Colombier t_slant(int n) {
4617dd7cddfSDavid du Colombier 	fontslant = n;
4627dd7cddfSDavid du Colombier 	curpostfontid = -1;
4637dd7cddfSDavid du Colombier }
4647dd7cddfSDavid du Colombier 
4657dd7cddfSDavid du Colombier /* Set character height to n points. Disabled if n is 0 or the current size. */
4667dd7cddfSDavid du Colombier 
4677dd7cddfSDavid du Colombier void
t_charht(int n)4687dd7cddfSDavid du Colombier t_charht(int n) {
4697dd7cddfSDavid du Colombier 	fontheight = (n == fontsize) ? 0 : n;
4707dd7cddfSDavid du Colombier 	curpostfontid = -1;
4717dd7cddfSDavid du Colombier }
472