1219b2ee8SDavid du Colombier /*
2219b2ee8SDavid du Colombier * postprint - PostScript translator for ASCII files.
3219b2ee8SDavid du Colombier *
4219b2ee8SDavid du Colombier * A simple program that translates ASCII files into PostScript. All it really
5219b2ee8SDavid du Colombier * does is expand tabs and backspaces, handle character quoting, print text lines,
6219b2ee8SDavid du Colombier * and control when pages are started based on the requested number of lines per
7219b2ee8SDavid du Colombier * page.
8219b2ee8SDavid du Colombier *
9219b2ee8SDavid du Colombier * The PostScript prologue is copied from *prologue before any of the input files
10219b2ee8SDavid du Colombier * are translated. The program expects that the following procedures are defined
11219b2ee8SDavid du Colombier * in that file:
12219b2ee8SDavid du Colombier *
13219b2ee8SDavid du Colombier * setup
14219b2ee8SDavid du Colombier *
15219b2ee8SDavid du Colombier * mark ... setup -
16219b2ee8SDavid du Colombier *
17219b2ee8SDavid du Colombier * Handles special initialization stuff that depends on how the program
18219b2ee8SDavid du Colombier * was called. Expects to find a mark followed by key/value pairs on the
19219b2ee8SDavid du Colombier * stack. The def operator is applied to each pair up to the mark, then
20219b2ee8SDavid du Colombier * the default state is set up.
21219b2ee8SDavid du Colombier *
22219b2ee8SDavid du Colombier * pagesetup
23219b2ee8SDavid du Colombier *
24219b2ee8SDavid du Colombier * page pagesetup -
25219b2ee8SDavid du Colombier *
26219b2ee8SDavid du Colombier * Does whatever is needed to set things up for the next page. Expects
27219b2ee8SDavid du Colombier * to find the current page number on the stack.
28219b2ee8SDavid du Colombier *
29219b2ee8SDavid du Colombier * l
30219b2ee8SDavid du Colombier *
31219b2ee8SDavid du Colombier * string l -
32219b2ee8SDavid du Colombier *
33219b2ee8SDavid du Colombier * Prints string starting in the first column and then goes to the next
34219b2ee8SDavid du Colombier * line.
35219b2ee8SDavid du Colombier *
36219b2ee8SDavid du Colombier * L
37219b2ee8SDavid du Colombier *
38219b2ee8SDavid du Colombier * mark string column string column ... L mark
39219b2ee8SDavid du Colombier *
40219b2ee8SDavid du Colombier * Prints each string on the stack starting at the horizontal position
41219b2ee8SDavid du Colombier * selected by column. Used when tabs and spaces can be sufficiently well
42219b2ee8SDavid du Colombier * compressed to make the printer overhead worthwhile. Always used when
43219b2ee8SDavid du Colombier * we have to back up.
44219b2ee8SDavid du Colombier *
45219b2ee8SDavid du Colombier * LL
46219b2ee8SDavid du Colombier *
47219b2ee8SDavid du Colombier * mark string column string column ... LL mark
48219b2ee8SDavid du Colombier *
49219b2ee8SDavid du Colombier * Like L, but only used to prevent potential PostScript stack overflow
50219b2ee8SDavid du Colombier * from too many string/column pairs. Stays on the current line. It will
51219b2ee8SDavid du Colombier * not be needed often!!
52219b2ee8SDavid du Colombier *
53219b2ee8SDavid du Colombier * done
54219b2ee8SDavid du Colombier *
55219b2ee8SDavid du Colombier * done
56219b2ee8SDavid du Colombier *
57219b2ee8SDavid du Colombier * Makes sure the last page is printed. Only needed when we're printing
58219b2ee8SDavid du Colombier * more than one page on each sheet of paper.
59219b2ee8SDavid du Colombier *
60219b2ee8SDavid du Colombier * Almost everything has been changed in this version of postprint. The program
61219b2ee8SDavid du Colombier * is more intelligent, especially about tabs, spaces, and backspacing, and as a
62219b2ee8SDavid du Colombier * result output files usually print faster. Output files also now conform to
63219b2ee8SDavid du Colombier * Adobe's file structuring conventions, which is undoubtedly something I should
64219b2ee8SDavid du Colombier * have done in the first version of the program. If the number of lines per page
65219b2ee8SDavid du Colombier * is set to 0, which can be done using the -l option, pointsize will be used to
66219b2ee8SDavid du Colombier * guess a reasonable value. The estimate is based on the values of LINESPP,
67219b2ee8SDavid du Colombier * POINTSIZE, and pointsize, and assumes LINESPP lines would fit on a page if
68219b2ee8SDavid du Colombier * we printed in size POINTSIZE. Selecting a point size using the -s option and
69219b2ee8SDavid du Colombier * adding -l0 to the command line forces the guess to be made.
70219b2ee8SDavid du Colombier *
71219b2ee8SDavid du Colombier * Many default values, like the magnification and orientation, are defined in
72219b2ee8SDavid du Colombier * the prologue, which is where they belong. If they're changed (by options), an
73219b2ee8SDavid du Colombier * appropriate definition is made after the prologue is added to the output file.
74219b2ee8SDavid du Colombier * The -P option passes arbitrary PostScript through to the output file. Among
75219b2ee8SDavid du Colombier * other things it can be used to set (or change) values that can't be accessed by
76219b2ee8SDavid du Colombier * other options.
77219b2ee8SDavid du Colombier */
78219b2ee8SDavid du Colombier
79219b2ee8SDavid du Colombier #include <stdio.h>
80*14f51593SDavid du Colombier #include <stdlib.h>
81*14f51593SDavid du Colombier #include <string.h>
82219b2ee8SDavid du Colombier #include <signal.h>
83219b2ee8SDavid du Colombier #include <ctype.h>
84219b2ee8SDavid du Colombier #ifdef plan9
85219b2ee8SDavid du Colombier #define isascii(c) ((unsigned char)(c)<=0177)
86219b2ee8SDavid du Colombier #endif
87219b2ee8SDavid du Colombier #include <sys/types.h>
88219b2ee8SDavid du Colombier #include <fcntl.h>
89219b2ee8SDavid du Colombier
90219b2ee8SDavid du Colombier #include "comments.h" /* PostScript file structuring comments */
91219b2ee8SDavid du Colombier #include "gen.h" /* general purpose definitions */
92219b2ee8SDavid du Colombier #include "path.h" /* for the prologue */
93219b2ee8SDavid du Colombier #include "ext.h" /* external variable declarations */
94219b2ee8SDavid du Colombier #include "postprint.h" /* a few special definitions */
95219b2ee8SDavid du Colombier
96219b2ee8SDavid du Colombier char *optnames = "a:c:ef:l:m:n:o:p:r:s:t:x:y:A:C:E:J:L:P:R:DI";
97219b2ee8SDavid du Colombier
98219b2ee8SDavid du Colombier char *prologue = POSTPRINT; /* default PostScript prologue */
99219b2ee8SDavid du Colombier char *formfile = FORMFILE; /* stuff for multiple pages per sheet */
100219b2ee8SDavid du Colombier
101219b2ee8SDavid du Colombier int formsperpage = 1; /* page images on each piece of paper */
102219b2ee8SDavid du Colombier int copies = 1; /* and this many copies of each sheet */
103219b2ee8SDavid du Colombier
104219b2ee8SDavid du Colombier int linespp = LINESPP; /* number of lines per page */
105219b2ee8SDavid du Colombier int pointsize = POINTSIZE; /* in this point size */
106219b2ee8SDavid du Colombier int tabstops = TABSTOPS; /* tabs set at these columns */
107219b2ee8SDavid du Colombier int crmode = 0; /* carriage return mode - 0, 1, or 2 */
108219b2ee8SDavid du Colombier int extended = TRUE; /* use escapes for unprintable chars */
109219b2ee8SDavid du Colombier
110219b2ee8SDavid du Colombier int col = 1; /* next character goes in this column */
111219b2ee8SDavid du Colombier int line = 1; /* on this line */
112219b2ee8SDavid du Colombier
113219b2ee8SDavid du Colombier int stringcount = 0; /* number of strings on the stack */
114219b2ee8SDavid du Colombier int stringstart = 1; /* column where current one starts */
115219b2ee8SDavid du Colombier
116219b2ee8SDavid du Colombier Fontmap fontmap[] = FONTMAP; /* for translating font names */
117219b2ee8SDavid du Colombier char *fontname = "Courier"; /* use this PostScript font */
118219b2ee8SDavid du Colombier
119219b2ee8SDavid du Colombier int page = 0; /* page we're working on */
120219b2ee8SDavid du Colombier int printed = 0; /* printed this many pages */
121219b2ee8SDavid du Colombier
122219b2ee8SDavid du Colombier FILE *fp_in = stdin; /* read from this file */
123219b2ee8SDavid du Colombier FILE *fp_out = stdout; /* and write stuff here */
124219b2ee8SDavid du Colombier FILE *fp_acct = NULL; /* for accounting data */
125219b2ee8SDavid du Colombier
126219b2ee8SDavid du Colombier /*****************************************************************************/
127219b2ee8SDavid du Colombier
main(agc,agv)128219b2ee8SDavid du Colombier main(agc, agv)
129219b2ee8SDavid du Colombier
130219b2ee8SDavid du Colombier int agc;
131219b2ee8SDavid du Colombier char *agv[];
132219b2ee8SDavid du Colombier
133219b2ee8SDavid du Colombier {
134219b2ee8SDavid du Colombier
135219b2ee8SDavid du Colombier /*
136219b2ee8SDavid du Colombier *
137219b2ee8SDavid du Colombier * A simple program that translates ASCII files into PostScript. If there's more
138219b2ee8SDavid du Colombier * than one input file, each begins on a new page.
139219b2ee8SDavid du Colombier *
140219b2ee8SDavid du Colombier */
141219b2ee8SDavid du Colombier
142219b2ee8SDavid du Colombier argc = agc; /* other routines may want them */
143219b2ee8SDavid du Colombier argv = agv;
144219b2ee8SDavid du Colombier
145219b2ee8SDavid du Colombier prog_name = argv[0]; /* really just for error messages */
146219b2ee8SDavid du Colombier
147219b2ee8SDavid du Colombier init_signals(); /* sets up interrupt handling */
148219b2ee8SDavid du Colombier header(); /* PostScript header and prologue */
149219b2ee8SDavid du Colombier options(); /* handle the command line options */
150219b2ee8SDavid du Colombier setup(); /* for PostScript */
151219b2ee8SDavid du Colombier arguments(); /* followed by each input file */
152219b2ee8SDavid du Colombier done(); /* print the last page etc. */
153219b2ee8SDavid du Colombier account(); /* job accounting data */
154219b2ee8SDavid du Colombier
155219b2ee8SDavid du Colombier exit(x_stat); /* not much could be wrong */
156219b2ee8SDavid du Colombier
157219b2ee8SDavid du Colombier } /* End of main */
158219b2ee8SDavid du Colombier
159219b2ee8SDavid du Colombier /*****************************************************************************/
160219b2ee8SDavid du Colombier
init_signals()161219b2ee8SDavid du Colombier init_signals()
162219b2ee8SDavid du Colombier
163219b2ee8SDavid du Colombier {
164219b2ee8SDavid du Colombier
165219b2ee8SDavid du Colombier /*
166219b2ee8SDavid du Colombier *
167219b2ee8SDavid du Colombier * Makes sure we handle interrupts.
168219b2ee8SDavid du Colombier *
169219b2ee8SDavid du Colombier */
170219b2ee8SDavid du Colombier
171219b2ee8SDavid du Colombier if ( signal(SIGINT, interrupt) == SIG_IGN ) {
172219b2ee8SDavid du Colombier signal(SIGINT, SIG_IGN);
173219b2ee8SDavid du Colombier signal(SIGQUIT, SIG_IGN);
174219b2ee8SDavid du Colombier signal(SIGHUP, SIG_IGN);
175219b2ee8SDavid du Colombier } else {
176219b2ee8SDavid du Colombier signal(SIGHUP, interrupt);
177219b2ee8SDavid du Colombier signal(SIGQUIT, interrupt);
178219b2ee8SDavid du Colombier } /* End else */
179219b2ee8SDavid du Colombier
180219b2ee8SDavid du Colombier signal(SIGTERM, interrupt);
181219b2ee8SDavid du Colombier
182219b2ee8SDavid du Colombier } /* End of init_signals */
183219b2ee8SDavid du Colombier
184219b2ee8SDavid du Colombier /*****************************************************************************/
185219b2ee8SDavid du Colombier
header()186219b2ee8SDavid du Colombier header()
187219b2ee8SDavid du Colombier
188219b2ee8SDavid du Colombier {
189219b2ee8SDavid du Colombier
190219b2ee8SDavid du Colombier int ch; /* return value from getopt() */
191219b2ee8SDavid du Colombier int old_optind = optind; /* for restoring optind - should be 1 */
192219b2ee8SDavid du Colombier
193219b2ee8SDavid du Colombier /*
194219b2ee8SDavid du Colombier *
195219b2ee8SDavid du Colombier * Scans the option list looking for things, like the prologue file, that we need
196219b2ee8SDavid du Colombier * right away but could be changed from the default. Doing things this way is an
197219b2ee8SDavid du Colombier * attempt to conform to Adobe's latest file structuring conventions. In particular
198219b2ee8SDavid du Colombier * they now say there should be nothing executed in the prologue, and they have
199219b2ee8SDavid du Colombier * added two new comments that delimit global initialization calls. Once we know
200219b2ee8SDavid du Colombier * where things really are we write out the job header, follow it by the prologue,
201219b2ee8SDavid du Colombier * and then add the ENDPROLOG and BEGINSETUP comments.
202219b2ee8SDavid du Colombier *
203219b2ee8SDavid du Colombier */
204219b2ee8SDavid du Colombier
205219b2ee8SDavid du Colombier while ( (ch = getopt(argc, argv, optnames)) != EOF )
206219b2ee8SDavid du Colombier if ( ch == 'L' )
207219b2ee8SDavid du Colombier prologue = optarg;
208219b2ee8SDavid du Colombier else if ( ch == '?' )
209219b2ee8SDavid du Colombier error(FATAL, "");
210219b2ee8SDavid du Colombier
211219b2ee8SDavid du Colombier optind = old_optind; /* get ready for option scanning */
212219b2ee8SDavid du Colombier
213219b2ee8SDavid du Colombier fprintf(stdout, "%s", CONFORMING);
214219b2ee8SDavid du Colombier fprintf(stdout, "%s %s\n", VERSION, PROGRAMVERSION);
215219b2ee8SDavid du Colombier fprintf(stdout, "%s %s\n", DOCUMENTFONTS, ATEND);
216219b2ee8SDavid du Colombier fprintf(stdout, "%s %s\n", PAGES, ATEND);
217219b2ee8SDavid du Colombier fprintf(stdout, "%s", ENDCOMMENTS);
218219b2ee8SDavid du Colombier
219219b2ee8SDavid du Colombier if ( cat(prologue) == FALSE )
220219b2ee8SDavid du Colombier error(FATAL, "can't read %s", prologue);
221219b2ee8SDavid du Colombier
222219b2ee8SDavid du Colombier if ( DOROUND )
223219b2ee8SDavid du Colombier cat(ROUNDPAGE);
224219b2ee8SDavid du Colombier
225219b2ee8SDavid du Colombier fprintf(stdout, "%s", ENDPROLOG);
226219b2ee8SDavid du Colombier fprintf(stdout, "%s", BEGINSETUP);
227219b2ee8SDavid du Colombier fprintf(stdout, "mark\n");
228219b2ee8SDavid du Colombier
229219b2ee8SDavid du Colombier } /* End of header */
230219b2ee8SDavid du Colombier
231219b2ee8SDavid du Colombier /*****************************************************************************/
232219b2ee8SDavid du Colombier
options()233219b2ee8SDavid du Colombier options()
234219b2ee8SDavid du Colombier
235219b2ee8SDavid du Colombier {
236219b2ee8SDavid du Colombier
237219b2ee8SDavid du Colombier int ch; /* return value from getopt() */
238219b2ee8SDavid du Colombier
239219b2ee8SDavid du Colombier /*
240219b2ee8SDavid du Colombier *
241219b2ee8SDavid du Colombier * Reads and processes the command line options. Added the -P option so arbitrary
242219b2ee8SDavid du Colombier * PostScript code can be passed through. Expect it could be useful for changing
243219b2ee8SDavid du Colombier * definitions in the prologue for which options have not been defined.
244219b2ee8SDavid du Colombier *
245219b2ee8SDavid du Colombier * Although any PostScript font can be used, things will only work well for
246219b2ee8SDavid du Colombier * constant width fonts.
247219b2ee8SDavid du Colombier *
248219b2ee8SDavid du Colombier */
249219b2ee8SDavid du Colombier
250219b2ee8SDavid du Colombier while ( (ch = getopt(argc, argv, optnames)) != EOF ) {
251219b2ee8SDavid du Colombier switch ( ch ) {
252219b2ee8SDavid du Colombier
253219b2ee8SDavid du Colombier case 'a': /* aspect ratio */
254219b2ee8SDavid du Colombier fprintf(stdout, "/aspectratio %s def\n", optarg);
255219b2ee8SDavid du Colombier break;
256219b2ee8SDavid du Colombier
257219b2ee8SDavid du Colombier case 'c': /* copies */
258219b2ee8SDavid du Colombier copies = atoi(optarg);
259219b2ee8SDavid du Colombier fprintf(stdout, "/#copies %s store\n", optarg);
260219b2ee8SDavid du Colombier break;
261219b2ee8SDavid du Colombier
262219b2ee8SDavid du Colombier case 'e': /* obsolete - it's now always on */
263219b2ee8SDavid du Colombier extended = TRUE;
264219b2ee8SDavid du Colombier break;
265219b2ee8SDavid du Colombier
266219b2ee8SDavid du Colombier case 'f': /* use this PostScript font */
267219b2ee8SDavid du Colombier fontname = get_font(optarg);
268219b2ee8SDavid du Colombier fprintf(stdout, "/font /%s def\n", fontname);
269219b2ee8SDavid du Colombier break;
270219b2ee8SDavid du Colombier
271219b2ee8SDavid du Colombier case 'l': /* lines per page */
272219b2ee8SDavid du Colombier linespp = atoi(optarg);
273219b2ee8SDavid du Colombier break;
274219b2ee8SDavid du Colombier
275219b2ee8SDavid du Colombier case 'm': /* magnification */
276219b2ee8SDavid du Colombier fprintf(stdout, "/magnification %s def\n", optarg);
277219b2ee8SDavid du Colombier break;
278219b2ee8SDavid du Colombier
279219b2ee8SDavid du Colombier case 'n': /* forms per page */
280219b2ee8SDavid du Colombier formsperpage = atoi(optarg);
281219b2ee8SDavid du Colombier fprintf(stdout, "%s %s\n", FORMSPERPAGE, optarg);
282219b2ee8SDavid du Colombier fprintf(stdout, "/formsperpage %s def\n", optarg);
283219b2ee8SDavid du Colombier break;
284219b2ee8SDavid du Colombier
285219b2ee8SDavid du Colombier case 'o': /* output page list */
286219b2ee8SDavid du Colombier out_list(optarg);
287219b2ee8SDavid du Colombier break;
288219b2ee8SDavid du Colombier
289219b2ee8SDavid du Colombier case 'p': /* landscape or portrait mode */
290219b2ee8SDavid du Colombier if ( *optarg == 'l' )
291219b2ee8SDavid du Colombier fprintf(stdout, "/landscape true def\n");
292219b2ee8SDavid du Colombier else fprintf(stdout, "/landscape false def\n");
293219b2ee8SDavid du Colombier break;
294219b2ee8SDavid du Colombier
295219b2ee8SDavid du Colombier case 'r': /* carriage return mode */
296219b2ee8SDavid du Colombier crmode = atoi(optarg);
297219b2ee8SDavid du Colombier break;
298219b2ee8SDavid du Colombier
299219b2ee8SDavid du Colombier case 's': /* point size */
300219b2ee8SDavid du Colombier pointsize = atoi(optarg);
301219b2ee8SDavid du Colombier fprintf(stdout, "/pointsize %s def\n", optarg);
302219b2ee8SDavid du Colombier break;
303219b2ee8SDavid du Colombier
304219b2ee8SDavid du Colombier case 't': /* tabstops */
305219b2ee8SDavid du Colombier tabstops = atoi(optarg);
306219b2ee8SDavid du Colombier break;
307219b2ee8SDavid du Colombier
308219b2ee8SDavid du Colombier case 'x': /* shift things horizontally */
309219b2ee8SDavid du Colombier fprintf(stdout, "/xoffset %s def\n", optarg);
310219b2ee8SDavid du Colombier break;
311219b2ee8SDavid du Colombier
312219b2ee8SDavid du Colombier case 'y': /* and vertically on the page */
313219b2ee8SDavid du Colombier fprintf(stdout, "/yoffset %s def\n", optarg);
314219b2ee8SDavid du Colombier break;
315219b2ee8SDavid du Colombier
316219b2ee8SDavid du Colombier case 'A': /* force job accounting */
317219b2ee8SDavid du Colombier case 'J':
318219b2ee8SDavid du Colombier if ( (fp_acct = fopen(optarg, "a")) == NULL )
319219b2ee8SDavid du Colombier error(FATAL, "can't open accounting file %s", optarg);
320219b2ee8SDavid du Colombier break;
321219b2ee8SDavid du Colombier
322219b2ee8SDavid du Colombier case 'C': /* copy file straight to output */
323219b2ee8SDavid du Colombier if ( cat(optarg) == FALSE )
324219b2ee8SDavid du Colombier error(FATAL, "can't read %s", optarg);
325219b2ee8SDavid du Colombier break;
326219b2ee8SDavid du Colombier
327219b2ee8SDavid du Colombier case 'E': /* text font encoding */
328219b2ee8SDavid du Colombier fontencoding = optarg;
329219b2ee8SDavid du Colombier break;
330219b2ee8SDavid du Colombier
331219b2ee8SDavid du Colombier case 'L': /* PostScript prologue file */
332219b2ee8SDavid du Colombier prologue = optarg;
333219b2ee8SDavid du Colombier break;
334219b2ee8SDavid du Colombier
335219b2ee8SDavid du Colombier case 'P': /* PostScript pass through */
336219b2ee8SDavid du Colombier fprintf(stdout, "%s\n", optarg);
337219b2ee8SDavid du Colombier break;
338219b2ee8SDavid du Colombier
339219b2ee8SDavid du Colombier case 'R': /* special global or page level request */
340219b2ee8SDavid du Colombier saverequest(optarg);
341219b2ee8SDavid du Colombier break;
342219b2ee8SDavid du Colombier
343219b2ee8SDavid du Colombier case 'D': /* debug flag */
344219b2ee8SDavid du Colombier debug = ON;
345219b2ee8SDavid du Colombier break;
346219b2ee8SDavid du Colombier
347219b2ee8SDavid du Colombier case 'I': /* ignore FATAL errors */
348219b2ee8SDavid du Colombier ignore = ON;
349219b2ee8SDavid du Colombier break;
350219b2ee8SDavid du Colombier
351219b2ee8SDavid du Colombier case '?': /* don't understand the option */
352219b2ee8SDavid du Colombier error(FATAL, "");
353219b2ee8SDavid du Colombier break;
354219b2ee8SDavid du Colombier
355219b2ee8SDavid du Colombier default: /* don't know what to do for ch */
356219b2ee8SDavid du Colombier error(FATAL, "missing case for option %c\n", ch);
357219b2ee8SDavid du Colombier break;
358219b2ee8SDavid du Colombier } /* End switch */
359219b2ee8SDavid du Colombier } /* End while */
360219b2ee8SDavid du Colombier
361219b2ee8SDavid du Colombier argc -= optind; /* get ready for non-option args */
362219b2ee8SDavid du Colombier argv += optind;
363219b2ee8SDavid du Colombier
364219b2ee8SDavid du Colombier } /* End of options */
365219b2ee8SDavid du Colombier
366219b2ee8SDavid du Colombier /*****************************************************************************/
367219b2ee8SDavid du Colombier
get_font(name)368219b2ee8SDavid du Colombier char *get_font(name)
369219b2ee8SDavid du Colombier
370219b2ee8SDavid du Colombier char *name; /* name the user asked for */
371219b2ee8SDavid du Colombier
372219b2ee8SDavid du Colombier {
373219b2ee8SDavid du Colombier
374219b2ee8SDavid du Colombier int i; /* for looking through fontmap[] */
375219b2ee8SDavid du Colombier
376219b2ee8SDavid du Colombier /*
377219b2ee8SDavid du Colombier *
378219b2ee8SDavid du Colombier * Called from options() to map a user's font name into a legal PostScript name.
379219b2ee8SDavid du Colombier * If the lookup fails *name is returned to the caller. That should let you choose
380219b2ee8SDavid du Colombier * any PostScript font, although things will only work well for constant width
381219b2ee8SDavid du Colombier * fonts.
382219b2ee8SDavid du Colombier *
383219b2ee8SDavid du Colombier */
384219b2ee8SDavid du Colombier
385219b2ee8SDavid du Colombier for ( i = 0; fontmap[i].name != NULL; i++ )
386219b2ee8SDavid du Colombier if ( strcmp(name, fontmap[i].name) == 0 )
387219b2ee8SDavid du Colombier return(fontmap[i].val);
388219b2ee8SDavid du Colombier
389219b2ee8SDavid du Colombier return(name);
390219b2ee8SDavid du Colombier
391219b2ee8SDavid du Colombier } /* End of get_font */
392219b2ee8SDavid du Colombier
393219b2ee8SDavid du Colombier /*****************************************************************************/
394219b2ee8SDavid du Colombier
setup()395219b2ee8SDavid du Colombier setup()
396219b2ee8SDavid du Colombier
397219b2ee8SDavid du Colombier {
398219b2ee8SDavid du Colombier
399219b2ee8SDavid du Colombier /*
400219b2ee8SDavid du Colombier *
401219b2ee8SDavid du Colombier * Handles things that must be done after the options are read but before the
402219b2ee8SDavid du Colombier * input files are processed. linespp (lines per page) can be set using the -l
403219b2ee8SDavid du Colombier * option. If it's not positive we calculate a reasonable value using the
404219b2ee8SDavid du Colombier * requested point size - assuming LINESPP lines fit on a page in point size
405219b2ee8SDavid du Colombier * POINTSIZE.
406219b2ee8SDavid du Colombier *
407219b2ee8SDavid du Colombier */
408219b2ee8SDavid du Colombier
409219b2ee8SDavid du Colombier writerequest(0, stdout); /* global requests eg. manual feed */
410219b2ee8SDavid du Colombier setencoding(fontencoding);
411219b2ee8SDavid du Colombier fprintf(stdout, "setup\n");
412219b2ee8SDavid du Colombier
413219b2ee8SDavid du Colombier if ( formsperpage > 1 ) {
414219b2ee8SDavid du Colombier if ( cat(formfile) == FALSE )
415219b2ee8SDavid du Colombier error(FATAL, "can't read %s", formfile);
416219b2ee8SDavid du Colombier fprintf(stdout, "%d setupforms\n", formsperpage);
417219b2ee8SDavid du Colombier } /* End if */
418219b2ee8SDavid du Colombier
419219b2ee8SDavid du Colombier fprintf(stdout, "%s", ENDSETUP);
420219b2ee8SDavid du Colombier
421219b2ee8SDavid du Colombier if ( linespp <= 0 )
422219b2ee8SDavid du Colombier linespp = LINESPP * POINTSIZE / pointsize;
423219b2ee8SDavid du Colombier
424219b2ee8SDavid du Colombier } /* End of setup */
425219b2ee8SDavid du Colombier
426219b2ee8SDavid du Colombier /*****************************************************************************/
427219b2ee8SDavid du Colombier
arguments()428219b2ee8SDavid du Colombier arguments()
429219b2ee8SDavid du Colombier
430219b2ee8SDavid du Colombier {
431219b2ee8SDavid du Colombier
432219b2ee8SDavid du Colombier /*
433219b2ee8SDavid du Colombier *
434219b2ee8SDavid du Colombier * Makes sure all the non-option command line arguments are processed. If we get
435219b2ee8SDavid du Colombier * here and there aren't any arguments left, or if '-' is one of the input files
436219b2ee8SDavid du Colombier * we'll translate stdin.
437219b2ee8SDavid du Colombier *
438219b2ee8SDavid du Colombier */
439219b2ee8SDavid du Colombier
440219b2ee8SDavid du Colombier if ( argc < 1 )
441219b2ee8SDavid du Colombier text();
442219b2ee8SDavid du Colombier else { /* at least one argument is left */
443219b2ee8SDavid du Colombier while ( argc > 0 ) {
444219b2ee8SDavid du Colombier if ( strcmp(*argv, "-") == 0 )
445219b2ee8SDavid du Colombier fp_in = stdin;
446219b2ee8SDavid du Colombier else if ( (fp_in = fopen(*argv, "r")) == NULL )
447219b2ee8SDavid du Colombier error(FATAL, "can't open %s", *argv);
448219b2ee8SDavid du Colombier text();
449219b2ee8SDavid du Colombier if ( fp_in != stdin )
450219b2ee8SDavid du Colombier fclose(fp_in);
451219b2ee8SDavid du Colombier argc--;
452219b2ee8SDavid du Colombier argv++;
453219b2ee8SDavid du Colombier } /* End while */
454219b2ee8SDavid du Colombier } /* End else */
455219b2ee8SDavid du Colombier
456219b2ee8SDavid du Colombier } /* End of arguments */
457219b2ee8SDavid du Colombier
458219b2ee8SDavid du Colombier /*****************************************************************************/
459219b2ee8SDavid du Colombier
done()460219b2ee8SDavid du Colombier done()
461219b2ee8SDavid du Colombier
462219b2ee8SDavid du Colombier {
463219b2ee8SDavid du Colombier
464219b2ee8SDavid du Colombier /*
465219b2ee8SDavid du Colombier *
466219b2ee8SDavid du Colombier * Finished with all the input files, so mark the end of the pages with a TRAILER
467219b2ee8SDavid du Colombier * comment, make sure the last page prints, and add things like the PAGES comment
468219b2ee8SDavid du Colombier * that can only be determined after all the input files have been read.
469219b2ee8SDavid du Colombier *
470219b2ee8SDavid du Colombier */
471219b2ee8SDavid du Colombier
472219b2ee8SDavid du Colombier fprintf(stdout, "%s", TRAILER);
473219b2ee8SDavid du Colombier fprintf(stdout, "done\n");
474219b2ee8SDavid du Colombier fprintf(stdout, "%s %s\n", DOCUMENTFONTS, fontname);
475219b2ee8SDavid du Colombier fprintf(stdout, "%s %d\n", PAGES, printed);
476219b2ee8SDavid du Colombier
477219b2ee8SDavid du Colombier } /* End of done */
478219b2ee8SDavid du Colombier
479219b2ee8SDavid du Colombier /*****************************************************************************/
480219b2ee8SDavid du Colombier
account()481219b2ee8SDavid du Colombier account()
482219b2ee8SDavid du Colombier
483219b2ee8SDavid du Colombier {
484219b2ee8SDavid du Colombier
485219b2ee8SDavid du Colombier /*
486219b2ee8SDavid du Colombier *
487219b2ee8SDavid du Colombier * Writes an accounting record to *fp_acct provided it's not NULL. Accounting is
488219b2ee8SDavid du Colombier * requested using the -A or -J options.
489219b2ee8SDavid du Colombier *
490219b2ee8SDavid du Colombier */
491219b2ee8SDavid du Colombier
492219b2ee8SDavid du Colombier if ( fp_acct != NULL )
493219b2ee8SDavid du Colombier fprintf(fp_acct, " print %d\n copies %d\n", printed, copies);
494219b2ee8SDavid du Colombier
495219b2ee8SDavid du Colombier } /* End of account */
496219b2ee8SDavid du Colombier
497219b2ee8SDavid du Colombier /*****************************************************************************/
498219b2ee8SDavid du Colombier
text()499219b2ee8SDavid du Colombier text()
500219b2ee8SDavid du Colombier
501219b2ee8SDavid du Colombier {
502219b2ee8SDavid du Colombier
503219b2ee8SDavid du Colombier int ch; /* next input character */
504219b2ee8SDavid du Colombier
505219b2ee8SDavid du Colombier /*
506219b2ee8SDavid du Colombier *
507219b2ee8SDavid du Colombier * Translates *fp_in into PostScript. Intercepts space, tab, backspace, newline,
508219b2ee8SDavid du Colombier * return, and formfeed. Everything else goes to oput(), which handles quoting
509219b2ee8SDavid du Colombier * (if needed) and escapes for nonascii characters if extended is TRUE. The
510219b2ee8SDavid du Colombier * redirect(-1) call forces the initial output to go to /dev/null - so stuff
511219b2ee8SDavid du Colombier * that formfeed() does at the end of each page goes to /dev/null rather than
512219b2ee8SDavid du Colombier * the real output file.
513219b2ee8SDavid du Colombier *
514219b2ee8SDavid du Colombier */
515219b2ee8SDavid du Colombier
516219b2ee8SDavid du Colombier redirect(-1); /* get ready for the first page */
517219b2ee8SDavid du Colombier formfeed(); /* force PAGE comment etc. */
518219b2ee8SDavid du Colombier
519219b2ee8SDavid du Colombier while ( (ch = getc(fp_in)) != EOF )
520219b2ee8SDavid du Colombier switch ( ch ) {
521219b2ee8SDavid du Colombier case '\n':
522219b2ee8SDavid du Colombier newline();
523219b2ee8SDavid du Colombier break;
524219b2ee8SDavid du Colombier
525219b2ee8SDavid du Colombier case '\t':
526219b2ee8SDavid du Colombier case '\b':
527219b2ee8SDavid du Colombier case ' ':
528219b2ee8SDavid du Colombier spaces(ch);
529219b2ee8SDavid du Colombier break;
530219b2ee8SDavid du Colombier
531219b2ee8SDavid du Colombier case '\014':
532219b2ee8SDavid du Colombier formfeed();
533219b2ee8SDavid du Colombier break;
534219b2ee8SDavid du Colombier
535219b2ee8SDavid du Colombier case '\r':
536219b2ee8SDavid du Colombier if ( crmode == 1 )
537219b2ee8SDavid du Colombier spaces(ch);
538219b2ee8SDavid du Colombier else if ( crmode == 2 )
539219b2ee8SDavid du Colombier newline();
540219b2ee8SDavid du Colombier break;
541219b2ee8SDavid du Colombier
542219b2ee8SDavid du Colombier default:
543219b2ee8SDavid du Colombier oput(ch);
544219b2ee8SDavid du Colombier break;
545219b2ee8SDavid du Colombier } /* End switch */
546219b2ee8SDavid du Colombier
547219b2ee8SDavid du Colombier formfeed(); /* next file starts on a new page? */
548219b2ee8SDavid du Colombier
549219b2ee8SDavid du Colombier } /* End of text */
550219b2ee8SDavid du Colombier
551219b2ee8SDavid du Colombier /*****************************************************************************/
552219b2ee8SDavid du Colombier
formfeed()553219b2ee8SDavid du Colombier formfeed()
554219b2ee8SDavid du Colombier
555219b2ee8SDavid du Colombier {
556219b2ee8SDavid du Colombier
557219b2ee8SDavid du Colombier /*
558219b2ee8SDavid du Colombier *
559219b2ee8SDavid du Colombier * Called whenever we've finished with the last page and want to get ready for the
560219b2ee8SDavid du Colombier * next one. Also used at the beginning and end of each input file, so we have to
561219b2ee8SDavid du Colombier * be careful about what's done. The first time through (up to the redirect() call)
562219b2ee8SDavid du Colombier * output goes to /dev/null.
563219b2ee8SDavid du Colombier *
564219b2ee8SDavid du Colombier * Adobe now recommends that the showpage operator occur after the page level
565219b2ee8SDavid du Colombier * restore so it can be easily redefined to have side-effects in the printer's VM.
566219b2ee8SDavid du Colombier * Although it seems reasonable I haven't implemented it, because it makes other
567219b2ee8SDavid du Colombier * things, like selectively setting manual feed or choosing an alternate paper
568219b2ee8SDavid du Colombier * tray, clumsy - at least on a per page basis.
569219b2ee8SDavid du Colombier *
570219b2ee8SDavid du Colombier */
571219b2ee8SDavid du Colombier
572219b2ee8SDavid du Colombier if ( fp_out == stdout ) /* count the last page */
573219b2ee8SDavid du Colombier printed++;
574219b2ee8SDavid du Colombier
575219b2ee8SDavid du Colombier endline(); /* print the last line */
576219b2ee8SDavid du Colombier
577219b2ee8SDavid du Colombier fprintf(fp_out, "cleartomark\n");
578219b2ee8SDavid du Colombier fprintf(fp_out, "showpage\n");
579219b2ee8SDavid du Colombier fprintf(fp_out, "saveobj restore\n");
580219b2ee8SDavid du Colombier fprintf(fp_out, "%s %d %d\n", ENDPAGE, page, printed);
581219b2ee8SDavid du Colombier
582219b2ee8SDavid du Colombier if ( ungetc(getc(fp_in), fp_in) == EOF )
583219b2ee8SDavid du Colombier redirect(-1);
584219b2ee8SDavid du Colombier else redirect(++page);
585219b2ee8SDavid du Colombier
586219b2ee8SDavid du Colombier fprintf(fp_out, "%s %d %d\n", PAGE, page, printed+1);
587219b2ee8SDavid du Colombier fprintf(fp_out, "/saveobj save def\n");
588219b2ee8SDavid du Colombier fprintf(fp_out, "mark\n");
589219b2ee8SDavid du Colombier writerequest(printed+1, fp_out);
590219b2ee8SDavid du Colombier fprintf(fp_out, "%d pagesetup\n", printed+1);
591219b2ee8SDavid du Colombier
592219b2ee8SDavid du Colombier line = 1;
593219b2ee8SDavid du Colombier
594219b2ee8SDavid du Colombier } /* End of formfeed */
595219b2ee8SDavid du Colombier
596219b2ee8SDavid du Colombier /*****************************************************************************/
597219b2ee8SDavid du Colombier
newline()598219b2ee8SDavid du Colombier newline()
599219b2ee8SDavid du Colombier
600219b2ee8SDavid du Colombier {
601219b2ee8SDavid du Colombier
602219b2ee8SDavid du Colombier /*
603219b2ee8SDavid du Colombier *
604219b2ee8SDavid du Colombier * Called when we've read a newline character. The call to startline() ensures
605219b2ee8SDavid du Colombier * that at least an empty string is on the stack.
606219b2ee8SDavid du Colombier *
607219b2ee8SDavid du Colombier */
608219b2ee8SDavid du Colombier
609219b2ee8SDavid du Colombier startline();
610219b2ee8SDavid du Colombier endline(); /* print the current line */
611219b2ee8SDavid du Colombier
612219b2ee8SDavid du Colombier if ( ++line > linespp ) /* done with this page */
613219b2ee8SDavid du Colombier formfeed();
614219b2ee8SDavid du Colombier
615219b2ee8SDavid du Colombier } /* End of newline */
616219b2ee8SDavid du Colombier
617219b2ee8SDavid du Colombier /*****************************************************************************/
618219b2ee8SDavid du Colombier
spaces(ch)619219b2ee8SDavid du Colombier spaces(ch)
620219b2ee8SDavid du Colombier
621219b2ee8SDavid du Colombier int ch; /* next input character */
622219b2ee8SDavid du Colombier
623219b2ee8SDavid du Colombier {
624219b2ee8SDavid du Colombier
625219b2ee8SDavid du Colombier int endcol; /* ending column */
626219b2ee8SDavid du Colombier int i; /* final distance - in spaces */
627219b2ee8SDavid du Colombier
628219b2ee8SDavid du Colombier /*
629219b2ee8SDavid du Colombier *
630219b2ee8SDavid du Colombier * Counts consecutive spaces, tabs, and backspaces and figures out where the next
631219b2ee8SDavid du Colombier * string should start. Once that's been done we try to choose an efficient way
632219b2ee8SDavid du Colombier * to output the required number of spaces. The choice is between using procedure
633219b2ee8SDavid du Colombier * l with a single string on the stack and L with several string and column pairs.
634219b2ee8SDavid du Colombier * We usually break even, in terms of the size of the output file, if we need four
635219b2ee8SDavid du Colombier * consecutive spaces. More means using L decreases the size of the file. For now
636219b2ee8SDavid du Colombier * if there are less than 6 consecutive spaces we just add them to the current
637219b2ee8SDavid du Colombier * string, otherwise we end that string, follow it by its starting position, and
638219b2ee8SDavid du Colombier * begin a new one that starts at endcol. Backspacing is always handled this way.
639219b2ee8SDavid du Colombier *
640219b2ee8SDavid du Colombier */
641219b2ee8SDavid du Colombier
642219b2ee8SDavid du Colombier startline(); /* so col makes sense */
643219b2ee8SDavid du Colombier endcol = col;
644219b2ee8SDavid du Colombier
645219b2ee8SDavid du Colombier do {
646219b2ee8SDavid du Colombier if ( ch == ' ' )
647219b2ee8SDavid du Colombier endcol++;
648219b2ee8SDavid du Colombier else if ( ch == '\t' )
649219b2ee8SDavid du Colombier endcol += tabstops - ((endcol - 1) % tabstops);
650219b2ee8SDavid du Colombier else if ( ch == '\b' )
651219b2ee8SDavid du Colombier endcol--;
652219b2ee8SDavid du Colombier else if ( ch == '\r' )
653219b2ee8SDavid du Colombier endcol = 1;
654219b2ee8SDavid du Colombier else break;
655219b2ee8SDavid du Colombier } while ( ch = getc(fp_in) ); /* if ch is 0 we'd quit anyway */
656219b2ee8SDavid du Colombier
657219b2ee8SDavid du Colombier ungetc(ch, fp_in); /* wasn't a space, tab, or backspace */
658219b2ee8SDavid du Colombier
659219b2ee8SDavid du Colombier if ( endcol < 1 ) /* can't move past left edge */
660219b2ee8SDavid du Colombier endcol = 1;
661219b2ee8SDavid du Colombier
662219b2ee8SDavid du Colombier if ( (i = endcol - col) >= 0 && i < 6 )
663219b2ee8SDavid du Colombier for ( ; i > 0; i-- )
664219b2ee8SDavid du Colombier oput((int)' ');
665219b2ee8SDavid du Colombier else {
666219b2ee8SDavid du Colombier endstring();
667219b2ee8SDavid du Colombier col = stringstart = endcol;
668219b2ee8SDavid du Colombier } /* End else */
669219b2ee8SDavid du Colombier
670219b2ee8SDavid du Colombier } /* End of spaces */
671219b2ee8SDavid du Colombier
672219b2ee8SDavid du Colombier /*****************************************************************************/
673219b2ee8SDavid du Colombier
startline()674219b2ee8SDavid du Colombier startline()
675219b2ee8SDavid du Colombier
676219b2ee8SDavid du Colombier {
677219b2ee8SDavid du Colombier
678219b2ee8SDavid du Colombier /*
679219b2ee8SDavid du Colombier *
680219b2ee8SDavid du Colombier * Called whenever we want to be certain we're ready to start pushing characters
681219b2ee8SDavid du Colombier * into an open string on the stack. If stringcount is positive we've already
682219b2ee8SDavid du Colombier * started, so there's nothing to do. The first string starts in column 1.
683219b2ee8SDavid du Colombier *
684219b2ee8SDavid du Colombier */
685219b2ee8SDavid du Colombier
686219b2ee8SDavid du Colombier if ( stringcount < 1 ) {
687219b2ee8SDavid du Colombier putc('(', fp_out);
688219b2ee8SDavid du Colombier stringstart = col = 1;
689219b2ee8SDavid du Colombier stringcount = 1;
690219b2ee8SDavid du Colombier } /* End if */
691219b2ee8SDavid du Colombier
692219b2ee8SDavid du Colombier } /* End of startline */
693219b2ee8SDavid du Colombier
694219b2ee8SDavid du Colombier /*****************************************************************************/
695219b2ee8SDavid du Colombier
endstring()696219b2ee8SDavid du Colombier endstring()
697219b2ee8SDavid du Colombier
698219b2ee8SDavid du Colombier {
699219b2ee8SDavid du Colombier
700219b2ee8SDavid du Colombier /*
701219b2ee8SDavid du Colombier *
702219b2ee8SDavid du Colombier * End the current string and start a new one.
703219b2ee8SDavid du Colombier *
704219b2ee8SDavid du Colombier */
705219b2ee8SDavid du Colombier
706219b2ee8SDavid du Colombier if ( stringcount > 100 ) { /* don't put too much on the stack */
707219b2ee8SDavid du Colombier fprintf(fp_out, ")%d LL\n(", stringstart-1);
708219b2ee8SDavid du Colombier stringcount = 2; /* kludge - don't let endline() use l */
709219b2ee8SDavid du Colombier } else {
710219b2ee8SDavid du Colombier fprintf(fp_out, ")%d(", stringstart-1);
711219b2ee8SDavid du Colombier stringcount++;
712219b2ee8SDavid du Colombier } /* End else */
713219b2ee8SDavid du Colombier
714219b2ee8SDavid du Colombier } /* End of endstring */
715219b2ee8SDavid du Colombier
716219b2ee8SDavid du Colombier /*****************************************************************************/
717219b2ee8SDavid du Colombier
endline()718219b2ee8SDavid du Colombier endline()
719219b2ee8SDavid du Colombier
720219b2ee8SDavid du Colombier {
721219b2ee8SDavid du Colombier
722219b2ee8SDavid du Colombier /*
723219b2ee8SDavid du Colombier *
724219b2ee8SDavid du Colombier * Generates a call to the PostScript procedure that processes all the text on
725219b2ee8SDavid du Colombier * the stack - provided stringcount is positive. If one string is on the stack
726219b2ee8SDavid du Colombier * the fast procedure (ie. l) is used to print the line, otherwise the slower
727219b2ee8SDavid du Colombier * one that processes string and column pairs is used.
728219b2ee8SDavid du Colombier *
729219b2ee8SDavid du Colombier */
730219b2ee8SDavid du Colombier
731219b2ee8SDavid du Colombier if ( stringcount == 1 )
732219b2ee8SDavid du Colombier fprintf(fp_out, ")l\n");
733219b2ee8SDavid du Colombier else if ( stringcount > 1 )
734219b2ee8SDavid du Colombier fprintf(fp_out, ")%d L\n", stringstart-1);
735219b2ee8SDavid du Colombier
736219b2ee8SDavid du Colombier stringcount = 0;
737219b2ee8SDavid du Colombier
738219b2ee8SDavid du Colombier } /* End of endline */
739219b2ee8SDavid du Colombier
740219b2ee8SDavid du Colombier /*****************************************************************************/
741219b2ee8SDavid du Colombier
oput(ch)742219b2ee8SDavid du Colombier oput(ch)
743219b2ee8SDavid du Colombier
744219b2ee8SDavid du Colombier int ch; /* next output character */
745219b2ee8SDavid du Colombier
746219b2ee8SDavid du Colombier {
747219b2ee8SDavid du Colombier
748219b2ee8SDavid du Colombier /*
749219b2ee8SDavid du Colombier *
750219b2ee8SDavid du Colombier * Responsible for adding all printing characters from the input file to the
751219b2ee8SDavid du Colombier * open string on top of the stack.
752219b2ee8SDavid du Colombier *
753219b2ee8SDavid du Colombier */
754219b2ee8SDavid du Colombier
755219b2ee8SDavid du Colombier if ( isascii(ch) && isprint(ch) ) {
756219b2ee8SDavid du Colombier startline();
757219b2ee8SDavid du Colombier if ( ch == '(' || ch == ')' || ch == '\\' )
758219b2ee8SDavid du Colombier putc('\\', fp_out);
759219b2ee8SDavid du Colombier putc(ch, fp_out);
760219b2ee8SDavid du Colombier col++;
761219b2ee8SDavid du Colombier } else if ( extended == TRUE ) {
762219b2ee8SDavid du Colombier startline();
763219b2ee8SDavid du Colombier fprintf(fp_out, "\\%.3o", ch & 0377);
764219b2ee8SDavid du Colombier col++;
765219b2ee8SDavid du Colombier } /* End if */
766219b2ee8SDavid du Colombier
767219b2ee8SDavid du Colombier } /* End of oput */
768219b2ee8SDavid du Colombier
769219b2ee8SDavid du Colombier /*****************************************************************************/
770219b2ee8SDavid du Colombier
redirect(pg)771219b2ee8SDavid du Colombier redirect(pg)
772219b2ee8SDavid du Colombier
773219b2ee8SDavid du Colombier int pg; /* next page we're printing */
774219b2ee8SDavid du Colombier
775219b2ee8SDavid du Colombier {
776219b2ee8SDavid du Colombier
777219b2ee8SDavid du Colombier static FILE *fp_null = NULL; /* if output is turned off */
778219b2ee8SDavid du Colombier
779219b2ee8SDavid du Colombier /*
780219b2ee8SDavid du Colombier *
781219b2ee8SDavid du Colombier * If we're not supposed to print page pg, fp_out will be directed to /dev/null,
782219b2ee8SDavid du Colombier * otherwise output goes to stdout.
783219b2ee8SDavid du Colombier *
784219b2ee8SDavid du Colombier */
785219b2ee8SDavid du Colombier
786219b2ee8SDavid du Colombier if ( pg >= 0 && in_olist(pg) == ON )
787219b2ee8SDavid du Colombier fp_out = stdout;
788219b2ee8SDavid du Colombier else if ( (fp_out = fp_null) == NULL )
789219b2ee8SDavid du Colombier fp_out = fp_null = fopen("/dev/null", "w");
790219b2ee8SDavid du Colombier
791219b2ee8SDavid du Colombier } /* End of redirect */
792219b2ee8SDavid du Colombier
793219b2ee8SDavid du Colombier /*****************************************************************************/
794219b2ee8SDavid du Colombier
795