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 (®ex_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 (®ex_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