]> git.saurik.com Git - bison.git/blame - src/output.c
(PARAMS): Remove.
[bison.git] / src / output.c
CommitLineData
c3e23647 1/* Output the generated parsing program for bison,
5bb18f9a 2 Copyright (C) 1984, 1986, 1989, 1992, 2000, 2001, 2002
255ef638 3 Free Software Foundation, Inc.
c3e23647 4
9ee3c97b 5 This file is part of Bison, the GNU Compiler Compiler.
c3e23647 6
9ee3c97b
AD
7 Bison is free software; you can redistribute it and/or modify it
8 under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2, or (at your option)
10 any later version.
c3e23647 11
9ee3c97b
AD
12 Bison is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 General Public License for more details.
c3e23647 16
9ee3c97b
AD
17 You should have received a copy of the GNU General Public License
18 along with Bison; see the file COPYING. If not, write to the Free
19 Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
20 02111-1307, USA. */
c3e23647
RS
21
22
c3e23647 23#include "system.h"
14d3eb9b 24#include "quotearg.h"
be2a1a68 25#include "error.h"
573312ac 26#include "subpipe.h"
ceed8467 27#include "getargs.h"
c3e23647
RS
28#include "files.h"
29#include "gram.h"
a0f6b076 30#include "complain.h"
6c89f1c1 31#include "output.h"
a70083a3 32#include "reader.h"
ad949da9 33#include "symtab.h"
c6f1a33c 34#include "tables.h"
11d82f03 35#include "muscle_tab.h"
be2a1a68 36
be2a1a68 37/* From src/scan-skel.l. */
573312ac 38void scan_skel (FILE *);
d019d655 39
12b0043a 40
f87685c3 41static struct obstack format_obstack;
c3e23647 42
c7925b99
MA
43int error_verbose = 0;
44
f0440388 45
133c20e2 46
5372019f
AD
47/*-------------------------------------------------------------------.
48| Create a function NAME which associates to the muscle NAME the |
49| result of formatting the FIRST and then TABLE_DATA[BEGIN..END[ (of |
50| TYPE), and to the muscle NAME_max, the max value of the |
51| TABLE_DATA. |
52`-------------------------------------------------------------------*/
62a3e4f0 53
62a3e4f0 54
5372019f 55#define GENERATE_MUSCLE_INSERT_TABLE(Name, Type) \
5fbb0954 56 \
5372019f
AD
57static void \
58Name (const char *name, \
5fbb0954
AD
59 Type *table_data, \
60 Type first, \
61 int begin, \
62 int end) \
63{ \
12b0043a 64 Type min = first; \
0c2d3f4c 65 Type max = first; \
5fbb0954
AD
66 int i; \
67 int j = 1; \
68 \
5372019f 69 obstack_fgrow1 (&format_obstack, "%6d", first); \
5fbb0954
AD
70 for (i = begin; i < end; ++i) \
71 { \
5372019f 72 obstack_1grow (&format_obstack, ','); \
5fbb0954
AD
73 if (j >= 10) \
74 { \
5372019f 75 obstack_sgrow (&format_obstack, "\n "); \
5fbb0954
AD
76 j = 1; \
77 } \
78 else \
79 ++j; \
5372019f 80 obstack_fgrow1 (&format_obstack, "%6d", table_data[i]); \
12b0043a
AD
81 if (table_data[i] < min) \
82 min = table_data[i]; \
83 if (max < table_data[i]) \
5fbb0954
AD
84 max = table_data[i]; \
85 } \
5372019f
AD
86 obstack_1grow (&format_obstack, 0); \
87 muscle_insert (name, obstack_finish (&format_obstack)); \
5fbb0954 88 \
12b0043a
AD
89 /* Build `NAME_min' and `NAME_max' in the obstack. */ \
90 obstack_fgrow1 (&format_obstack, "%s_min", name); \
91 obstack_1grow (&format_obstack, 0); \
92 MUSCLE_INSERT_LONG_INT (obstack_finish (&format_obstack), \
93 (long int) min); \
5372019f
AD
94 obstack_fgrow1 (&format_obstack, "%s_max", name); \
95 obstack_1grow (&format_obstack, 0); \
0c2d3f4c
AD
96 MUSCLE_INSERT_LONG_INT (obstack_finish (&format_obstack), \
97 (long int) max); \
62a3e4f0
AD
98}
99
5372019f 100GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_unsigned_int_table, unsigned int)
12b0043a 101GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_int_table, int)
5372019f 102GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_short_table, short)
12b0043a
AD
103GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_base_table, base_t)
104GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_rule_number_table, rule_number_t)
a49aecd5 105GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_symbol_number_table, symbol_number_t)
5372019f 106GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_item_number_table, item_number_t)
d57650a5 107GENERATE_MUSCLE_INSERT_TABLE(muscle_insert_state_number_table, state_number_t)
c3e23647
RS
108
109
39912f52
AD
110/*------------------------------------------------------------------.
111| Prepare the muscles related to the symbols: translate, tname, and |
112| toknum. |
113`------------------------------------------------------------------*/
b0940840 114
4a120d45 115static void
39912f52 116prepare_symbols (void)
c3e23647 117{
39912f52
AD
118 MUSCLE_INSERT_INT ("tokens_number", ntokens);
119 MUSCLE_INSERT_INT ("nterms_number", nvars);
120 MUSCLE_INSERT_INT ("undef_token_number", undeftoken->number);
121 MUSCLE_INSERT_INT ("user_token_number_max", max_user_token_number);
122
a49aecd5 123 muscle_insert_symbol_number_table ("translate",
fc5734fe
AD
124 token_translations,
125 token_translations[0],
126 1, max_user_token_number + 1);
c3e23647 127
39912f52 128 /* tname -- token names. */
b0940840
AD
129 {
130 int i;
131 int j = 0;
132 for (i = 0; i < nsyms; i++)
133 {
6b98e4b5
AD
134 /* Be sure not to use twice the same QUOTEARG slot:
135 SYMBOL_TAG_GET uses slot 0. */
b0940840
AD
136 const char *cp =
137 quotearg_n_style (1, c_quoting_style,
97650f4e 138 symbols[i]->tag);
b0940840
AD
139 /* Width of the next token, including the two quotes, the coma
140 and the space. */
141 int strsize = strlen (cp) + 2;
142
143 if (j + strsize > 75)
144 {
145 obstack_sgrow (&format_obstack, "\n ");
146 j = 2;
147 }
148
149 obstack_sgrow (&format_obstack, cp);
150 obstack_sgrow (&format_obstack, ", ");
151 j += strsize;
152 }
153 /* Add a NULL entry to list of tokens (well, 0, as NULL might not be
154 defined). */
155 obstack_sgrow (&format_obstack, "0");
c3e23647 156
b0940840
AD
157 /* Finish table and store. */
158 obstack_1grow (&format_obstack, 0);
159 muscle_insert ("tname", obstack_finish (&format_obstack));
160 }
161
12b0043a 162 /* Output YYTOKNUM. */
b2ed6e58
AD
163 {
164 int i;
3650b4b8
AD
165 int *values = XCALLOC (int, ntokens);
166 for (i = 0; i < ntokens; ++i)
b0940840 167 values[i] = symbols[i]->user_token_number;
12b0043a 168 muscle_insert_int_table ("toknum", values,
3650b4b8 169 values[0], 1, ntokens);
b0940840 170 free (values);
b2ed6e58 171 }
b0940840 172}
b2ed6e58 173
b0940840
AD
174
175/*-------------------------------------------------------------.
176| Prepare the muscles related to the rules: rhs, prhs, r1, r2, |
676385e2 177| rline, dprec, merger |
b0940840
AD
178`-------------------------------------------------------------*/
179
180static void
181prepare_rules (void)
182{
9222837b 183 rule_number_t r;
5df5f6d5 184 unsigned int i = 0;
62a3e4f0 185 item_number_t *rhs = XMALLOC (item_number_t, nritems);
4b3d3a8e
AD
186 unsigned int *prhs = XMALLOC (unsigned int, nrules);
187 unsigned int *rline = XMALLOC (unsigned int, nrules);
188 symbol_number_t *r1 = XMALLOC (symbol_number_t, nrules);
189 unsigned int *r2 = XMALLOC (unsigned int, nrules);
190 short *dprec = XMALLOC (short, nrules);
191 short *merger = XMALLOC (short, nrules);
192
193 for (r = 0; r < nrules; ++r)
b0940840 194 {
9222837b 195 item_number_t *rhsp = NULL;
b0940840
AD
196 /* Index of rule R in RHS. */
197 prhs[r] = i;
198 /* RHS of the rule R. */
199 for (rhsp = rules[r].rhs; *rhsp >= 0; ++rhsp)
200 rhs[i++] = *rhsp;
201 /* LHS of the rule R. */
202 r1[r] = rules[r].lhs->number;
203 /* Length of rule R's RHS. */
204 r2[r] = i - prhs[r];
205 /* Separator in RHS. */
206 rhs[i++] = -1;
207 /* Line where rule was defined. */
8efe435c 208 rline[r] = rules[r].location.first_line;
676385e2
PH
209 /* Dynamic precedence (GLR) */
210 dprec[r] = rules[r].dprec;
211 /* Merger-function index (GLR) */
212 merger[r] = rules[r].merger;
b0940840
AD
213 }
214 assert (i == nritems);
215
5372019f 216 muscle_insert_item_number_table ("rhs", rhs, ritem[0], 1, nritems);
4b3d3a8e
AD
217 muscle_insert_unsigned_int_table ("prhs", prhs, 0, 0, nrules);
218 muscle_insert_unsigned_int_table ("rline", rline, 0, 0, nrules);
219 muscle_insert_symbol_number_table ("r1", r1, 0, 0, nrules);
220 muscle_insert_unsigned_int_table ("r2", r2, 0, 0, nrules);
221 muscle_insert_short_table ("dprec", dprec, 0, 0, nrules);
222 muscle_insert_short_table ("merger", merger, 0, 0, nrules);
796d61fb 223
39912f52
AD
224 MUSCLE_INSERT_INT ("rules_number", nrules);
225
b0940840
AD
226 free (rhs);
227 free (prhs);
5df5f6d5
AD
228 free (rline);
229 free (r1);
b0940840 230 free (r2);
676385e2
PH
231 free (dprec);
232 free (merger);
c3e23647
RS
233}
234
b0940840
AD
235/*--------------------------------------------.
236| Prepare the muscles related to the states. |
237`--------------------------------------------*/
c3e23647 238
4a120d45 239static void
b0940840 240prepare_states (void)
c3e23647 241{
d57650a5 242 state_number_t i;
a49aecd5
AD
243 symbol_number_t *values =
244 (symbol_number_t *) alloca (sizeof (symbol_number_t) * nstates);
9703cc49 245 for (i = 0; i < nstates; ++i)
29e88316 246 values[i] = states[i]->accessing_symbol;
a49aecd5 247 muscle_insert_symbol_number_table ("stos", values,
d57650a5 248 0, 1, nstates);
39912f52
AD
249
250 MUSCLE_INSERT_INT ("last", high);
251 MUSCLE_INSERT_INT ("final_state_number", final_state->number);
252 MUSCLE_INSERT_INT ("states_number", nstates);
c3e23647
RS
253}
254
255
c3e23647 256
bb33f19a
PE
257/*---------------------------------.
258| Output the user actions to OUT. |
259`---------------------------------*/
12b0043a 260
4a120d45 261static void
c6f1a33c 262user_actions_output (FILE *out)
3f96f4dc 263{
9222837b 264 rule_number_t r;
dafdc66f
AD
265
266 fputs ("m4_define([b4_actions], \n[[", out);
4b3d3a8e 267 for (r = 0; r < nrules; ++r)
9222837b 268 if (rules[r].action)
3f96f4dc 269 {
4b3d3a8e 270 fprintf (out, " case %d:\n", r + 1);
3f96f4dc
AD
271
272 if (!no_lines_flag)
ea52d706 273 fprintf (out, muscle_find ("linef"),
9222837b 274 rules[r].action_location.first_line,
ea52d706
AD
275 quotearg_style (c_quoting_style,
276 muscle_find ("filename")));
e9955c83 277 fprintf (out, " %s\n break;\n\n",
9222837b 278 rules[r].action);
3f96f4dc 279 }
dafdc66f 280 fputs ("]])\n\n", out);
3f96f4dc
AD
281}
282
676385e2
PH
283/*--------------------------------------.
284| Output the merge functions to OUT. |
285`--------------------------------------*/
286
41442480 287static void
676385e2
PH
288merger_output (FILE *out)
289{
290 int n;
291 merger_list* p;
292
293 fputs ("m4_define([b4_mergers], \n[[", out);
e0e5bf84 294 for (n = 1, p = merge_functions; p != NULL; n += 1, p = p->next)
676385e2 295 {
e0e5bf84 296 if (p->type[0] == '\0')
676385e2
PH
297 fprintf (out, " case %d: yyval = %s (*yy0, *yy1); break;\n",
298 n, p->name);
299 else
300 fprintf (out, " case %d: yyval.%s = %s (*yy0, *yy1); break;\n",
301 n, p->type, p->name);
302 }
303 fputs ("]])\n\n", out);
304}
3f96f4dc 305
ae7453f2
AD
306/*--------------------------------------.
307| Output the tokens definition to OUT. |
308`--------------------------------------*/
091e20bb 309
c6f1a33c 310static void
be2a1a68 311token_definitions_output (FILE *out)
091e20bb
AD
312{
313 int i;
0d8bed56 314 int first = 1;
dafdc66f
AD
315
316 fputs ("m4_define([b4_tokens], \n[", out);
091e20bb
AD
317 for (i = 0; i < ntokens; ++i)
318 {
db8837cb 319 symbol_t *symbol = symbols[i];
091e20bb
AD
320 int number = symbol->user_token_number;
321
b87f8b21
AD
322 /* At this stage, if there are literal aliases, they are part of
323 SYMBOLS, so we should not find symbols which are the aliases
324 here. */
325 assert (number != USER_NUMBER_ALIAS);
326
091e20bb 327 /* Skip error token. */
007a50a4 328 if (symbol == errtoken)
091e20bb 329 continue;
b87f8b21
AD
330
331 /* If this string has an alias, then it is necessarily the alias
332 which is to be output. */
333 if (symbol->alias)
334 symbol = symbol->alias;
335
336 /* Don't output literal chars or strings (when defined only as a
337 string). Note that must be done after the alias resolution:
338 think about `%token 'f' "f"'. */
339 if (symbol->tag[0] == '\'' || symbol->tag[0] == '\"')
340 continue;
091e20bb
AD
341
342 /* Don't #define nonliteral tokens whose names contain periods
343 or '$' (as does the default value of the EOF token). */
344 if (strchr (symbol->tag, '.') || strchr (symbol->tag, '$'))
345 continue;
346
83ccf991 347 fprintf (out, "%s[[[%s]], [%d]]",
0d8bed56 348 first ? "" : ",\n", symbol->tag, number);
b87f8b21 349
0d8bed56 350 first = 0;
091e20bb 351 }
dafdc66f 352 fputs ("])\n\n", out);
091e20bb
AD
353}
354
355
ae7453f2
AD
356/*---------------------------------------.
357| Output the symbol destructors to OUT. |
358`---------------------------------------*/
9280d3ef
AD
359
360static void
361symbol_destructors_output (FILE *out)
362{
363 int i;
364 int first = 1;
365
366 fputs ("m4_define([b4_symbol_destructors], \n[", out);
367 for (i = 0; i < nsyms; ++i)
368 if (symbols[i]->destructor)
369 {
370 symbol_t *symbol = symbols[i];
371
24c0aad7
AD
372 /* Filename, lineno,
373 Symbol-name, Symbol-number,
374 destructor, typename. */
375 fprintf (out, "%s[[[%s]], [[%d]], [[%s]], [[%d]], [[%s]], [[%s]]]",
9280d3ef 376 first ? "" : ",\n",
24c0aad7 377 infile, symbol->destructor_location.first_line,
97650f4e 378 symbol->tag,
24c0aad7
AD
379 symbol->number,
380 symbol->destructor,
381 symbol->type_name);
9280d3ef
AD
382
383 first = 0;
384 }
385 fputs ("])\n\n", out);
386}
387
388
ae7453f2
AD
389/*------------------------------------.
390| Output the symbol printers to OUT. |
391`------------------------------------*/
366eea36
AD
392
393static void
394symbol_printers_output (FILE *out)
395{
396 int i;
397 int first = 1;
398
399 fputs ("m4_define([b4_symbol_printers], \n[", out);
400 for (i = 0; i < nsyms; ++i)
401 if (symbols[i]->destructor)
402 {
403 symbol_t *symbol = symbols[i];
404
405 /* Filename, lineno,
406 Symbol-name, Symbol-number,
ae7453f2 407 printer, typename. */
366eea36
AD
408 fprintf (out, "%s[[[%s]], [[%d]], [[%s]], [[%d]], [[%s]], [[%s]]]",
409 first ? "" : ",\n",
410 infile, symbol->printer_location.first_line,
97650f4e 411 symbol->tag,
366eea36
AD
412 symbol->number,
413 symbol->printer,
414 symbol->type_name);
415
416 first = 0;
417 }
418 fputs ("])\n\n", out);
419}
420
421
4a120d45 422static void
c6f1a33c 423prepare_actions (void)
6c89f1c1 424{
c6f1a33c
AD
425 /* Figure out the actions for the specified state, indexed by
426 lookahead token type. */
bbb5bcc6 427
c6f1a33c
AD
428 muscle_insert_rule_number_table ("defact", yydefact,
429 yydefact[0], 1, nstates);
6c89f1c1 430
c6f1a33c
AD
431 /* Figure out what to do after reducing with each rule, depending on
432 the saved state from before the beginning of parsing the data
433 that matched this rule. */
d57650a5
AD
434 muscle_insert_state_number_table ("defgoto", yydefgoto,
435 yydefgoto[0], 1, nsyms - ntokens);
12b0043a 436
12b0043a 437
12b0043a
AD
438 /* Output PACT. */
439 muscle_insert_base_table ("pact", base,
5372019f 440 base[0], 1, nstates);
12b0043a 441 MUSCLE_INSERT_INT ("pact_ninf", base_ninf);
c3e23647 442
12b0043a
AD
443 /* Output PGOTO. */
444 muscle_insert_base_table ("pgoto", base,
5372019f 445 base[nstates], nstates + 1, nvectors);
c3e23647 446
12b0043a
AD
447 muscle_insert_base_table ("table", table,
448 table[0], 1, high + 1);
449 MUSCLE_INSERT_INT ("table_ninf", table_ninf);
676385e2 450
12b0043a
AD
451 muscle_insert_base_table ("check", check,
452 check[0], 1, high + 1);
c3e23647 453
ea99527d
AD
454 /* GLR parsing slightly modifies YYTABLE and YYCHECK (and thus
455 YYPACT) so that in states with unresolved conflicts, the default
456 reduction is not used in the conflicted entries, so that there is
457 a place to put a conflict pointer.
458
459 This means that YYCONFLP and YYCONFL are nonsense for a non-GLR
460 parser, so we could avoid accidents by not writing them out in
461 that case. Nevertheless, it seems even better to be able to use
462 the GLR skeletons even without the non-deterministic tables. */
463 muscle_insert_unsigned_int_table ("conflict_list_heads", conflict_table,
39912f52 464 conflict_table[0], 1, high + 1);
ea99527d
AD
465 muscle_insert_unsigned_int_table ("conflicting_rules", conflict_list,
466 conflict_list[0], 1, conflict_list_cnt);
6c89f1c1 467}
c3e23647 468
652def80 469\f
1239777d
AD
470/*---------------------------.
471| Call the skeleton parser. |
472`---------------------------*/
c3e23647 473
4a120d45 474static void
1239777d 475output_skeleton (void)
9b3add5b 476{
573312ac
PE
477 FILE *in;
478 FILE *out;
479 int filter_fd[2];
480 char const *argv[7];
481 pid_t pid;
482
483 /* Compute the names of the package data dir and skeleton file.
484 Test whether m4sugar.m4 is readable, to check for proper
485 installation. A faulty installation can cause deadlock, so a
486 cheap sanity check is worthwhile. */
487 char const m4sugar[] = "m4sugar/m4sugar.m4";
488 char *full_path;
489 char const *p;
490 char const *m4 = (p = getenv ("M4")) ? p : M4;
491 char const *pkgdatadir = (p = getenv ("BISON_PKGDATADIR")) ? p : PKGDATADIR;
492 size_t skeleton_size = strlen (skeleton) + 1;
493 size_t pkgdatadirlen = strlen (pkgdatadir);
494 while (pkgdatadirlen && pkgdatadir[pkgdatadirlen - 1] == '/')
495 pkgdatadirlen--;
496 full_path = xmalloc (pkgdatadirlen + 1
497 + (skeleton_size < sizeof m4sugar
498 ? sizeof m4sugar : skeleton_size));
499 strcpy (full_path, pkgdatadir);
500 full_path[pkgdatadirlen] = '/';
501 strcpy (full_path + pkgdatadirlen + 1, m4sugar);
502 in = fopen (full_path, "r");
503 if (! in || fclose (in) != 0)
504 error (EXIT_FAILURE, errno, "%s", full_path);
505 strcpy (full_path + pkgdatadirlen + 1, skeleton);
506
507 /* Create an m4 subprocess connected to us via two pipes. */
508
509 if (trace_flag & trace_tools)
510 fprintf (stderr, "running: %s -I %s %s - %s\n",
511 m4, pkgdatadir, m4sugar, full_path);
512
513 argv[0] = m4;
514 argv[1] = "-I";
515 argv[2] = pkgdatadir;
516 argv[3] = m4sugar;
517 argv[4] = "-";
518 argv[5] = full_path;
519 argv[6] = NULL;
520
521 init_subpipe ();
522 pid = create_subpipe (argv, filter_fd);
523 free (full_path);
524
525 out = fdopen (filter_fd[0], "w");
526 if (! out)
527 error (EXIT_FAILURE, errno, "fdopen");
528
529 /* Output the definitions of all the muscles. */
381fb12e
AD
530
531 /* There are no comments, especially not `#': we do want M4 expansion
532 after `#': think of CPP macros! */
533 fputs ("m4_changecom()\n", out);
534 fputs ("m4_init()\n", out);
535
c6f1a33c 536 user_actions_output (out);
676385e2 537 merger_output (out);
381fb12e 538 token_definitions_output (out);
9280d3ef 539 symbol_destructors_output (out);
366eea36 540 symbol_printers_output (out);
be2a1a68 541
381fb12e 542 muscles_m4_output (out);
be2a1a68 543
381fb12e
AD
544 fputs ("m4_wrap([m4_divert_pop(0)])\n", out);
545 fputs ("m4_divert_push(0)dnl\n", out);
573312ac
PE
546 if (ferror (out))
547 error (EXIT_FAILURE, 0, "pipe output error");
381fb12e 548 xfclose (out);
be2a1a68 549
573312ac 550 /* Read and process m4's output. */
1509d42f 551 timevar_push (TV_M4);
573312ac
PE
552 in = fdopen (filter_fd[1], "r");
553 if (! in)
554 error (EXIT_FAILURE, errno, "fdopen");
555 scan_skel (in);
556 if (ferror (in))
557 error (EXIT_FAILURE, 0, "pipe input error");
558 xfclose (in);
559 reap_subpipe (pid, m4);
1509d42f 560 timevar_pop (TV_M4);
26f609ff
RA
561}
562
563static void
564prepare (void)
565{
7db2ed2d
AD
566 /* Flags. */
567 MUSCLE_INSERT_INT ("locations_flag", locations_flag);
568 MUSCLE_INSERT_INT ("defines_flag", defines_flag);
569 MUSCLE_INSERT_INT ("error_verbose", error_verbose);
11d82f03 570 MUSCLE_INSERT_INT ("pure", pure_parser);
11d82f03 571 MUSCLE_INSERT_INT ("debug", debug_flag);
11d82f03 572
b85810ae
AD
573 /* FIXME: This is wrong: the muscles should decide whether they hold
574 a copy or not, but the situation is too obscure currently. */
7db2ed2d 575 MUSCLE_INSERT_STRING ("prefix", spec_name_prefix ? spec_name_prefix : "yy");
be2a1a68
AD
576 MUSCLE_INSERT_STRING ("output_infix", output_infix ? output_infix : "");
577 MUSCLE_INSERT_STRING ("output_prefix", short_base_name);
578 MUSCLE_INSERT_STRING ("output_parser_name", parser_file_name);
579 MUSCLE_INSERT_STRING ("output_header_name", spec_defines_file);
b85810ae 580
7db2ed2d 581 /* User Code. */
0dd1580a
RA
582 obstack_1grow (&pre_prologue_obstack, 0);
583 obstack_1grow (&post_prologue_obstack, 0);
584 muscle_insert ("pre_prologue", obstack_finish (&pre_prologue_obstack));
585 muscle_insert ("post_prologue", obstack_finish (&post_prologue_obstack));
381fb12e
AD
586
587 /* Find the right skeleton file. */
588 if (!skeleton)
676385e2
PH
589 {
590 if (glr_parser)
591 skeleton = "glr.c";
592 else
593 skeleton = "yacc.c";
594 }
381fb12e
AD
595
596 /* Parse the skeleton file and output the needed parsers. */
66d30cd4 597 MUSCLE_INSERT_STRING ("skeleton", skeleton);
26f609ff 598}
c3e23647 599
93ede233 600
6c89f1c1
AD
601/*----------------------------------------------------------.
602| Output the parsing tables and the parser code to ftable. |
603`----------------------------------------------------------*/
c3e23647 604
6c89f1c1
AD
605void
606output (void)
607{
f87685c3 608 obstack_init (&format_obstack);
dd60faec 609
39912f52 610 prepare_symbols ();
b0940840
AD
611 prepare_rules ();
612 prepare_states ();
536545f3 613 prepare_actions ();
342b8b6e 614
26f609ff 615 prepare ();
652def80 616
9b3add5b
RA
617 /* Process the selected skeleton file. */
618 output_skeleton ();
619
f499b062 620 obstack_free (&format_obstack, NULL);
0dd1580a
RA
621 obstack_free (&pre_prologue_obstack, NULL);
622 obstack_free (&post_prologue_obstack, NULL);
c3e23647 623}