xref: /freebsd-src/contrib/flex/src/filter.c (revision 7e38239042df09edbbdc443ccb4825f9155c6bb7)
1*7e382390SJung-uk Kim /* filter - postprocessing of flex output through filters */
2*7e382390SJung-uk Kim 
3*7e382390SJung-uk Kim /*  This file is part of flex. */
4*7e382390SJung-uk Kim 
5*7e382390SJung-uk Kim /*  Redistribution and use in source and binary forms, with or without */
6*7e382390SJung-uk Kim /*  modification, are permitted provided that the following conditions */
7*7e382390SJung-uk Kim /*  are met: */
8*7e382390SJung-uk Kim 
9*7e382390SJung-uk Kim /*  1. Redistributions of source code must retain the above copyright */
10*7e382390SJung-uk Kim /*     notice, this list of conditions and the following disclaimer. */
11*7e382390SJung-uk Kim /*  2. Redistributions in binary form must reproduce the above copyright */
12*7e382390SJung-uk Kim /*     notice, this list of conditions and the following disclaimer in the */
13*7e382390SJung-uk Kim /*     documentation and/or other materials provided with the distribution. */
14*7e382390SJung-uk Kim 
15*7e382390SJung-uk Kim /*  Neither the name of the University nor the names of its contributors */
16*7e382390SJung-uk Kim /*  may be used to endorse or promote products derived from this software */
17*7e382390SJung-uk Kim /*  without specific prior written permission. */
18*7e382390SJung-uk Kim 
19*7e382390SJung-uk Kim /*  THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR */
20*7e382390SJung-uk Kim /*  IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED */
21*7e382390SJung-uk Kim /*  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR */
22*7e382390SJung-uk Kim /*  PURPOSE. */
23*7e382390SJung-uk Kim 
24*7e382390SJung-uk Kim #include "flexdef.h"
25*7e382390SJung-uk Kim static const char * check_4_gnu_m4 =
26*7e382390SJung-uk Kim     "m4_dnl ifdef(`__gnu__', ,"
27*7e382390SJung-uk Kim     "`errprint(Flex requires GNU M4. Set the PATH or set the M4 environment variable to its path name.)"
28*7e382390SJung-uk Kim     " m4exit(2)')\n";
29*7e382390SJung-uk Kim 
30*7e382390SJung-uk Kim 
31*7e382390SJung-uk Kim /** global chain. */
32*7e382390SJung-uk Kim struct filter *output_chain = NULL;
33*7e382390SJung-uk Kim 
34*7e382390SJung-uk Kim /* Allocate and initialize an external filter.
35*7e382390SJung-uk Kim  * @param chain the current chain or NULL for new chain
36*7e382390SJung-uk Kim  * @param cmd the command to execute.
37*7e382390SJung-uk Kim  * @param ... a NULL terminated list of (const char*) arguments to command,
38*7e382390SJung-uk Kim  *            not including argv[0].
39*7e382390SJung-uk Kim  * @return newest filter in chain
40*7e382390SJung-uk Kim  */
filter_create_ext(struct filter * chain,const char * cmd,...)41*7e382390SJung-uk Kim struct filter *filter_create_ext (struct filter *chain, const char *cmd,
42*7e382390SJung-uk Kim 				  ...)
43*7e382390SJung-uk Kim {
44*7e382390SJung-uk Kim 	struct filter *f;
45*7e382390SJung-uk Kim 	int     max_args;
46*7e382390SJung-uk Kim 	const char *s;
47*7e382390SJung-uk Kim 	va_list ap;
48*7e382390SJung-uk Kim 
49*7e382390SJung-uk Kim 	/* allocate and initialize new filter */
50*7e382390SJung-uk Kim 	f = malloc(sizeof(struct filter));
51*7e382390SJung-uk Kim 	if (!f)
52*7e382390SJung-uk Kim 		flexerror(_("malloc failed (f) in filter_create_ext"));
53*7e382390SJung-uk Kim 	memset (f, 0, sizeof (*f));
54*7e382390SJung-uk Kim 	f->filter_func = NULL;
55*7e382390SJung-uk Kim 	f->extra = NULL;
56*7e382390SJung-uk Kim 	f->next = NULL;
57*7e382390SJung-uk Kim 	f->argc = 0;
58*7e382390SJung-uk Kim 
59*7e382390SJung-uk Kim 	if (chain != NULL) {
60*7e382390SJung-uk Kim 		/* append f to end of chain */
61*7e382390SJung-uk Kim 		while (chain->next)
62*7e382390SJung-uk Kim 			chain = chain->next;
63*7e382390SJung-uk Kim 		chain->next = f;
64*7e382390SJung-uk Kim 	}
65*7e382390SJung-uk Kim 
66*7e382390SJung-uk Kim 
67*7e382390SJung-uk Kim 	/* allocate argv, and populate it with the argument list. */
68*7e382390SJung-uk Kim 	max_args = 8;
69*7e382390SJung-uk Kim 	f->argv = malloc(sizeof(char *) * (size_t) (max_args + 1));
70*7e382390SJung-uk Kim 	if (!f->argv)
71*7e382390SJung-uk Kim 		flexerror(_("malloc failed (f->argv) in filter_create_ext"));
72*7e382390SJung-uk Kim 	f->argv[f->argc++] = cmd;
73*7e382390SJung-uk Kim 
74*7e382390SJung-uk Kim 	va_start (ap, cmd);
75*7e382390SJung-uk Kim 	while ((s = va_arg (ap, const char *)) != NULL) {
76*7e382390SJung-uk Kim 		if (f->argc >= max_args) {
77*7e382390SJung-uk Kim 			max_args += 8;
78*7e382390SJung-uk Kim 			f->argv = realloc(f->argv, sizeof(char*) * (size_t) (max_args + 1));
79*7e382390SJung-uk Kim 		}
80*7e382390SJung-uk Kim 		f->argv[f->argc++] = s;
81*7e382390SJung-uk Kim 	}
82*7e382390SJung-uk Kim 	f->argv[f->argc] = NULL;
83*7e382390SJung-uk Kim 
84*7e382390SJung-uk Kim 	va_end (ap);
85*7e382390SJung-uk Kim 	return f;
86*7e382390SJung-uk Kim }
87*7e382390SJung-uk Kim 
88*7e382390SJung-uk Kim /* Allocate and initialize an internal filter.
89*7e382390SJung-uk Kim  * @param chain the current chain or NULL for new chain
90*7e382390SJung-uk Kim  * @param filter_func The function that will perform the filtering.
91*7e382390SJung-uk Kim  *        filter_func should return 0 if successful, and -1
92*7e382390SJung-uk Kim  *        if an error occurs -- or it can simply exit().
93*7e382390SJung-uk Kim  * @param extra optional user-defined data to pass to the filter.
94*7e382390SJung-uk Kim  * @return newest filter in chain
95*7e382390SJung-uk Kim  */
filter_create_int(struct filter * chain,int (* filter_func)(struct filter *),void * extra)96*7e382390SJung-uk Kim struct filter *filter_create_int (struct filter *chain,
97*7e382390SJung-uk Kim 				  int (*filter_func) (struct filter *),
98*7e382390SJung-uk Kim 				  void *extra)
99*7e382390SJung-uk Kim {
100*7e382390SJung-uk Kim 	struct filter *f;
101*7e382390SJung-uk Kim 
102*7e382390SJung-uk Kim 	/* allocate and initialize new filter */
103*7e382390SJung-uk Kim 	f = malloc(sizeof(struct filter));
104*7e382390SJung-uk Kim 	if (!f)
105*7e382390SJung-uk Kim 		flexerror(_("malloc failed in filter_create_int"));
106*7e382390SJung-uk Kim 	memset (f, 0, sizeof (*f));
107*7e382390SJung-uk Kim 	f->next = NULL;
108*7e382390SJung-uk Kim 	f->argc = 0;
109*7e382390SJung-uk Kim 	f->argv = NULL;
110*7e382390SJung-uk Kim 
111*7e382390SJung-uk Kim 	f->filter_func = filter_func;
112*7e382390SJung-uk Kim 	f->extra = extra;
113*7e382390SJung-uk Kim 
114*7e382390SJung-uk Kim 	if (chain != NULL) {
115*7e382390SJung-uk Kim 		/* append f to end of chain */
116*7e382390SJung-uk Kim 		while (chain->next)
117*7e382390SJung-uk Kim 			chain = chain->next;
118*7e382390SJung-uk Kim 		chain->next = f;
119*7e382390SJung-uk Kim 	}
120*7e382390SJung-uk Kim 
121*7e382390SJung-uk Kim 	return f;
122*7e382390SJung-uk Kim }
123*7e382390SJung-uk Kim 
124*7e382390SJung-uk Kim /** Fork and exec entire filter chain.
125*7e382390SJung-uk Kim  *  @param chain The head of the chain.
126*7e382390SJung-uk Kim  *  @return true on success.
127*7e382390SJung-uk Kim  */
filter_apply_chain(struct filter * chain)128*7e382390SJung-uk Kim bool filter_apply_chain (struct filter * chain)
129*7e382390SJung-uk Kim {
130*7e382390SJung-uk Kim 	int     pid, pipes[2];
131*7e382390SJung-uk Kim 
132*7e382390SJung-uk Kim 
133*7e382390SJung-uk Kim 	/* Tricky recursion, since we want to begin the chain
134*7e382390SJung-uk Kim 	 * at the END. Why? Because we need all the forked processes
135*7e382390SJung-uk Kim 	 * to be children of the main flex process.
136*7e382390SJung-uk Kim 	 */
137*7e382390SJung-uk Kim 	if (chain)
138*7e382390SJung-uk Kim 		filter_apply_chain (chain->next);
139*7e382390SJung-uk Kim 	else
140*7e382390SJung-uk Kim 		return true;
141*7e382390SJung-uk Kim 
142*7e382390SJung-uk Kim 	/* Now we are the right-most unprocessed link in the chain.
143*7e382390SJung-uk Kim 	 */
144*7e382390SJung-uk Kim 
145*7e382390SJung-uk Kim 	fflush (stdout);
146*7e382390SJung-uk Kim 	fflush (stderr);
147*7e382390SJung-uk Kim 
148*7e382390SJung-uk Kim 
149*7e382390SJung-uk Kim 	if (pipe (pipes) == -1)
150*7e382390SJung-uk Kim 		flexerror (_("pipe failed"));
151*7e382390SJung-uk Kim 
152*7e382390SJung-uk Kim 	if ((pid = fork ()) == -1)
153*7e382390SJung-uk Kim 		flexerror (_("fork failed"));
154*7e382390SJung-uk Kim 
155*7e382390SJung-uk Kim 	if (pid == 0) {
156*7e382390SJung-uk Kim 		/* child */
157*7e382390SJung-uk Kim 
158*7e382390SJung-uk Kim         /* We need stdin (the FILE* stdin) to connect to this new pipe.
159*7e382390SJung-uk Kim          * There is no portable way to set stdin to a new file descriptor,
160*7e382390SJung-uk Kim          * as stdin is not an lvalue on some systems (BSD).
161*7e382390SJung-uk Kim          * So we dup the new pipe onto the stdin descriptor and use a no-op fseek
162*7e382390SJung-uk Kim          * to sync the stream. This is a Hail Mary situation. It seems to work.
163*7e382390SJung-uk Kim          */
164*7e382390SJung-uk Kim 		close (pipes[1]);
165*7e382390SJung-uk Kim clearerr(stdin);
166*7e382390SJung-uk Kim 		if (dup2 (pipes[0], fileno (stdin)) == -1)
167*7e382390SJung-uk Kim 			flexfatal (_("dup2(pipes[0],0)"));
168*7e382390SJung-uk Kim 		close (pipes[0]);
169*7e382390SJung-uk Kim         fseek (stdin, 0, SEEK_CUR);
170*7e382390SJung-uk Kim         ungetc(' ', stdin); /* still an evil hack, but one that works better */
171*7e382390SJung-uk Kim         (void)fgetc(stdin); /* on NetBSD than the fseek attempt does */
172*7e382390SJung-uk Kim 
173*7e382390SJung-uk Kim 		/* run as a filter, either internally or by exec */
174*7e382390SJung-uk Kim 		if (chain->filter_func) {
175*7e382390SJung-uk Kim 			int     r;
176*7e382390SJung-uk Kim 
177*7e382390SJung-uk Kim 			if ((r = chain->filter_func (chain)) == -1)
178*7e382390SJung-uk Kim 				flexfatal (_("filter_func failed"));
179*7e382390SJung-uk Kim 			FLEX_EXIT (0);
180*7e382390SJung-uk Kim 		}
181*7e382390SJung-uk Kim 		else {
182*7e382390SJung-uk Kim 			execvp (chain->argv[0],
183*7e382390SJung-uk Kim 				(char **const) (chain->argv));
184*7e382390SJung-uk Kim             lerr_fatal ( _("exec of %s failed"),
185*7e382390SJung-uk Kim                     chain->argv[0]);
186*7e382390SJung-uk Kim 		}
187*7e382390SJung-uk Kim 
188*7e382390SJung-uk Kim 		FLEX_EXIT (1);
189*7e382390SJung-uk Kim 	}
190*7e382390SJung-uk Kim 
191*7e382390SJung-uk Kim 	/* Parent */
192*7e382390SJung-uk Kim 	close (pipes[0]);
193*7e382390SJung-uk Kim 	if (dup2 (pipes[1], fileno (stdout)) == -1)
194*7e382390SJung-uk Kim 		flexfatal (_("dup2(pipes[1],1)"));
195*7e382390SJung-uk Kim 	close (pipes[1]);
196*7e382390SJung-uk Kim     fseek (stdout, 0, SEEK_CUR);
197*7e382390SJung-uk Kim 
198*7e382390SJung-uk Kim 	return true;
199*7e382390SJung-uk Kim }
200*7e382390SJung-uk Kim 
201*7e382390SJung-uk Kim /** Truncate the chain to max_len number of filters.
202*7e382390SJung-uk Kim  * @param chain the current chain.
203*7e382390SJung-uk Kim  * @param max_len the maximum length of the chain.
204*7e382390SJung-uk Kim  * @return the resulting length of the chain.
205*7e382390SJung-uk Kim  */
filter_truncate(struct filter * chain,int max_len)206*7e382390SJung-uk Kim int filter_truncate (struct filter *chain, int max_len)
207*7e382390SJung-uk Kim {
208*7e382390SJung-uk Kim 	int     len = 1;
209*7e382390SJung-uk Kim 
210*7e382390SJung-uk Kim 	if (!chain)
211*7e382390SJung-uk Kim 		return 0;
212*7e382390SJung-uk Kim 
213*7e382390SJung-uk Kim 	while (chain->next && len < max_len) {
214*7e382390SJung-uk Kim 		chain = chain->next;
215*7e382390SJung-uk Kim 		++len;
216*7e382390SJung-uk Kim 	}
217*7e382390SJung-uk Kim 
218*7e382390SJung-uk Kim 	chain->next = NULL;
219*7e382390SJung-uk Kim 	return len;
220*7e382390SJung-uk Kim }
221*7e382390SJung-uk Kim 
222*7e382390SJung-uk Kim /** Splits the chain in order to write to a header file.
223*7e382390SJung-uk Kim  *  Similar in spirit to the 'tee' program.
224*7e382390SJung-uk Kim  *  The header file name is in extra.
225*7e382390SJung-uk Kim  *  @return 0 (zero) on success, and -1 on failure.
226*7e382390SJung-uk Kim  */
filter_tee_header(struct filter * chain)227*7e382390SJung-uk Kim int filter_tee_header (struct filter *chain)
228*7e382390SJung-uk Kim {
229*7e382390SJung-uk Kim 	/* This function reads from stdin and writes to both the C file and the
230*7e382390SJung-uk Kim 	 * header file at the same time.
231*7e382390SJung-uk Kim 	 */
232*7e382390SJung-uk Kim 
233*7e382390SJung-uk Kim 	const int readsz = 512;
234*7e382390SJung-uk Kim 	char   *buf;
235*7e382390SJung-uk Kim 	int     to_cfd = -1;
236*7e382390SJung-uk Kim 	FILE   *to_c = NULL, *to_h = NULL;
237*7e382390SJung-uk Kim 	bool    write_header;
238*7e382390SJung-uk Kim 
239*7e382390SJung-uk Kim 	write_header = (chain->extra != NULL);
240*7e382390SJung-uk Kim 
241*7e382390SJung-uk Kim 	/* Store a copy of the stdout pipe, which is already piped to C file
242*7e382390SJung-uk Kim 	 * through the running chain. Then create a new pipe to the H file as
243*7e382390SJung-uk Kim 	 * stdout, and fork the rest of the chain again.
244*7e382390SJung-uk Kim 	 */
245*7e382390SJung-uk Kim 
246*7e382390SJung-uk Kim 	if ((to_cfd = dup (1)) == -1)
247*7e382390SJung-uk Kim 		flexfatal (_("dup(1) failed"));
248*7e382390SJung-uk Kim 	to_c = fdopen (to_cfd, "w");
249*7e382390SJung-uk Kim 
250*7e382390SJung-uk Kim 	if (write_header) {
251*7e382390SJung-uk Kim 		if (freopen ((char *) chain->extra, "w", stdout) == NULL)
252*7e382390SJung-uk Kim 			flexfatal (_("freopen(headerfilename) failed"));
253*7e382390SJung-uk Kim 
254*7e382390SJung-uk Kim 		filter_apply_chain (chain->next);
255*7e382390SJung-uk Kim 		to_h = stdout;
256*7e382390SJung-uk Kim 	}
257*7e382390SJung-uk Kim 
258*7e382390SJung-uk Kim 	/* Now to_c is a pipe to the C branch, and to_h is a pipe to the H branch.
259*7e382390SJung-uk Kim 	 */
260*7e382390SJung-uk Kim 
261*7e382390SJung-uk Kim 	if (write_header) {
262*7e382390SJung-uk Kim         fputs (check_4_gnu_m4, to_h);
263*7e382390SJung-uk Kim 		fputs ("m4_changecom`'m4_dnl\n", to_h);
264*7e382390SJung-uk Kim 		fputs ("m4_changequote`'m4_dnl\n", to_h);
265*7e382390SJung-uk Kim 		fputs ("m4_changequote([[,]])[[]]m4_dnl\n", to_h);
266*7e382390SJung-uk Kim 	    fputs ("m4_define([[M4_YY_NOOP]])[[]]m4_dnl\n", to_h);
267*7e382390SJung-uk Kim 		fputs ("m4_define( [[M4_YY_IN_HEADER]],[[]])m4_dnl\n",
268*7e382390SJung-uk Kim 		       to_h);
269*7e382390SJung-uk Kim 		fprintf (to_h, "#ifndef %sHEADER_H\n", prefix);
270*7e382390SJung-uk Kim 		fprintf (to_h, "#define %sHEADER_H 1\n", prefix);
271*7e382390SJung-uk Kim 		fprintf (to_h, "#define %sIN_HEADER 1\n\n", prefix);
272*7e382390SJung-uk Kim 		fprintf (to_h,
273*7e382390SJung-uk Kim 			 "m4_define( [[M4_YY_OUTFILE_NAME]],[[%s]])m4_dnl\n",
274*7e382390SJung-uk Kim 			 headerfilename ? headerfilename : "<stdout>");
275*7e382390SJung-uk Kim 
276*7e382390SJung-uk Kim 	}
277*7e382390SJung-uk Kim 
278*7e382390SJung-uk Kim     fputs (check_4_gnu_m4, to_c);
279*7e382390SJung-uk Kim 	fputs ("m4_changecom`'m4_dnl\n", to_c);
280*7e382390SJung-uk Kim 	fputs ("m4_changequote`'m4_dnl\n", to_c);
281*7e382390SJung-uk Kim 	fputs ("m4_changequote([[,]])[[]]m4_dnl\n", to_c);
282*7e382390SJung-uk Kim 	fputs ("m4_define([[M4_YY_NOOP]])[[]]m4_dnl\n", to_c);
283*7e382390SJung-uk Kim 	fprintf (to_c, "m4_define( [[M4_YY_OUTFILE_NAME]],[[%s]])m4_dnl\n",
284*7e382390SJung-uk Kim 		 outfilename ? outfilename : "<stdout>");
285*7e382390SJung-uk Kim 
286*7e382390SJung-uk Kim 	buf = malloc((size_t) readsz);
287*7e382390SJung-uk Kim 	if (!buf)
288*7e382390SJung-uk Kim 		flexerror(_("malloc failed in filter_tee_header"));
289*7e382390SJung-uk Kim 	while (fgets (buf, readsz, stdin)) {
290*7e382390SJung-uk Kim 		fputs (buf, to_c);
291*7e382390SJung-uk Kim 		if (write_header)
292*7e382390SJung-uk Kim 			fputs (buf, to_h);
293*7e382390SJung-uk Kim 	}
294*7e382390SJung-uk Kim 
295*7e382390SJung-uk Kim 	if (write_header) {
296*7e382390SJung-uk Kim 		fprintf (to_h, "\n");
297*7e382390SJung-uk Kim 
298*7e382390SJung-uk Kim 		/* write a fake line number. It will get fixed by the linedir filter. */
299*7e382390SJung-uk Kim 		if (gen_line_dirs)
300*7e382390SJung-uk Kim 			fprintf (to_h, "#line 4000 \"M4_YY_OUTFILE_NAME\"\n");
301*7e382390SJung-uk Kim 
302*7e382390SJung-uk Kim 		fprintf (to_h, "#undef %sIN_HEADER\n", prefix);
303*7e382390SJung-uk Kim 		fprintf (to_h, "#endif /* %sHEADER_H */\n", prefix);
304*7e382390SJung-uk Kim 		fputs ("m4_undefine( [[M4_YY_IN_HEADER]])m4_dnl\n", to_h);
305*7e382390SJung-uk Kim 
306*7e382390SJung-uk Kim 		fflush (to_h);
307*7e382390SJung-uk Kim 		if (ferror (to_h))
308*7e382390SJung-uk Kim 			lerr (_("error writing output file %s"),
309*7e382390SJung-uk Kim 				(char *) chain->extra);
310*7e382390SJung-uk Kim 
311*7e382390SJung-uk Kim 		else if (fclose (to_h))
312*7e382390SJung-uk Kim 			lerr (_("error closing output file %s"),
313*7e382390SJung-uk Kim 				(char *) chain->extra);
314*7e382390SJung-uk Kim 	}
315*7e382390SJung-uk Kim 
316*7e382390SJung-uk Kim 	fflush (to_c);
317*7e382390SJung-uk Kim 	if (ferror (to_c))
318*7e382390SJung-uk Kim 		lerr (_("error writing output file %s"),
319*7e382390SJung-uk Kim 			outfilename ? outfilename : "<stdout>");
320*7e382390SJung-uk Kim 
321*7e382390SJung-uk Kim 	else if (fclose (to_c))
322*7e382390SJung-uk Kim 		lerr (_("error closing output file %s"),
323*7e382390SJung-uk Kim 			outfilename ? outfilename : "<stdout>");
324*7e382390SJung-uk Kim 
325*7e382390SJung-uk Kim 	while (wait (0) > 0) ;
326*7e382390SJung-uk Kim 
327*7e382390SJung-uk Kim 	FLEX_EXIT (0);
328*7e382390SJung-uk Kim 	return 0;
329*7e382390SJung-uk Kim }
330*7e382390SJung-uk Kim 
331*7e382390SJung-uk Kim /** Adjust the line numbers in the #line directives of the generated scanner.
332*7e382390SJung-uk Kim  * After the m4 expansion, the line numbers are incorrect since the m4 macros
333*7e382390SJung-uk Kim  * can add or remove lines.  This only adjusts line numbers for generated code,
334*7e382390SJung-uk Kim  * not user code. This also happens to be a good place to squeeze multiple
335*7e382390SJung-uk Kim  * blank lines into a single blank line.
336*7e382390SJung-uk Kim  */
filter_fix_linedirs(struct filter * chain)337*7e382390SJung-uk Kim int filter_fix_linedirs (struct filter *chain)
338*7e382390SJung-uk Kim {
339*7e382390SJung-uk Kim 	char   *buf;
340*7e382390SJung-uk Kim 	const size_t readsz = 512;
341*7e382390SJung-uk Kim 	int     lineno = 1;
342*7e382390SJung-uk Kim 	bool    in_gen = true;	/* in generated code */
343*7e382390SJung-uk Kim 	bool    last_was_blank = false;
344*7e382390SJung-uk Kim 
345*7e382390SJung-uk Kim 	if (!chain)
346*7e382390SJung-uk Kim 		return 0;
347*7e382390SJung-uk Kim 
348*7e382390SJung-uk Kim 	buf = malloc(readsz);
349*7e382390SJung-uk Kim 	if (!buf)
350*7e382390SJung-uk Kim 		flexerror(_("malloc failed in filter_fix_linedirs"));
351*7e382390SJung-uk Kim 
352*7e382390SJung-uk Kim 	while (fgets (buf, (int) readsz, stdin)) {
353*7e382390SJung-uk Kim 
354*7e382390SJung-uk Kim 		regmatch_t m[10];
355*7e382390SJung-uk Kim 
356*7e382390SJung-uk Kim 		/* Check for #line directive. */
357*7e382390SJung-uk Kim 		if (buf[0] == '#'
358*7e382390SJung-uk Kim 			&& regexec (&regex_linedir, buf, 3, m, 0) == 0) {
359*7e382390SJung-uk Kim 
360*7e382390SJung-uk Kim 			char   *fname;
361*7e382390SJung-uk Kim 
362*7e382390SJung-uk Kim 			/* extract the line number and filename */
363*7e382390SJung-uk Kim 			fname = regmatch_dup (&m[2], buf);
364*7e382390SJung-uk Kim 
365*7e382390SJung-uk Kim 			if (strcmp (fname,
366*7e382390SJung-uk Kim 				outfilename ? outfilename : "<stdout>")
367*7e382390SJung-uk Kim 					== 0
368*7e382390SJung-uk Kim 			 || strcmp (fname,
369*7e382390SJung-uk Kim 			 	headerfilename ? headerfilename : "<stdout>")
370*7e382390SJung-uk Kim 					== 0) {
371*7e382390SJung-uk Kim 
372*7e382390SJung-uk Kim 				char    *s1, *s2;
373*7e382390SJung-uk Kim 				char	filename[MAXLINE];
374*7e382390SJung-uk Kim 
375*7e382390SJung-uk Kim 				s1 = fname;
376*7e382390SJung-uk Kim 				s2 = filename;
377*7e382390SJung-uk Kim 
378*7e382390SJung-uk Kim 				while ((s2 - filename) < (MAXLINE - 1) && *s1) {
379*7e382390SJung-uk Kim 					/* Escape the backslash */
380*7e382390SJung-uk Kim 					if (*s1 == '\\')
381*7e382390SJung-uk Kim 						*s2++ = '\\';
382*7e382390SJung-uk Kim 					/* Escape the double quote */
383*7e382390SJung-uk Kim 					if (*s1 == '\"')
384*7e382390SJung-uk Kim 						*s2++ = '\\';
385*7e382390SJung-uk Kim 					/* Copy the character as usual */
386*7e382390SJung-uk Kim 					*s2++ = *s1++;
387*7e382390SJung-uk Kim 				}
388*7e382390SJung-uk Kim 
389*7e382390SJung-uk Kim 				*s2 = '\0';
390*7e382390SJung-uk Kim 
391*7e382390SJung-uk Kim 				/* Adjust the line directives. */
392*7e382390SJung-uk Kim 				in_gen = true;
393*7e382390SJung-uk Kim 				snprintf (buf, readsz, "#line %d \"%s\"\n",
394*7e382390SJung-uk Kim 					  lineno, filename);
395*7e382390SJung-uk Kim 			}
396*7e382390SJung-uk Kim 			else {
397*7e382390SJung-uk Kim 				/* it's a #line directive for code we didn't write */
398*7e382390SJung-uk Kim 				in_gen = false;
399*7e382390SJung-uk Kim 			}
400*7e382390SJung-uk Kim 
401*7e382390SJung-uk Kim 			free (fname);
402*7e382390SJung-uk Kim 			last_was_blank = false;
403*7e382390SJung-uk Kim 		}
404*7e382390SJung-uk Kim 
405*7e382390SJung-uk Kim 		/* squeeze blank lines from generated code */
406*7e382390SJung-uk Kim 		else if (in_gen
407*7e382390SJung-uk Kim 			 && regexec (&regex_blank_line, buf, 0, NULL,
408*7e382390SJung-uk Kim 				     0) == 0) {
409*7e382390SJung-uk Kim 			if (last_was_blank)
410*7e382390SJung-uk Kim 				continue;
411*7e382390SJung-uk Kim 			else
412*7e382390SJung-uk Kim 				last_was_blank = true;
413*7e382390SJung-uk Kim 		}
414*7e382390SJung-uk Kim 
415*7e382390SJung-uk Kim 		else {
416*7e382390SJung-uk Kim 			/* it's a line of normal, non-empty code. */
417*7e382390SJung-uk Kim 			last_was_blank = false;
418*7e382390SJung-uk Kim 		}
419*7e382390SJung-uk Kim 
420*7e382390SJung-uk Kim 		fputs (buf, stdout);
421*7e382390SJung-uk Kim 		lineno++;
422*7e382390SJung-uk Kim 	}
423*7e382390SJung-uk Kim 	fflush (stdout);
424*7e382390SJung-uk Kim 	if (ferror (stdout))
425*7e382390SJung-uk Kim 		lerr (_("error writing output file %s"),
426*7e382390SJung-uk Kim 			outfilename ? outfilename : "<stdout>");
427*7e382390SJung-uk Kim 
428*7e382390SJung-uk Kim 	else if (fclose (stdout))
429*7e382390SJung-uk Kim 		lerr (_("error closing output file %s"),
430*7e382390SJung-uk Kim 			outfilename ? outfilename : "<stdout>");
431*7e382390SJung-uk Kim 
432*7e382390SJung-uk Kim 	return 0;
433*7e382390SJung-uk Kim }
434*7e382390SJung-uk Kim 
435*7e382390SJung-uk Kim /* vim:set expandtab cindent tabstop=4 softtabstop=4 shiftwidth=4 textwidth=0: */
436