1 /* Bison Grammar Parser                             -*- C -*-
 
   2    Copyright (C) 2002 Free Software Foundation, Inc.
 
   4    This file is part of Bison, the GNU Compiler Compiler.
 
   6    This program is free software; you can redistribute it and/or modify
 
   7    it under the terms of the GNU General Public License as published by
 
   8    the Free Software Foundation; either version 2 of the License, or
 
   9    (at your option) any later version.
 
  11    This program is distributed in the hope that it will be useful,
 
  12    but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  14    GNU General Public License for more details.
 
  16    You should have received a copy of the GNU General Public License
 
  17    along with this program; if not, write to the Free Software
 
  18    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
 
  33 #include "muscle_tab.h"
 
  40 #include "conflicts.h"
 
  42 /* Produce verbose parse errors.  */
 
  43 #define YYERROR_VERBOSE 1
 
  44 #define YYLLOC_DEFAULT(Current, Rhs, N)                 \
 
  48     Current.first_column  = Rhs[1].first_column;        \
 
  49     Current.first_line    = Rhs[1].first_line;          \
 
  50     Current.last_column   = Rhs[N].last_column;         \
 
  51     Current.last_line     = Rhs[N].last_line;           \
 
  59 /* Pass the control structure to YYPARSE and YYLEX. */
 
  60 #define YYPARSE_PARAM gram_control
 
  61 #define YYLEX_PARAM gram_control
 
  62 /* YYPARSE receives GRAM_CONTROL as a void *.  Provide a
 
  63    correctly typed access to it.  */
 
  64 #define yycontrol ((gram_control_t *) gram_control)
 
  66 /* Request detailed parse error messages, and pass them to
 
  69 #define yyerror(Msg) \
 
  70         gram_error (yycontrol, &yylloc, Msg)
 
  72 #define YYPRINT(File, Type, Value) \
 
  73         yyprint (File, Type, &Value)
 
  74 static void yyprint (FILE *file, int type, const yystype *value);
 
  76 symbol_class current_class = unknown_sym;
 
  77 char *current_type = 0;
 
  78 symbol_t *current_lhs;
 
  79 location_t current_lhs_location;
 
  80 associativity current_assoc;
 
  82 braced_code_t current_braced_code = action_braced_code;
 
  86 /* Only NUMBERS have a value.  */
 
  96 /* Define the tokens together with there human representation. */
 
  97 %token GRAM_EOF 0 "end of string"
 
  98 %token STRING CHARACTER
 
 101 %token PERCENT_TOKEN       "%token"
 
 102 %token PERCENT_NTERM       "%nterm"
 
 104 %token PERCENT_TYPE        "%type"
 
 105 %token PERCENT_DESTRUCTOR  "%destructor"
 
 106 %token PERCENT_PRINTER     "%printer"
 
 108 %token PERCENT_UNION       "%union"
 
 110 %token PERCENT_LEFT        "%left"
 
 111 %token PERCENT_RIGHT       "%right"
 
 112 %token PERCENT_NONASSOC    "%nonassoc"
 
 114 %token PERCENT_EXPECT "%expect"
 
 115 %token PERCENT_START "%start"
 
 116 %token PERCENT_PREC     "%prec"
 
 117 %token PERCENT_VERBOSE  "%verbose"
 
 118 %token PERCENT_ERROR_VERBOSE "%error-verbose"
 
 120 %token PERCENT_OUTPUT "%output"
 
 121 %token PERCENT_FILE_PREFIX "%file-prefix"
 
 122 %token PERCENT_NAME_PREFIX "%name-prefix"
 
 124 %token PERCENT_DEFINE "%define"
 
 125 %token PERCENT_PURE_PARSER "%pure-parser"
 
 127 %token PERCENT_DEFINES "%defines"
 
 129 %token PERCENT_YACC "%yacc"
 
 131 %token PERCENT_DEBUG "%debug"
 
 132 %token PERCENT_LOCATIONS "%locations"
 
 133 %token PERCENT_NO_LINES "%no-lines"
 
 134 %token PERCENT_SKELETON "%skeleton"
 
 135 %token PERCENT_TOKEN_TABLE "%token-table"
 
 142 %token ID "identifier"
 
 143 %token PERCENT_PERCENT "%%"
 
 144 %token PROLOGUE EPILOGUE
 
 147 %type <string> CHARACTER TYPE STRING string_content
 
 148                BRACED_CODE PROLOGUE EPILOGUE epilogue.opt action
 
 150 %type <symbol> ID symbol string_as_id
 
 151 %type <assoc> precedence_declarator
 
 152 %type <list>  symbols.1
 
 156   declarations "%%" grammar epilogue.opt
 
 158       yycontrol->errcode = 0;
 
 159       epilogue_set ($4, @4);
 
 164         /*------------------------------------.
 
 165         | Declarations: before the first %%.  |
 
 166         `------------------------------------*/
 
 170 | declarations declaration semi_colon.opt
 
 175 | PROLOGUE                                 { prologue_augment ($1, @1); }
 
 176 | "%debug"                                 { debug_flag = 1; }
 
 177 | "%define" string_content string_content  { muscle_insert ($2, $3); }
 
 178 | "%defines"                               { defines_flag = 1; }
 
 179 | "%error-verbose"                         { error_verbose = 1; }
 
 180 | "%expect" INT                            { expected_conflicts = $2; }
 
 181 | "%file-prefix" "=" string_content        { spec_file_prefix = $3; }
 
 182 | "%locations"                             { locations_flag = 1; }
 
 183 | "%name-prefix" "=" string_content        { spec_name_prefix = $3; }
 
 184 | "%no-lines"                              { no_lines_flag = 1; }
 
 185 | "%output" "=" string_content             { spec_outfile = $3; }
 
 186 | "%pure-parser"                           { pure_parser = 1; }
 
 187 | "%skeleton" string_content               { skeleton = $2; }
 
 188 | "%token-table"                           { token_table_flag = 1; }
 
 189 | "%verbose"                               { report_flag = 1; }
 
 190 | "%yacc"                                  { yacc_flag = 1; }
 
 194   precedence_declaration
 
 198       grammar_start_symbol_set ($2, @2);
 
 200 | "%union" BRACED_CODE
 
 203       MUSCLE_INSERT_INT ("stype_line", @2.first_line);
 
 204       muscle_insert ("stype", $2);
 
 207     { current_braced_code = destructor_braced_code; }
 
 208   BRACED_CODE symbols.1
 
 211       for (list = $4; list; list = list->next)
 
 212         symbol_destructor_set (list->sym, $3, @3);
 
 213       symbol_list_free ($4);
 
 214       current_braced_code = action_braced_code;
 
 217     { current_braced_code = printer_braced_code; }
 
 218   BRACED_CODE symbols.1
 
 221       for (list = $4; list; list = list->next)
 
 222         symbol_printer_set (list->sym, $3, list->location);
 
 223       symbol_list_free ($4);
 
 224       current_braced_code = action_braced_code;
 
 229   "%nterm" { current_class = nterm_sym; } symbol_defs.1
 
 231       current_class = unknown_sym;
 
 234 | "%token" { current_class = token_sym; } symbol_defs.1
 
 236       current_class = unknown_sym;
 
 239 | "%type" TYPE symbols.1
 
 242       for (list = $3; list; list = list->next)
 
 243         symbol_type_set (list->sym, $2, @2);
 
 244       symbol_list_free ($3);
 
 248 precedence_declaration:
 
 249   precedence_declarator type.opt symbols.1
 
 253       for (list = $3; list; list = list->next)
 
 255           symbol_type_set (list->sym, current_type, @2);
 
 256           symbol_precedence_set (list->sym, current_prec, $1, @1);
 
 258       symbol_list_free ($3);
 
 263 precedence_declarator:
 
 264   "%left"     { $$ = left_assoc; }
 
 265 | "%right"    { $$ = right_assoc; }
 
 266 | "%nonassoc" { $$ = non_assoc; }
 
 270   /* Nothing. */ { current_type = NULL;}
 
 271 | TYPE           { current_type = $1; }
 
 274 /* One or more nonterminals to be %typed. */
 
 277   symbol            { $$ = symbol_list_new ($1, @1); }
 
 278 | symbols.1 symbol  { $$ = symbol_list_prepend ($1, $2, @2); }
 
 281 /* One token definition.  */
 
 289        symbol_class_set ($1, current_class, @1);
 
 290        symbol_type_set ($1, current_type, @1);
 
 294       symbol_class_set ($1, current_class, @1);
 
 295       symbol_type_set ($1, current_type, @1);
 
 296       symbol_user_token_number_set ($1, $2, @2);
 
 300       symbol_class_set ($1, current_class, @1);
 
 301       symbol_type_set ($1, current_type, @1);
 
 302       symbol_make_alias ($1, $2);
 
 304 | ID INT string_as_id
 
 306       symbol_class_set ($1, current_class, @1);
 
 307       symbol_type_set ($1, current_type, @1);
 
 308       symbol_user_token_number_set ($1, $2, @2);
 
 309       symbol_make_alias ($1, $3);
 
 313 /* One or more symbol definitions. */
 
 317 | symbol_defs.1 symbol_def
 
 322         /*------------------------------------------.
 
 323         | The grammar section: between the two %%.  |
 
 324         `------------------------------------------*/
 
 327   rules_or_grammar_declaration
 
 328 | grammar rules_or_grammar_declaration
 
 331 /* As a Bison extension, one can use the grammar declarations in the
 
 332    body of the grammar.  But to remain LALR(1), they must be ended
 
 333    with a semi-colon.  */
 
 334 rules_or_grammar_declaration:
 
 336 | grammar_declaration ";"
 
 340   ID ":" { current_lhs = $1; current_lhs_location = @1; } rhses.1 ";"
 
 345   rhs                { grammar_rule_end (@1); }
 
 346 | rhses.1 "|" rhs    { grammar_rule_end (@3); }
 
 351     { grammar_rule_begin (current_lhs, current_lhs_location); }
 
 353     { grammar_current_rule_symbol_append ($2, @2); }
 
 355     { grammar_current_rule_action_append ($2, @2); }
 
 357     { grammar_current_rule_prec_set ($3, @3); }
 
 362 | string_as_id    { $$ = $1; }
 
 363 | CHARACTER       { $$ = getsym ($1, @1); }
 
 371 /* A string used as an ID: we have to keep the quotes. */
 
 375       $$ = getsym ($1, @1);
 
 376       symbol_class_set ($$, token_sym, @1);
 
 380 /* A string used for its contents.  Strip the quotes. */
 
 385       $$[strlen ($$) - 1] = '\0';
 
 405 /*------------------------------------------------------------------.
 
 406 | When debugging the parser, display tokens' locations and values.  |
 
 407 `------------------------------------------------------------------*/
 
 411          int type, const yystype *value)
 
 417       fprintf (file, " = '%s'", value->string);
 
 421       fprintf (file, " = %s", value->symbol->tag);
 
 425       fprintf (file, " = %d", value->integer);
 
 429       fprintf (file, " = \"%s\"", value->string);
 
 433       fprintf (file, " = <%s>", value->string);
 
 439       fprintf (file, " = {{ %s }}", value->string);
 
 445 gram_error (gram_control_t *control ATTRIBUTE_UNUSED,
 
 446             location_t *yylloc, const char *msg)
 
 448   LOCATION_PRINT (stderr, *yylloc);
 
 449   fprintf (stderr, ": %s\n", msg);