xref: /plan9/sys/src/cmd/postscript/common/common.c (revision 219b2ee8daee37f4aad58d63f21287faa8e4ffdc)
1*219b2ee8SDavid du Colombier #include <u.h>
2*219b2ee8SDavid du Colombier #include <libc.h>
3*219b2ee8SDavid du Colombier #include <bio.h>
4*219b2ee8SDavid du Colombier #include <ctype.h>
5*219b2ee8SDavid du Colombier #include "common.h"
6*219b2ee8SDavid du Colombier #include "comments.h"
7*219b2ee8SDavid du Colombier #include "path.h"
8*219b2ee8SDavid du Colombier 
9*219b2ee8SDavid du Colombier struct strtab charcode[FONTSIZE] = {
10*219b2ee8SDavid du Colombier 	{4, "\\000"}, {4, "\\001"}, {4, "\\002"}, {4, "\\003"},
11*219b2ee8SDavid du Colombier 	{4, "\\004"}, {4, "\\005"}, {4, "\\006"}, {4, "\\007"},
12*219b2ee8SDavid du Colombier 	{4, "\\010"}, {4, "\\011"}, {4, "\\012"}, {4, "\\013"},
13*219b2ee8SDavid du Colombier 	{4, "\\014"}, {4, "\\015"}, {4, "\\016"}, {4, "\\017"},
14*219b2ee8SDavid du Colombier 	{4, "\\020"}, {4, "\\021"}, {4, "\\022"}, {4, "\\023"},
15*219b2ee8SDavid du Colombier 	{4, "\\024"}, {4, "\\025"}, {4, "\\026"}, {4, "\\027"},
16*219b2ee8SDavid du Colombier 	{4, "\\030"}, {4, "\\031"}, {4, "\\032"}, {4, "\\033"},
17*219b2ee8SDavid du Colombier 	{4, "\\034"}, {4, "\\035"}, {4, "\\036"}, {4, "\\037"},
18*219b2ee8SDavid du Colombier 	{1, " "}, {1, "!"}, {1, "\""}, {1, "#"},
19*219b2ee8SDavid du Colombier 	{1, "$"}, {1, "%"}, {1, "&"}, {1, "'"},
20*219b2ee8SDavid du Colombier 	{2, "\\("}, {2, "\\)"}, {1, "*"}, {1, "+"},
21*219b2ee8SDavid du Colombier 	{1, ","}, {1, "-"}, {1, "."}, {1, "/"},
22*219b2ee8SDavid du Colombier 	{1, "0"}, {1, "1"}, {1, "2"}, {1, "3"},
23*219b2ee8SDavid du Colombier 	{1, "4"}, {1, "5"}, {1, "6"}, {1, "7"},
24*219b2ee8SDavid du Colombier 	{1, "8"}, {1, "9"}, {1, ":"}, {1, ";"},
25*219b2ee8SDavid du Colombier 	{1, "<"}, {1, "="}, {1, ">"}, {1, "?"},
26*219b2ee8SDavid du Colombier 	{1, "@"}, {1, "A"}, {1, "B"}, {1, "C"},
27*219b2ee8SDavid du Colombier 	{1, "D"}, {1, "E"}, {1, "F"}, {1, "G"},
28*219b2ee8SDavid du Colombier 	{1, "H"}, {1, "I"}, {1, "J"}, {1, "K"},
29*219b2ee8SDavid du Colombier 	{1, "L"}, {1, "M"}, {1, "N"}, {1, "O"},
30*219b2ee8SDavid du Colombier 	{1, "P"}, {1, "Q"}, {1, "R"}, {1, "S"},
31*219b2ee8SDavid du Colombier 	{1, "T"}, {1, "U"}, {1, "V"}, {1, "W"},
32*219b2ee8SDavid du Colombier 	{1, "X"}, {1, "Y"}, {1, "Z"}, {1, "["},
33*219b2ee8SDavid du Colombier 	{2, "\\\\"}, {1, "]"}, {1, "^"}, {1, "_"},
34*219b2ee8SDavid du Colombier 	{1, "`"}, {1, "a"}, {1, "b"}, {1, "c"},
35*219b2ee8SDavid du Colombier 	{1, "d"}, {1, "e"}, {1, "f"}, {1, "g"},
36*219b2ee8SDavid du Colombier 	{1, "h"}, {1, "i"}, {1, "j"}, {1, "k"},
37*219b2ee8SDavid du Colombier 	{1, "l"}, {1, "m"}, {1, "n"}, {1, "o"},
38*219b2ee8SDavid du Colombier 	{1, "p"}, {1, "q"}, {1, "r"}, {1, "s"},
39*219b2ee8SDavid du Colombier 	{1, "t"}, {1, "u"}, {1, "v"}, {1, "w"},
40*219b2ee8SDavid du Colombier 	{1, "x"}, {1, "y"}, {1, "z"}, {1, "{"},
41*219b2ee8SDavid du Colombier 	{1, "|"}, {1, "}"}, {1, "~"}, {4, "\\177"},
42*219b2ee8SDavid du Colombier 	{4, "\\200"}, {4, "\\201"}, {4, "\\202"}, {4, "\\203"},
43*219b2ee8SDavid du Colombier 	{4, "\\204"}, {4, "\\205"}, {4, "\\206"}, {4, "\\207"},
44*219b2ee8SDavid du Colombier 	{4, "\\210"}, {4, "\\211"}, {4, "\\212"}, {4, "\\213"},
45*219b2ee8SDavid du Colombier 	{4, "\\214"}, {4, "\\215"}, {4, "\\216"}, {4, "\\217"},
46*219b2ee8SDavid du Colombier 	{4, "\\220"}, {4, "\\221"}, {4, "\\222"}, {4, "\\223"},
47*219b2ee8SDavid du Colombier 	{4, "\\224"}, {4, "\\225"}, {4, "\\226"}, {4, "\\227"},
48*219b2ee8SDavid du Colombier 	{4, "\\230"}, {4, "\\231"}, {4, "\\232"}, {4, "\\233"},
49*219b2ee8SDavid du Colombier 	{4, "\\234"}, {4, "\\235"}, {4, "\\236"}, {4, "\\237"},
50*219b2ee8SDavid du Colombier 	{4, "\\240"}, {4, "\\241"}, {4, "\\242"}, {4, "\\243"},
51*219b2ee8SDavid du Colombier 	{4, "\\244"}, {4, "\\245"}, {4, "\\246"}, {4, "\\247"},
52*219b2ee8SDavid du Colombier 	{4, "\\250"}, {4, "\\251"}, {4, "\\252"}, {4, "\\253"},
53*219b2ee8SDavid du Colombier 	{4, "\\254"}, {4, "\\255"}, {4, "\\256"}, {4, "\\257"},
54*219b2ee8SDavid du Colombier 	{4, "\\260"}, {4, "\\261"}, {4, "\\262"}, {4, "\\263"},
55*219b2ee8SDavid du Colombier 	{4, "\\264"}, {4, "\\265"}, {4, "\\266"}, {4, "\\267"},
56*219b2ee8SDavid du Colombier 	{4, "\\270"}, {4, "\\271"}, {4, "\\272"}, {4, "\\273"},
57*219b2ee8SDavid du Colombier 	{4, "\\274"}, {4, "\\275"}, {4, "\\276"}, {4, "\\277"},
58*219b2ee8SDavid du Colombier 	{4, "\\300"}, {4, "\\301"}, {4, "\\302"}, {4, "\\303"},
59*219b2ee8SDavid du Colombier 	{4, "\\304"}, {4, "\\305"}, {4, "\\306"}, {4, "\\307"},
60*219b2ee8SDavid du Colombier 	{4, "\\310"}, {4, "\\311"}, {4, "\\312"}, {4, "\\313"},
61*219b2ee8SDavid du Colombier 	{4, "\\314"}, {4, "\\315"}, {4, "\\316"}, {4, "\\317"},
62*219b2ee8SDavid du Colombier 	{4, "\\320"}, {4, "\\321"}, {4, "\\322"}, {4, "\\323"},
63*219b2ee8SDavid du Colombier 	{4, "\\324"}, {4, "\\325"}, {4, "\\326"}, {4, "\\327"},
64*219b2ee8SDavid du Colombier 	{4, "\\330"}, {4, "\\331"}, {4, "\\332"}, {4, "\\333"},
65*219b2ee8SDavid du Colombier 	{4, "\\334"}, {4, "\\335"}, {4, "\\336"}, {4, "\\337"},
66*219b2ee8SDavid du Colombier 	{4, "\\340"}, {4, "\\341"}, {4, "\\342"}, {4, "\\343"},
67*219b2ee8SDavid du Colombier 	{4, "\\344"}, {4, "\\345"}, {4, "\\346"}, {4, "\\347"},
68*219b2ee8SDavid du Colombier 	{4, "\\350"}, {4, "\\351"}, {4, "\\352"}, {4, "\\353"},
69*219b2ee8SDavid du Colombier 	{4, "\\354"}, {4, "\\355"}, {4, "\\356"}, {4, "\\357"},
70*219b2ee8SDavid du Colombier 	{4, "\\360"}, {4, "\\361"}, {4, "\\362"}, {4, "\\363"},
71*219b2ee8SDavid du Colombier 	{4, "\\364"}, {4, "\\365"}, {4, "\\366"}, {4, "\\367"},
72*219b2ee8SDavid du Colombier 	{4, "\\370"}, {4, "\\371"}, {4, "\\372"}, {4, "\\373"},
73*219b2ee8SDavid du Colombier 	{4, "\\374"}, {4, "\\375"}, {4, "\\376"}, {4, "\\377"}
74*219b2ee8SDavid du Colombier };
75*219b2ee8SDavid du Colombier 
76*219b2ee8SDavid du Colombier static BOOLEAN in_string = FALSE;
77*219b2ee8SDavid du Colombier int char_no = 0;
78*219b2ee8SDavid du Colombier int line_no = 0;
79*219b2ee8SDavid du Colombier int page_no = 0;		/* page number in a document */
80*219b2ee8SDavid du Colombier int pages_printed = 0;
81*219b2ee8SDavid du Colombier static int pplistmaxsize=0;
82*219b2ee8SDavid du Colombier 
83*219b2ee8SDavid du Colombier static unsigned char *pplist=0;	/* bitmap list for storing pages to print */
84*219b2ee8SDavid du Colombier 
85*219b2ee8SDavid du Colombier void
86*219b2ee8SDavid du Colombier pagelist(char *list) {
87*219b2ee8SDavid du Colombier 	char c;
88*219b2ee8SDavid du Colombier 	int n, m;
89*219b2ee8SDavid du Colombier 	int state, start;
90*219b2ee8SDavid du Colombier 
91*219b2ee8SDavid du Colombier 	if (list == 0) return;
92*219b2ee8SDavid du Colombier 	state = 1;
93*219b2ee8SDavid du Colombier 	start = 0;
94*219b2ee8SDavid du Colombier 	while ((c=*list) != '\0') {
95*219b2ee8SDavid du Colombier 		n = 0;
96*219b2ee8SDavid du Colombier 		while (isdigit(c)) {
97*219b2ee8SDavid du Colombier 			n = n * 10 + c - '0';
98*219b2ee8SDavid du Colombier 			c = *++list;
99*219b2ee8SDavid du Colombier 		}
100*219b2ee8SDavid du Colombier 		switch (state) {
101*219b2ee8SDavid du Colombier 		case 1:
102*219b2ee8SDavid du Colombier 			start = n;
103*219b2ee8SDavid du Colombier 		case 2:
104*219b2ee8SDavid du Colombier 			if (n/8+1 > pplistmaxsize) {
105*219b2ee8SDavid du Colombier 				pplistmaxsize = n/8+1;
106*219b2ee8SDavid du Colombier 				pplist = galloc(pplist, n/8+1, "page list");
107*219b2ee8SDavid du Colombier 			}
108*219b2ee8SDavid du Colombier 			for (m=start; m<=n; m++)
109*219b2ee8SDavid du Colombier 				pplist[m/8] |= 1<<(m%8);
110*219b2ee8SDavid du Colombier 			break;
111*219b2ee8SDavid du Colombier 		}
112*219b2ee8SDavid du Colombier 		switch (c) {
113*219b2ee8SDavid du Colombier 		case '-':
114*219b2ee8SDavid du Colombier 			state = 2;
115*219b2ee8SDavid du Colombier 			list++;
116*219b2ee8SDavid du Colombier 			break;
117*219b2ee8SDavid du Colombier 		case ',':
118*219b2ee8SDavid du Colombier 			state = 1;
119*219b2ee8SDavid du Colombier 			list++;
120*219b2ee8SDavid du Colombier 			break;
121*219b2ee8SDavid du Colombier 		case '\0':
122*219b2ee8SDavid du Colombier 			break;
123*219b2ee8SDavid du Colombier 		}
124*219b2ee8SDavid du Colombier 	}
125*219b2ee8SDavid du Colombier }
126*219b2ee8SDavid du Colombier 
127*219b2ee8SDavid du Colombier BOOLEAN
128*219b2ee8SDavid du Colombier pageon(void) {
129*219b2ee8SDavid du Colombier 	extern BOOLEAN debug;
130*219b2ee8SDavid du Colombier 	static BOOLEAN privdebug = FALSE;
131*219b2ee8SDavid du Colombier 
132*219b2ee8SDavid du Colombier 	if (pplist == 0 && page_no != 0) {
133*219b2ee8SDavid du Colombier 		if (privdebug && !debug) {
134*219b2ee8SDavid du Colombier 			privdebug = FALSE;
135*219b2ee8SDavid du Colombier 			debug = TRUE;
136*219b2ee8SDavid du Colombier 		}
137*219b2ee8SDavid du Colombier 		return(TRUE);	/* no page list, print all pages */
138*219b2ee8SDavid du Colombier 	}
139*219b2ee8SDavid du Colombier 	if (page_no/8 < pplistmaxsize && (pplist[page_no/8] & 1<<(page_no%8))) {
140*219b2ee8SDavid du Colombier 		if (privdebug && !debug) {
141*219b2ee8SDavid du Colombier 			privdebug = FALSE;
142*219b2ee8SDavid du Colombier 			debug = TRUE;
143*219b2ee8SDavid du Colombier 		}
144*219b2ee8SDavid du Colombier 		return(TRUE);
145*219b2ee8SDavid du Colombier 	} else {
146*219b2ee8SDavid du Colombier 		if (!privdebug && debug) {
147*219b2ee8SDavid du Colombier 			privdebug = TRUE;
148*219b2ee8SDavid du Colombier 			debug = FALSE;
149*219b2ee8SDavid du Colombier 		}
150*219b2ee8SDavid du Colombier 		return(FALSE);
151*219b2ee8SDavid du Colombier 	}
152*219b2ee8SDavid du Colombier }
153*219b2ee8SDavid du Colombier 
154*219b2ee8SDavid du Colombier static int stringhpos, stringvpos;
155*219b2ee8SDavid du Colombier 
156*219b2ee8SDavid du Colombier void
157*219b2ee8SDavid du Colombier startstring(void) {
158*219b2ee8SDavid du Colombier 	if (!in_string) {
159*219b2ee8SDavid du Colombier 		stringhpos = hpos;
160*219b2ee8SDavid du Colombier 		stringvpos = vpos;
161*219b2ee8SDavid du Colombier 		if (pageon()) Bprint(Bstdout, "(");
162*219b2ee8SDavid du Colombier 		in_string = 1;
163*219b2ee8SDavid du Colombier 	}
164*219b2ee8SDavid du Colombier }
165*219b2ee8SDavid du Colombier 
166*219b2ee8SDavid du Colombier void
167*219b2ee8SDavid du Colombier endstring(void) {
168*219b2ee8SDavid du Colombier 	if (in_string) {
169*219b2ee8SDavid du Colombier 		if (pageon()) Bprint(Bstdout, ") %d %d w\n", stringhpos, stringvpos);
170*219b2ee8SDavid du Colombier 		in_string = 0;
171*219b2ee8SDavid du Colombier 	}
172*219b2ee8SDavid du Colombier }
173*219b2ee8SDavid du Colombier 
174*219b2ee8SDavid du Colombier BOOLEAN
175*219b2ee8SDavid du Colombier isinstring(void) {
176*219b2ee8SDavid du Colombier 	return(in_string);
177*219b2ee8SDavid du Colombier }
178*219b2ee8SDavid du Colombier 
179*219b2ee8SDavid du Colombier void
180*219b2ee8SDavid du Colombier startpage(void) {
181*219b2ee8SDavid du Colombier 	++char_no;
182*219b2ee8SDavid du Colombier 	++line_no;
183*219b2ee8SDavid du Colombier 	++page_no;
184*219b2ee8SDavid du Colombier 	if (pageon()) {
185*219b2ee8SDavid du Colombier 		++pages_printed;
186*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "%s %d %d\n", PAGE, page_no, pages_printed);
187*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "/saveobj save def\n");
188*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "mark\n");
189*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "%d pagesetup\n", pages_printed);
190*219b2ee8SDavid du Colombier 	}
191*219b2ee8SDavid du Colombier }
192*219b2ee8SDavid du Colombier 
193*219b2ee8SDavid du Colombier void
194*219b2ee8SDavid du Colombier endpage(void) {
195*219b2ee8SDavid du Colombier 	endstring();
196*219b2ee8SDavid du Colombier 	curpostfontid = -1;
197*219b2ee8SDavid du Colombier 	line_no = 0;
198*219b2ee8SDavid du Colombier 	char_no = 0;
199*219b2ee8SDavid du Colombier 	if (pageon()) {
200*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "cleartomark\n");
201*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "showpage\n");
202*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "saveobj restore\n");
203*219b2ee8SDavid du Colombier 		Bprint(Bstdout, "%s %d %d\n", ENDPAGE, page_no, pages_printed);
204*219b2ee8SDavid du Colombier 	}
205*219b2ee8SDavid du Colombier }
206*219b2ee8SDavid du Colombier 
207*219b2ee8SDavid du Colombier /* This was taken from postprint */
208*219b2ee8SDavid du Colombier 
209*219b2ee8SDavid du Colombier int
210*219b2ee8SDavid du Colombier cat(char *filename) {
211*219b2ee8SDavid du Colombier 	Biobuf *bfile;
212*219b2ee8SDavid du Colombier 	Biobufhdr *Bfile;
213*219b2ee8SDavid du Colombier 	int n;
214*219b2ee8SDavid du Colombier 	static char buf[Bsize];
215*219b2ee8SDavid du Colombier 
216*219b2ee8SDavid du Colombier 	if ((bfile = Bopen(filename, OREAD)) == 0) {
217*219b2ee8SDavid du Colombier 		return(1);
218*219b2ee8SDavid du Colombier 	}
219*219b2ee8SDavid du Colombier 	Bfile = &(bfile->Biobufhdr);
220*219b2ee8SDavid du Colombier 	while ((n=Bread(Bfile, buf, Bsize)) > 0) {
221*219b2ee8SDavid du Colombier 		if (Bwrite(Bstdout, buf, n) != n)
222*219b2ee8SDavid du Colombier 			break;
223*219b2ee8SDavid du Colombier 	}
224*219b2ee8SDavid du Colombier 	Bterm(Bfile);
225*219b2ee8SDavid du Colombier 	if (n != 0) {
226*219b2ee8SDavid du Colombier 		return(1);
227*219b2ee8SDavid du Colombier 	}
228*219b2ee8SDavid du Colombier 	return(0);
229*219b2ee8SDavid du Colombier }
230*219b2ee8SDavid du Colombier extern int debug;
231*219b2ee8SDavid du Colombier void *
232*219b2ee8SDavid du Colombier galloc(void *ptr, int size, char *perstr) {
233*219b2ee8SDavid du Colombier 	void *x;
234*219b2ee8SDavid du Colombier 
235*219b2ee8SDavid du Colombier 	if ((x=realloc(ptr, size)) == 0) {
236*219b2ee8SDavid du Colombier 		perror(perstr);
237*219b2ee8SDavid du Colombier 		exits("malloc");
238*219b2ee8SDavid du Colombier 	}
239*219b2ee8SDavid du Colombier 	return(x);
240*219b2ee8SDavid du Colombier }
241*219b2ee8SDavid du Colombier 
242*219b2ee8SDavid du Colombier static char *errorstrings[] = {
243*219b2ee8SDavid du Colombier 	{""},	/* NONE */
244*219b2ee8SDavid du Colombier 	{"WARNING"},
245*219b2ee8SDavid du Colombier 	{"FATAL"}
246*219b2ee8SDavid du Colombier };
247*219b2ee8SDavid du Colombier 
248*219b2ee8SDavid du Colombier char *programname;
249*219b2ee8SDavid du Colombier char *inputfilename = "<stdin>";
250*219b2ee8SDavid du Colombier int inputlineno;
251*219b2ee8SDavid du Colombier 
252*219b2ee8SDavid du Colombier void
253*219b2ee8SDavid du Colombier error(int errtype, char *fmt, ...) {
254*219b2ee8SDavid du Colombier 	char buf[1024], *out;
255*219b2ee8SDavid du Colombier 
256*219b2ee8SDavid du Colombier 	Bflush(Bstdout);
257*219b2ee8SDavid du Colombier 	Bflush(Bstderr);
258*219b2ee8SDavid du Colombier 	fprint(2, "%s: %s:%d :%s: ", programname, inputfilename, inputlineno, errorstrings[errtype]);
259*219b2ee8SDavid du Colombier 	out = doprint(buf, buf+sizeof(buf), fmt, (&fmt+1));
260*219b2ee8SDavid du Colombier 	write(2, buf, out-buf);
261*219b2ee8SDavid du Colombier 	if (errtype == FATAL)
262*219b2ee8SDavid du Colombier 		exits("fatal error");
263*219b2ee8SDavid du Colombier }
264