diff options
Diffstat (limited to 'src/tools/lcc/lburg')
| -rw-r--r-- | src/tools/lcc/lburg/gram.c | 1817 | ||||
| -rw-r--r-- | src/tools/lcc/lburg/gram.y | 202 | ||||
| -rw-r--r-- | src/tools/lcc/lburg/lburg.1 | 179 | ||||
| -rw-r--r-- | src/tools/lcc/lburg/lburg.c | 671 | ||||
| -rw-r--r-- | src/tools/lcc/lburg/lburg.h | 65 | 
5 files changed, 2934 insertions, 0 deletions
diff --git a/src/tools/lcc/lburg/gram.c b/src/tools/lcc/lburg/gram.c new file mode 100644 index 0000000..79a9d84 --- /dev/null +++ b/src/tools/lcc/lburg/gram.c @@ -0,0 +1,1817 @@ +/* A Bison parser, made by GNU Bison 2.4.2.  */ + +/* Skeleton implementation for Bison's Yacc-like parsers in C +    +      Copyright (C) 1984, 1989-1990, 2000-2006, 2009-2010 Free Software +   Foundation, Inc. +    +   This program is free software: you can redistribute it and/or modify +   it under the terms of the GNU General Public License as published by +   the Free Software Foundation, either version 3 of the License, or +   (at your option) any later version. +    +   This program is distributed in the hope that it will be useful, +   but WITHOUT ANY WARRANTY; without even the implied warranty of +   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +   GNU General Public License for more details. +    +   You should have received a copy of the GNU General Public License +   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ + +/* As a special exception, you may create a larger work that contains +   part or all of the Bison parser skeleton and distribute that work +   under terms of your choice, so long as that work isn't itself a +   parser generator using the skeleton or a modified version thereof +   as a parser skeleton.  Alternatively, if you modify or redistribute +   the parser skeleton itself, you may (at your option) remove this +   special exception, which will cause the skeleton and the resulting +   Bison output files to be licensed under the GNU General Public +   License without this special exception. +    +   This special exception was added by the Free Software Foundation in +   version 2.2 of Bison.  */ + +/* C LALR(1) parser skeleton written by Richard Stallman, by +   simplifying the original so-called "semantic" parser.  */ + +/* All symbols defined below should begin with yy or YY, to avoid +   infringing on user name space.  This should be done even for local +   variables, as they might otherwise be expanded by user macros. +   There are some unavoidable exceptions within include files to +   define necessary library symbols; they are noted "INFRINGES ON +   USER NAME SPACE" below.  */ + +/* Identify Bison output.  */ +#define YYBISON 1 + +/* Bison version.  */ +#define YYBISON_VERSION "2.4.2" + +/* Skeleton name.  */ +#define YYSKELETON_NAME "yacc.c" + +/* Pure parsers.  */ +#define YYPURE 0 + +/* Push parsers.  */ +#define YYPUSH 0 + +/* Pull parsers.  */ +#define YYPULL 1 + +/* Using locations.  */ +#define YYLSP_NEEDED 0 + + + +/* Copy the first part of user declarations.  */ + +/* Line 189 of yacc.c  */ +#line 1 "src/tools/lcc/lburg/gram.y" + +#include <stdio.h> +#include "lburg.h" +static char rcsid[] = "$Id: gram.y 145 2001-10-17 21:53:10Z timo $"; +/*lint -e616 -e527 -e652 -esym(552,yynerrs) -esym(563,yynewstate,yyerrlab) */ +static int yylineno = 0; + + +/* Line 189 of yacc.c  */ +#line 81 "y.tab.c" + +/* Enabling traces.  */ +#ifndef YYDEBUG +# define YYDEBUG 0 +#endif + +/* Enabling verbose error messages.  */ +#ifdef YYERROR_VERBOSE +# undef YYERROR_VERBOSE +# define YYERROR_VERBOSE 1 +#else +# define YYERROR_VERBOSE 0 +#endif + +/* Enabling the token table.  */ +#ifndef YYTOKEN_TABLE +# define YYTOKEN_TABLE 0 +#endif + + +/* Tokens.  */ +#ifndef YYTOKENTYPE +# define YYTOKENTYPE +   /* Put the tokens into the symbol table, so that GDB and other debuggers +      know about them.  */ +   enum yytokentype { +     TERMINAL = 258, +     START = 259, +     PPERCENT = 260, +     ID = 261, +     TEMPLATE = 262, +     CODE = 263, +     INT = 264 +   }; +#endif +/* Tokens.  */ +#define TERMINAL 258 +#define START 259 +#define PPERCENT 260 +#define ID 261 +#define TEMPLATE 262 +#define CODE 263 +#define INT 264 + + + + +#if ! defined YYSTYPE && ! defined YYSTYPE_IS_DECLARED +typedef union YYSTYPE +{ + +/* Line 214 of yacc.c  */ +#line 8 "src/tools/lcc/lburg/gram.y" + +	int n; +	char *string; +	Tree tree; + + + +/* Line 214 of yacc.c  */ +#line 143 "y.tab.c" +} YYSTYPE; +# define YYSTYPE_IS_TRIVIAL 1 +# define yystype YYSTYPE /* obsolescent; will be withdrawn */ +# define YYSTYPE_IS_DECLARED 1 +#endif + + +/* Copy the second part of user declarations.  */ + + +/* Line 264 of yacc.c  */ +#line 155 "y.tab.c" + +#ifdef short +# undef short +#endif + +#ifdef YYTYPE_UINT8 +typedef YYTYPE_UINT8 yytype_uint8; +#else +typedef unsigned char yytype_uint8; +#endif + +#ifdef YYTYPE_INT8 +typedef YYTYPE_INT8 yytype_int8; +#elif (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +typedef signed char yytype_int8; +#else +typedef short int yytype_int8; +#endif + +#ifdef YYTYPE_UINT16 +typedef YYTYPE_UINT16 yytype_uint16; +#else +typedef unsigned short int yytype_uint16; +#endif + +#ifdef YYTYPE_INT16 +typedef YYTYPE_INT16 yytype_int16; +#else +typedef short int yytype_int16; +#endif + +#ifndef YYSIZE_T +# ifdef __SIZE_TYPE__ +#  define YYSIZE_T __SIZE_TYPE__ +# elif defined size_t +#  define YYSIZE_T size_t +# elif ! defined YYSIZE_T && (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +#  include <stddef.h> /* INFRINGES ON USER NAME SPACE */ +#  define YYSIZE_T size_t +# else +#  define YYSIZE_T unsigned int +# endif +#endif + +#define YYSIZE_MAXIMUM ((YYSIZE_T) -1) + +#ifndef YY_ +# if defined YYENABLE_NLS && YYENABLE_NLS +#  if ENABLE_NLS +#   include <libintl.h> /* INFRINGES ON USER NAME SPACE */ +#   define YY_(msgid) dgettext ("bison-runtime", msgid) +#  endif +# endif +# ifndef YY_ +#  define YY_(msgid) msgid +# endif +#endif + +/* Suppress unused-variable warnings by "using" E.  */ +#if ! defined lint || defined __GNUC__ +# define YYUSE(e) ((void) (e)) +#else +# define YYUSE(e) /* empty */ +#endif + +/* Identity function, used to suppress warnings about constant conditions.  */ +#ifndef lint +# define YYID(n) (n) +#else +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static int +YYID (int yyi) +#else +static int +YYID (yyi) +    int yyi; +#endif +{ +  return yyi; +} +#endif + +#if ! defined yyoverflow || YYERROR_VERBOSE + +/* The parser invokes alloca or malloc; define the necessary symbols.  */ + +# ifdef YYSTACK_USE_ALLOCA +#  if YYSTACK_USE_ALLOCA +#   ifdef __GNUC__ +#    define YYSTACK_ALLOC __builtin_alloca +#   elif defined __BUILTIN_VA_ARG_INCR +#    include <alloca.h> /* INFRINGES ON USER NAME SPACE */ +#   elif defined _AIX +#    define YYSTACK_ALLOC __alloca +#   elif defined _MSC_VER +#    include <malloc.h> /* INFRINGES ON USER NAME SPACE */ +#    define alloca _alloca +#   else +#    define YYSTACK_ALLOC alloca +#    if ! defined _ALLOCA_H && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +#     include <stdlib.h> /* INFRINGES ON USER NAME SPACE */ +#     ifndef _STDLIB_H +#      define _STDLIB_H 1 +#     endif +#    endif +#   endif +#  endif +# endif + +# ifdef YYSTACK_ALLOC +   /* Pacify GCC's `empty if-body' warning.  */ +#  define YYSTACK_FREE(Ptr) do { /* empty */; } while (YYID (0)) +#  ifndef YYSTACK_ALLOC_MAXIMUM +    /* The OS might guarantee only one guard page at the bottom of the stack, +       and a page size can be as small as 4096 bytes.  So we cannot safely +       invoke alloca (N) if N exceeds 4096.  Use a slightly smaller number +       to allow for a few compiler-allocated temporary stack slots.  */ +#   define YYSTACK_ALLOC_MAXIMUM 4032 /* reasonable circa 2006 */ +#  endif +# else +#  define YYSTACK_ALLOC YYMALLOC +#  define YYSTACK_FREE YYFREE +#  ifndef YYSTACK_ALLOC_MAXIMUM +#   define YYSTACK_ALLOC_MAXIMUM YYSIZE_MAXIMUM +#  endif +#  if (defined __cplusplus && ! defined _STDLIB_H \ +       && ! ((defined YYMALLOC || defined malloc) \ +	     && (defined YYFREE || defined free))) +#   include <stdlib.h> /* INFRINGES ON USER NAME SPACE */ +#   ifndef _STDLIB_H +#    define _STDLIB_H 1 +#   endif +#  endif +#  ifndef YYMALLOC +#   define YYMALLOC malloc +#   if ! defined malloc && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +void *malloc (YYSIZE_T); /* INFRINGES ON USER NAME SPACE */ +#   endif +#  endif +#  ifndef YYFREE +#   define YYFREE free +#   if ! defined free && ! defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +void free (void *); /* INFRINGES ON USER NAME SPACE */ +#   endif +#  endif +# endif +#endif /* ! defined yyoverflow || YYERROR_VERBOSE */ + + +#if (! defined yyoverflow \ +     && (! defined __cplusplus \ +	 || (defined YYSTYPE_IS_TRIVIAL && YYSTYPE_IS_TRIVIAL))) + +/* A type that is properly aligned for any stack member.  */ +union yyalloc +{ +  yytype_int16 yyss_alloc; +  YYSTYPE yyvs_alloc; +}; + +/* The size of the maximum gap between one aligned stack and the next.  */ +# define YYSTACK_GAP_MAXIMUM (sizeof (union yyalloc) - 1) + +/* The size of an array large to enough to hold all stacks, each with +   N elements.  */ +# define YYSTACK_BYTES(N) \ +     ((N) * (sizeof (yytype_int16) + sizeof (YYSTYPE)) \ +      + YYSTACK_GAP_MAXIMUM) + +/* Copy COUNT objects from FROM to TO.  The source and destination do +   not overlap.  */ +# ifndef YYCOPY +#  if defined __GNUC__ && 1 < __GNUC__ +#   define YYCOPY(To, From, Count) \ +      __builtin_memcpy (To, From, (Count) * sizeof (*(From))) +#  else +#   define YYCOPY(To, From, Count)		\ +      do					\ +	{					\ +	  YYSIZE_T yyi;				\ +	  for (yyi = 0; yyi < (Count); yyi++)	\ +	    (To)[yyi] = (From)[yyi];		\ +	}					\ +      while (YYID (0)) +#  endif +# endif + +/* Relocate STACK from its old location to the new one.  The +   local variables YYSIZE and YYSTACKSIZE give the old and new number of +   elements in the stack, and YYPTR gives the new location of the +   stack.  Advance YYPTR to a properly aligned location for the next +   stack.  */ +# define YYSTACK_RELOCATE(Stack_alloc, Stack)				\ +    do									\ +      {									\ +	YYSIZE_T yynewbytes;						\ +	YYCOPY (&yyptr->Stack_alloc, Stack, yysize);			\ +	Stack = &yyptr->Stack_alloc;					\ +	yynewbytes = yystacksize * sizeof (*Stack) + YYSTACK_GAP_MAXIMUM; \ +	yyptr += yynewbytes / sizeof (*yyptr);				\ +      }									\ +    while (YYID (0)) + +#endif + +/* YYFINAL -- State number of the termination state.  */ +#define YYFINAL  3 +/* YYLAST -- Last index in YYTABLE.  */ +#define YYLAST   35 + +/* YYNTOKENS -- Number of terminals.  */ +#define YYNTOKENS  16 +/* YYNNTS -- Number of nonterminals.  */ +#define YYNNTS  9 +/* YYNRULES -- Number of rules.  */ +#define YYNRULES  20 +/* YYNRULES -- Number of states.  */ +#define YYNSTATES  37 + +/* YYTRANSLATE(YYLEX) -- Bison symbol number corresponding to YYLEX.  */ +#define YYUNDEFTOK  2 +#define YYMAXUTOK   264 + +#define YYTRANSLATE(YYX)						\ +  ((unsigned int) (YYX) <= YYMAXUTOK ? yytranslate[YYX] : YYUNDEFTOK) + +/* YYTRANSLATE[YYLEX] -- Bison symbol number corresponding to YYLEX.  */ +static const yytype_uint8 yytranslate[] = +{ +       0,     2,     2,     2,     2,     2,     2,     2,     2,     2, +      10,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +      13,    14,     2,     2,    15,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,    12,     2, +       2,    11,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     2,     2,     2,     2, +       2,     2,     2,     2,     2,     2,     1,     2,     3,     4, +       5,     6,     7,     8,     9 +}; + +#if YYDEBUG +/* YYPRHS[YYN] -- Index of the first RHS symbol of rule number YYN in +   YYRHS.  */ +static const yytype_uint8 yyprhs[] = +{ +       0,     0,     3,     7,     9,    10,    13,    17,    21,    23, +      26,    27,    32,    33,    41,    44,    48,    50,    52,    57, +      64 +}; + +/* YYRHS -- A `-1'-separated list of the rules' RHS.  */ +static const yytype_int8 yyrhs[] = +{ +      17,     0,    -1,    18,     5,    21,    -1,    18,    -1,    -1, +      18,    19,    -1,     3,    20,    10,    -1,     4,    22,    10, +      -1,    10,    -1,     1,    10,    -1,    -1,    20,     6,    11, +       9,    -1,    -1,    21,    22,    12,    23,     7,    24,    10, +      -1,    21,    10,    -1,    21,     1,    10,    -1,     6,    -1, +       6,    -1,     6,    13,    23,    14,    -1,     6,    13,    23, +      15,    23,    14,    -1,     8,    -1 +}; + +/* YYRLINE[YYN] -- source line where rule number YYN was defined.  */ +static const yytype_uint8 yyrline[] = +{ +       0,    22,    22,    23,    26,    27,    30,    31,    35,    36, +      39,    40,    43,    44,    45,    46,    49,    52,    53,    54, +      57 +}; +#endif + +#if YYDEBUG || YYERROR_VERBOSE || YYTOKEN_TABLE +/* YYTNAME[SYMBOL-NUM] -- String name of the symbol SYMBOL-NUM. +   First, the terminals, then, starting at YYNTOKENS, nonterminals.  */ +static const char *const yytname[] = +{ +  "$end", "error", "$undefined", "TERMINAL", "START", "PPERCENT", "ID", +  "TEMPLATE", "CODE", "INT", "'\\n'", "'='", "':'", "'('", "')'", "','", +  "$accept", "spec", "decls", "decl", "blist", "rules", "nonterm", "tree", +  "cost", 0 +}; +#endif + +# ifdef YYPRINT +/* YYTOKNUM[YYLEX-NUM] -- Internal token number corresponding to +   token YYLEX-NUM.  */ +static const yytype_uint16 yytoknum[] = +{ +       0,   256,   257,   258,   259,   260,   261,   262,   263,   264, +      10,    61,    58,    40,    41,    44 +}; +# endif + +/* YYR1[YYN] -- Symbol number of symbol that rule YYN derives.  */ +static const yytype_uint8 yyr1[] = +{ +       0,    16,    17,    17,    18,    18,    19,    19,    19,    19, +      20,    20,    21,    21,    21,    21,    22,    23,    23,    23, +      24 +}; + +/* YYR2[YYN] -- Number of symbols composing right hand side of rule YYN.  */ +static const yytype_uint8 yyr2[] = +{ +       0,     2,     3,     1,     0,     2,     3,     3,     1,     2, +       0,     4,     0,     7,     2,     3,     1,     1,     4,     6, +       1 +}; + +/* YYDEFACT[STATE-NAME] -- Default rule to reduce with in state +   STATE-NUM when YYTABLE doesn't specify something else to do.  Zero +   means the default is an error.  */ +static const yytype_uint8 yydefact[] = +{ +       4,     0,     0,     1,     0,    10,     0,    12,     8,     5, +       9,     0,    16,     0,     0,     0,     6,     7,     0,    14, +       0,     0,    15,     0,    11,    17,     0,     0,     0,     0, +      20,     0,    18,     0,    13,     0,    19 +}; + +/* YYDEFGOTO[NTERM-NUM].  */ +static const yytype_int8 yydefgoto[] = +{ +      -1,     1,     2,     9,    11,    14,    13,    26,    31 +}; + +/* YYPACT[STATE-NUM] -- Index in YYTABLE of the portion describing +   STATE-NUM.  */ +#define YYPACT_NINF -26 +static const yytype_int8 yypact[] = +{ +     -26,    11,     0,   -26,     5,   -26,     8,   -26,   -26,   -26, +     -26,     3,   -26,     7,     6,     9,   -26,   -26,    12,   -26, +      13,    14,   -26,    15,   -26,    16,    17,    15,    18,     4, +     -26,    20,   -26,    15,   -26,    19,   -26 +}; + +/* YYPGOTO[NTERM-NUM].  */ +static const yytype_int8 yypgoto[] = +{ +     -26,   -26,   -26,   -26,   -26,   -26,    21,   -25,   -26 +}; + +/* YYTABLE[YYPACT[STATE-NUM]].  What to do in state STATE-NUM.  If +   positive, shift that token.  If negative, reduce the rule which +   number is the opposite.  If zero, do what YYDEFACT says. +   If YYTABLE_NINF, syntax error.  */ +#define YYTABLE_NINF -4 +static const yytype_int8 yytable[] = +{ +      -3,     4,    29,     5,     6,     7,    -2,    18,    35,    15, +       8,     3,    12,    16,    12,    10,    19,    17,    32,    33, +      21,    25,    22,    24,    28,    23,    30,     0,     0,    27, +      34,     0,     0,    36,     0,    20 +}; + +static const yytype_int8 yycheck[] = +{ +       0,     1,    27,     3,     4,     5,     0,     1,    33,     6, +      10,     0,     6,    10,     6,    10,    10,    10,    14,    15, +      11,     6,    10,     9,     7,    12,     8,    -1,    -1,    13, +      10,    -1,    -1,    14,    -1,    14 +}; + +/* YYSTOS[STATE-NUM] -- The (internal number of the) accessing +   symbol of state STATE-NUM.  */ +static const yytype_uint8 yystos[] = +{ +       0,    17,    18,     0,     1,     3,     4,     5,    10,    19, +      10,    20,     6,    22,    21,     6,    10,    10,     1,    10, +      22,    11,    10,    12,     9,     6,    23,    13,     7,    23, +       8,    24,    14,    15,    10,    23,    14 +}; + +#define yyerrok		(yyerrstatus = 0) +#define yyclearin	(yychar = YYEMPTY) +#define YYEMPTY		(-2) +#define YYEOF		0 + +#define YYACCEPT	goto yyacceptlab +#define YYABORT		goto yyabortlab +#define YYERROR		goto yyerrorlab + + +/* Like YYERROR except do call yyerror.  This remains here temporarily +   to ease the transition to the new meaning of YYERROR, for GCC. +   Once GCC version 2 has supplanted version 1, this can go.  However, +   YYFAIL appears to be in use.  Nevertheless, it is formally deprecated +   in Bison 2.4.2's NEWS entry, where a plan to phase it out is +   discussed.  */ + +#define YYFAIL		goto yyerrlab +#if defined YYFAIL +  /* This is here to suppress warnings from the GCC cpp's +     -Wunused-macros.  Normally we don't worry about that warning, but +     some users do, and we want to make it easy for users to remove +     YYFAIL uses, which will produce warnings from Bison 2.5.  */ +#endif + +#define YYRECOVERING()  (!!yyerrstatus) + +#define YYBACKUP(Token, Value)					\ +do								\ +  if (yychar == YYEMPTY && yylen == 1)				\ +    {								\ +      yychar = (Token);						\ +      yylval = (Value);						\ +      yytoken = YYTRANSLATE (yychar);				\ +      YYPOPSTACK (1);						\ +      goto yybackup;						\ +    }								\ +  else								\ +    {								\ +      yyerror (YY_("syntax error: cannot back up")); \ +      YYERROR;							\ +    }								\ +while (YYID (0)) + + +#define YYTERROR	1 +#define YYERRCODE	256 + + +/* YYLLOC_DEFAULT -- Set CURRENT to span from RHS[1] to RHS[N]. +   If N is 0, then set CURRENT to the empty location which ends +   the previous symbol: RHS[0] (always defined).  */ + +#define YYRHSLOC(Rhs, K) ((Rhs)[K]) +#ifndef YYLLOC_DEFAULT +# define YYLLOC_DEFAULT(Current, Rhs, N)				\ +    do									\ +      if (YYID (N))                                                    \ +	{								\ +	  (Current).first_line   = YYRHSLOC (Rhs, 1).first_line;	\ +	  (Current).first_column = YYRHSLOC (Rhs, 1).first_column;	\ +	  (Current).last_line    = YYRHSLOC (Rhs, N).last_line;		\ +	  (Current).last_column  = YYRHSLOC (Rhs, N).last_column;	\ +	}								\ +      else								\ +	{								\ +	  (Current).first_line   = (Current).last_line   =		\ +	    YYRHSLOC (Rhs, 0).last_line;				\ +	  (Current).first_column = (Current).last_column =		\ +	    YYRHSLOC (Rhs, 0).last_column;				\ +	}								\ +    while (YYID (0)) +#endif + + +/* YY_LOCATION_PRINT -- Print the location on the stream. +   This macro was not mandated originally: define only if we know +   we won't break user code: when these are the locations we know.  */ + +#ifndef YY_LOCATION_PRINT +# if defined YYLTYPE_IS_TRIVIAL && YYLTYPE_IS_TRIVIAL +#  define YY_LOCATION_PRINT(File, Loc)			\ +     fprintf (File, "%d.%d-%d.%d",			\ +	      (Loc).first_line, (Loc).first_column,	\ +	      (Loc).last_line,  (Loc).last_column) +# else +#  define YY_LOCATION_PRINT(File, Loc) ((void) 0) +# endif +#endif + + +/* YYLEX -- calling `yylex' with the right arguments.  */ + +#ifdef YYLEX_PARAM +# define YYLEX yylex (YYLEX_PARAM) +#else +# define YYLEX yylex () +#endif + +/* Enable debugging if requested.  */ +#if YYDEBUG + +# ifndef YYFPRINTF +#  include <stdio.h> /* INFRINGES ON USER NAME SPACE */ +#  define YYFPRINTF fprintf +# endif + +# define YYDPRINTF(Args)			\ +do {						\ +  if (yydebug)					\ +    YYFPRINTF Args;				\ +} while (YYID (0)) + +# define YY_SYMBOL_PRINT(Title, Type, Value, Location)			  \ +do {									  \ +  if (yydebug)								  \ +    {									  \ +      YYFPRINTF (stderr, "%s ", Title);					  \ +      yy_symbol_print (stderr,						  \ +		  Type, Value); \ +      YYFPRINTF (stderr, "\n");						  \ +    }									  \ +} while (YYID (0)) + + +/*--------------------------------. +| Print this symbol on YYOUTPUT.  | +`--------------------------------*/ + +/*ARGSUSED*/ +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static void +yy_symbol_value_print (FILE *yyoutput, int yytype, YYSTYPE const * const yyvaluep) +#else +static void +yy_symbol_value_print (yyoutput, yytype, yyvaluep) +    FILE *yyoutput; +    int yytype; +    YYSTYPE const * const yyvaluep; +#endif +{ +  if (!yyvaluep) +    return; +# ifdef YYPRINT +  if (yytype < YYNTOKENS) +    YYPRINT (yyoutput, yytoknum[yytype], *yyvaluep); +# else +  YYUSE (yyoutput); +# endif +  switch (yytype) +    { +      default: +	break; +    } +} + + +/*--------------------------------. +| Print this symbol on YYOUTPUT.  | +`--------------------------------*/ + +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static void +yy_symbol_print (FILE *yyoutput, int yytype, YYSTYPE const * const yyvaluep) +#else +static void +yy_symbol_print (yyoutput, yytype, yyvaluep) +    FILE *yyoutput; +    int yytype; +    YYSTYPE const * const yyvaluep; +#endif +{ +  if (yytype < YYNTOKENS) +    YYFPRINTF (yyoutput, "token %s (", yytname[yytype]); +  else +    YYFPRINTF (yyoutput, "nterm %s (", yytname[yytype]); + +  yy_symbol_value_print (yyoutput, yytype, yyvaluep); +  YYFPRINTF (yyoutput, ")"); +} + +/*------------------------------------------------------------------. +| yy_stack_print -- Print the state stack from its BOTTOM up to its | +| TOP (included).                                                   | +`------------------------------------------------------------------*/ + +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static void +yy_stack_print (yytype_int16 *yybottom, yytype_int16 *yytop) +#else +static void +yy_stack_print (yybottom, yytop) +    yytype_int16 *yybottom; +    yytype_int16 *yytop; +#endif +{ +  YYFPRINTF (stderr, "Stack now"); +  for (; yybottom <= yytop; yybottom++) +    { +      int yybot = *yybottom; +      YYFPRINTF (stderr, " %d", yybot); +    } +  YYFPRINTF (stderr, "\n"); +} + +# define YY_STACK_PRINT(Bottom, Top)				\ +do {								\ +  if (yydebug)							\ +    yy_stack_print ((Bottom), (Top));				\ +} while (YYID (0)) + + +/*------------------------------------------------. +| Report that the YYRULE is going to be reduced.  | +`------------------------------------------------*/ + +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static void +yy_reduce_print (YYSTYPE *yyvsp, int yyrule) +#else +static void +yy_reduce_print (yyvsp, yyrule) +    YYSTYPE *yyvsp; +    int yyrule; +#endif +{ +  int yynrhs = yyr2[yyrule]; +  int yyi; +  unsigned long int yylno = yyrline[yyrule]; +  YYFPRINTF (stderr, "Reducing stack by rule %d (line %lu):\n", +	     yyrule - 1, yylno); +  /* The symbols being reduced.  */ +  for (yyi = 0; yyi < yynrhs; yyi++) +    { +      YYFPRINTF (stderr, "   $%d = ", yyi + 1); +      yy_symbol_print (stderr, yyrhs[yyprhs[yyrule] + yyi], +		       &(yyvsp[(yyi + 1) - (yynrhs)]) +		       		       ); +      YYFPRINTF (stderr, "\n"); +    } +} + +# define YY_REDUCE_PRINT(Rule)		\ +do {					\ +  if (yydebug)				\ +    yy_reduce_print (yyvsp, Rule); \ +} while (YYID (0)) + +/* Nonzero means print parse trace.  It is left uninitialized so that +   multiple parsers can coexist.  */ +int yydebug; +#else /* !YYDEBUG */ +# define YYDPRINTF(Args) +# define YY_SYMBOL_PRINT(Title, Type, Value, Location) +# define YY_STACK_PRINT(Bottom, Top) +# define YY_REDUCE_PRINT(Rule) +#endif /* !YYDEBUG */ + + +/* YYINITDEPTH -- initial size of the parser's stacks.  */ +#ifndef	YYINITDEPTH +# define YYINITDEPTH 200 +#endif + +/* YYMAXDEPTH -- maximum size the stacks can grow to (effective only +   if the built-in stack extension method is used). + +   Do not make this value too large; the results are undefined if +   YYSTACK_ALLOC_MAXIMUM < YYSTACK_BYTES (YYMAXDEPTH) +   evaluated with infinite-precision integer arithmetic.  */ + +#ifndef YYMAXDEPTH +# define YYMAXDEPTH 10000 +#endif + + + +#if YYERROR_VERBOSE + +# ifndef yystrlen +#  if defined __GLIBC__ && defined _STRING_H +#   define yystrlen strlen +#  else +/* Return the length of YYSTR.  */ +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static YYSIZE_T +yystrlen (const char *yystr) +#else +static YYSIZE_T +yystrlen (yystr) +    const char *yystr; +#endif +{ +  YYSIZE_T yylen; +  for (yylen = 0; yystr[yylen]; yylen++) +    continue; +  return yylen; +} +#  endif +# endif + +# ifndef yystpcpy +#  if defined __GLIBC__ && defined _STRING_H && defined _GNU_SOURCE +#   define yystpcpy stpcpy +#  else +/* Copy YYSRC to YYDEST, returning the address of the terminating '\0' in +   YYDEST.  */ +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static char * +yystpcpy (char *yydest, const char *yysrc) +#else +static char * +yystpcpy (yydest, yysrc) +    char *yydest; +    const char *yysrc; +#endif +{ +  char *yyd = yydest; +  const char *yys = yysrc; + +  while ((*yyd++ = *yys++) != '\0') +    continue; + +  return yyd - 1; +} +#  endif +# endif + +# ifndef yytnamerr +/* Copy to YYRES the contents of YYSTR after stripping away unnecessary +   quotes and backslashes, so that it's suitable for yyerror.  The +   heuristic is that double-quoting is unnecessary unless the string +   contains an apostrophe, a comma, or backslash (other than +   backslash-backslash).  YYSTR is taken from yytname.  If YYRES is +   null, do not copy; instead, return the length of what the result +   would have been.  */ +static YYSIZE_T +yytnamerr (char *yyres, const char *yystr) +{ +  if (*yystr == '"') +    { +      YYSIZE_T yyn = 0; +      char const *yyp = yystr; + +      for (;;) +	switch (*++yyp) +	  { +	  case '\'': +	  case ',': +	    goto do_not_strip_quotes; + +	  case '\\': +	    if (*++yyp != '\\') +	      goto do_not_strip_quotes; +	    /* Fall through.  */ +	  default: +	    if (yyres) +	      yyres[yyn] = *yyp; +	    yyn++; +	    break; + +	  case '"': +	    if (yyres) +	      yyres[yyn] = '\0'; +	    return yyn; +	  } +    do_not_strip_quotes: ; +    } + +  if (! yyres) +    return yystrlen (yystr); + +  return yystpcpy (yyres, yystr) - yyres; +} +# endif + +/* Copy into YYRESULT an error message about the unexpected token +   YYCHAR while in state YYSTATE.  Return the number of bytes copied, +   including the terminating null byte.  If YYRESULT is null, do not +   copy anything; just return the number of bytes that would be +   copied.  As a special case, return 0 if an ordinary "syntax error" +   message will do.  Return YYSIZE_MAXIMUM if overflow occurs during +   size calculation.  */ +static YYSIZE_T +yysyntax_error (char *yyresult, int yystate, int yychar) +{ +  int yyn = yypact[yystate]; + +  if (! (YYPACT_NINF < yyn && yyn <= YYLAST)) +    return 0; +  else +    { +      int yytype = YYTRANSLATE (yychar); +      YYSIZE_T yysize0 = yytnamerr (0, yytname[yytype]); +      YYSIZE_T yysize = yysize0; +      YYSIZE_T yysize1; +      int yysize_overflow = 0; +      enum { YYERROR_VERBOSE_ARGS_MAXIMUM = 5 }; +      char const *yyarg[YYERROR_VERBOSE_ARGS_MAXIMUM]; +      int yyx; + +# if 0 +      /* This is so xgettext sees the translatable formats that are +	 constructed on the fly.  */ +      YY_("syntax error, unexpected %s"); +      YY_("syntax error, unexpected %s, expecting %s"); +      YY_("syntax error, unexpected %s, expecting %s or %s"); +      YY_("syntax error, unexpected %s, expecting %s or %s or %s"); +      YY_("syntax error, unexpected %s, expecting %s or %s or %s or %s"); +# endif +      char *yyfmt; +      char const *yyf; +      static char const yyunexpected[] = "syntax error, unexpected %s"; +      static char const yyexpecting[] = ", expecting %s"; +      static char const yyor[] = " or %s"; +      char yyformat[sizeof yyunexpected +		    + sizeof yyexpecting - 1 +		    + ((YYERROR_VERBOSE_ARGS_MAXIMUM - 2) +		       * (sizeof yyor - 1))]; +      char const *yyprefix = yyexpecting; + +      /* Start YYX at -YYN if negative to avoid negative indexes in +	 YYCHECK.  */ +      int yyxbegin = yyn < 0 ? -yyn : 0; + +      /* Stay within bounds of both yycheck and yytname.  */ +      int yychecklim = YYLAST - yyn + 1; +      int yyxend = yychecklim < YYNTOKENS ? yychecklim : YYNTOKENS; +      int yycount = 1; + +      yyarg[0] = yytname[yytype]; +      yyfmt = yystpcpy (yyformat, yyunexpected); + +      for (yyx = yyxbegin; yyx < yyxend; ++yyx) +	if (yycheck[yyx + yyn] == yyx && yyx != YYTERROR) +	  { +	    if (yycount == YYERROR_VERBOSE_ARGS_MAXIMUM) +	      { +		yycount = 1; +		yysize = yysize0; +		yyformat[sizeof yyunexpected - 1] = '\0'; +		break; +	      } +	    yyarg[yycount++] = yytname[yyx]; +	    yysize1 = yysize + yytnamerr (0, yytname[yyx]); +	    yysize_overflow |= (yysize1 < yysize); +	    yysize = yysize1; +	    yyfmt = yystpcpy (yyfmt, yyprefix); +	    yyprefix = yyor; +	  } + +      yyf = YY_(yyformat); +      yysize1 = yysize + yystrlen (yyf); +      yysize_overflow |= (yysize1 < yysize); +      yysize = yysize1; + +      if (yysize_overflow) +	return YYSIZE_MAXIMUM; + +      if (yyresult) +	{ +	  /* Avoid sprintf, as that infringes on the user's name space. +	     Don't have undefined behavior even if the translation +	     produced a string with the wrong number of "%s"s.  */ +	  char *yyp = yyresult; +	  int yyi = 0; +	  while ((*yyp = *yyf) != '\0') +	    { +	      if (*yyp == '%' && yyf[1] == 's' && yyi < yycount) +		{ +		  yyp += yytnamerr (yyp, yyarg[yyi++]); +		  yyf += 2; +		} +	      else +		{ +		  yyp++; +		  yyf++; +		} +	    } +	} +      return yysize; +    } +} +#endif /* YYERROR_VERBOSE */ + + +/*-----------------------------------------------. +| Release the memory associated to this symbol.  | +`-----------------------------------------------*/ + +/*ARGSUSED*/ +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +static void +yydestruct (const char *yymsg, int yytype, YYSTYPE *yyvaluep) +#else +static void +yydestruct (yymsg, yytype, yyvaluep) +    const char *yymsg; +    int yytype; +    YYSTYPE *yyvaluep; +#endif +{ +  YYUSE (yyvaluep); + +  if (!yymsg) +    yymsg = "Deleting"; +  YY_SYMBOL_PRINT (yymsg, yytype, yyvaluep, yylocationp); + +  switch (yytype) +    { + +      default: +	break; +    } +} + +/* Prevent warnings from -Wmissing-prototypes.  */ +#ifdef YYPARSE_PARAM +#if defined __STDC__ || defined __cplusplus +int yyparse (void *YYPARSE_PARAM); +#else +int yyparse (); +#endif +#else /* ! YYPARSE_PARAM */ +#if defined __STDC__ || defined __cplusplus +int yyparse (void); +#else +int yyparse (); +#endif +#endif /* ! YYPARSE_PARAM */ + + +/* The lookahead symbol.  */ +int yychar; + +/* The semantic value of the lookahead symbol.  */ +YYSTYPE yylval; + +/* Number of syntax errors so far.  */ +int yynerrs; + + + +/*-------------------------. +| yyparse or yypush_parse.  | +`-------------------------*/ + +#ifdef YYPARSE_PARAM +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +int +yyparse (void *YYPARSE_PARAM) +#else +int +yyparse (YYPARSE_PARAM) +    void *YYPARSE_PARAM; +#endif +#else /* ! YYPARSE_PARAM */ +#if (defined __STDC__ || defined __C99__FUNC__ \ +     || defined __cplusplus || defined _MSC_VER) +int +yyparse (void) +#else +int +yyparse () + +#endif +#endif +{ + + +    int yystate; +    /* Number of tokens to shift before error messages enabled.  */ +    int yyerrstatus; + +    /* The stacks and their tools: +       `yyss': related to states. +       `yyvs': related to semantic values. + +       Refer to the stacks thru separate pointers, to allow yyoverflow +       to reallocate them elsewhere.  */ + +    /* The state stack.  */ +    yytype_int16 yyssa[YYINITDEPTH]; +    yytype_int16 *yyss; +    yytype_int16 *yyssp; + +    /* The semantic value stack.  */ +    YYSTYPE yyvsa[YYINITDEPTH]; +    YYSTYPE *yyvs; +    YYSTYPE *yyvsp; + +    YYSIZE_T yystacksize; + +  int yyn; +  int yyresult; +  /* Lookahead token as an internal (translated) token number.  */ +  int yytoken; +  /* The variables used to return semantic value and location from the +     action routines.  */ +  YYSTYPE yyval; + +#if YYERROR_VERBOSE +  /* Buffer for error messages, and its allocated size.  */ +  char yymsgbuf[128]; +  char *yymsg = yymsgbuf; +  YYSIZE_T yymsg_alloc = sizeof yymsgbuf; +#endif + +#define YYPOPSTACK(N)   (yyvsp -= (N), yyssp -= (N)) + +  /* The number of symbols on the RHS of the reduced rule. +     Keep to zero when no symbol should be popped.  */ +  int yylen = 0; + +  yytoken = 0; +  yyss = yyssa; +  yyvs = yyvsa; +  yystacksize = YYINITDEPTH; + +  YYDPRINTF ((stderr, "Starting parse\n")); + +  yystate = 0; +  yyerrstatus = 0; +  yynerrs = 0; +  yychar = YYEMPTY; /* Cause a token to be read.  */ + +  /* Initialize stack pointers. +     Waste one element of value and location stack +     so that they stay on the same level as the state stack. +     The wasted elements are never initialized.  */ +  yyssp = yyss; +  yyvsp = yyvs; + +  goto yysetstate; + +/*------------------------------------------------------------. +| yynewstate -- Push a new state, which is found in yystate.  | +`------------------------------------------------------------*/ + yynewstate: +  /* In all cases, when you get here, the value and location stacks +     have just been pushed.  So pushing a state here evens the stacks.  */ +  yyssp++; + + yysetstate: +  *yyssp = yystate; + +  if (yyss + yystacksize - 1 <= yyssp) +    { +      /* Get the current used size of the three stacks, in elements.  */ +      YYSIZE_T yysize = yyssp - yyss + 1; + +#ifdef yyoverflow +      { +	/* Give user a chance to reallocate the stack.  Use copies of +	   these so that the &'s don't force the real ones into +	   memory.  */ +	YYSTYPE *yyvs1 = yyvs; +	yytype_int16 *yyss1 = yyss; + +	/* Each stack pointer address is followed by the size of the +	   data in use in that stack, in bytes.  This used to be a +	   conditional around just the two extra args, but that might +	   be undefined if yyoverflow is a macro.  */ +	yyoverflow (YY_("memory exhausted"), +		    &yyss1, yysize * sizeof (*yyssp), +		    &yyvs1, yysize * sizeof (*yyvsp), +		    &yystacksize); + +	yyss = yyss1; +	yyvs = yyvs1; +      } +#else /* no yyoverflow */ +# ifndef YYSTACK_RELOCATE +      goto yyexhaustedlab; +# else +      /* Extend the stack our own way.  */ +      if (YYMAXDEPTH <= yystacksize) +	goto yyexhaustedlab; +      yystacksize *= 2; +      if (YYMAXDEPTH < yystacksize) +	yystacksize = YYMAXDEPTH; + +      { +	yytype_int16 *yyss1 = yyss; +	union yyalloc *yyptr = +	  (union yyalloc *) YYSTACK_ALLOC (YYSTACK_BYTES (yystacksize)); +	if (! yyptr) +	  goto yyexhaustedlab; +	YYSTACK_RELOCATE (yyss_alloc, yyss); +	YYSTACK_RELOCATE (yyvs_alloc, yyvs); +#  undef YYSTACK_RELOCATE +	if (yyss1 != yyssa) +	  YYSTACK_FREE (yyss1); +      } +# endif +#endif /* no yyoverflow */ + +      yyssp = yyss + yysize - 1; +      yyvsp = yyvs + yysize - 1; + +      YYDPRINTF ((stderr, "Stack size increased to %lu\n", +		  (unsigned long int) yystacksize)); + +      if (yyss + yystacksize - 1 <= yyssp) +	YYABORT; +    } + +  YYDPRINTF ((stderr, "Entering state %d\n", yystate)); + +  if (yystate == YYFINAL) +    YYACCEPT; + +  goto yybackup; + +/*-----------. +| yybackup.  | +`-----------*/ +yybackup: + +  /* Do appropriate processing given the current state.  Read a +     lookahead token if we need one and don't already have one.  */ + +  /* First try to decide what to do without reference to lookahead token.  */ +  yyn = yypact[yystate]; +  if (yyn == YYPACT_NINF) +    goto yydefault; + +  /* Not known => get a lookahead token if don't already have one.  */ + +  /* YYCHAR is either YYEMPTY or YYEOF or a valid lookahead symbol.  */ +  if (yychar == YYEMPTY) +    { +      YYDPRINTF ((stderr, "Reading a token: ")); +      yychar = YYLEX; +    } + +  if (yychar <= YYEOF) +    { +      yychar = yytoken = YYEOF; +      YYDPRINTF ((stderr, "Now at end of input.\n")); +    } +  else +    { +      yytoken = YYTRANSLATE (yychar); +      YY_SYMBOL_PRINT ("Next token is", yytoken, &yylval, &yylloc); +    } + +  /* If the proper action on seeing token YYTOKEN is to reduce or to +     detect an error, take that action.  */ +  yyn += yytoken; +  if (yyn < 0 || YYLAST < yyn || yycheck[yyn] != yytoken) +    goto yydefault; +  yyn = yytable[yyn]; +  if (yyn <= 0) +    { +      if (yyn == 0 || yyn == YYTABLE_NINF) +	goto yyerrlab; +      yyn = -yyn; +      goto yyreduce; +    } + +  /* Count tokens shifted since error; after three, turn off error +     status.  */ +  if (yyerrstatus) +    yyerrstatus--; + +  /* Shift the lookahead token.  */ +  YY_SYMBOL_PRINT ("Shifting", yytoken, &yylval, &yylloc); + +  /* Discard the shifted token.  */ +  yychar = YYEMPTY; + +  yystate = yyn; +  *++yyvsp = yylval; + +  goto yynewstate; + + +/*-----------------------------------------------------------. +| yydefault -- do the default action for the current state.  | +`-----------------------------------------------------------*/ +yydefault: +  yyn = yydefact[yystate]; +  if (yyn == 0) +    goto yyerrlab; +  goto yyreduce; + + +/*-----------------------------. +| yyreduce -- Do a reduction.  | +`-----------------------------*/ +yyreduce: +  /* yyn is the number of a rule to reduce with.  */ +  yylen = yyr2[yyn]; + +  /* If YYLEN is nonzero, implement the default value of the action: +     `$$ = $1'. + +     Otherwise, the following line sets YYVAL to garbage. +     This behavior is undocumented and Bison +     users should not rely upon it.  Assigning to YYVAL +     unconditionally makes the parser a bit smaller, and it avoids a +     GCC warning that YYVAL may be used uninitialized.  */ +  yyval = yyvsp[1-yylen]; + + +  YY_REDUCE_PRINT (yyn); +  switch (yyn) +    { +        case 2: + +/* Line 1464 of yacc.c  */ +#line 22 "src/tools/lcc/lburg/gram.y" +    { yylineno = 0; } +    break; + +  case 3: + +/* Line 1464 of yacc.c  */ +#line 23 "src/tools/lcc/lburg/gram.y" +    { yylineno = 0; } +    break; + +  case 7: + +/* Line 1464 of yacc.c  */ +#line 31 "src/tools/lcc/lburg/gram.y" +    { +		if (nonterm((yyvsp[(2) - (3)].string))->number != 1) +			yyerror("redeclaration of the start symbol\n"); +		} +    break; + +  case 9: + +/* Line 1464 of yacc.c  */ +#line 36 "src/tools/lcc/lburg/gram.y" +    { yyerrok; } +    break; + +  case 11: + +/* Line 1464 of yacc.c  */ +#line 40 "src/tools/lcc/lburg/gram.y" +    { term((yyvsp[(2) - (4)].string), (yyvsp[(4) - (4)].n)); } +    break; + +  case 13: + +/* Line 1464 of yacc.c  */ +#line 44 "src/tools/lcc/lburg/gram.y" +    { rule((yyvsp[(2) - (7)].string), (yyvsp[(4) - (7)].tree), (yyvsp[(5) - (7)].string), (yyvsp[(6) - (7)].string)); } +    break; + +  case 15: + +/* Line 1464 of yacc.c  */ +#line 46 "src/tools/lcc/lburg/gram.y" +    { yyerrok; } +    break; + +  case 16: + +/* Line 1464 of yacc.c  */ +#line 49 "src/tools/lcc/lburg/gram.y" +    { nonterm((yyval.string) = (yyvsp[(1) - (1)].string)); } +    break; + +  case 17: + +/* Line 1464 of yacc.c  */ +#line 52 "src/tools/lcc/lburg/gram.y" +    { (yyval.tree) = tree((yyvsp[(1) - (1)].string),  0,  0); } +    break; + +  case 18: + +/* Line 1464 of yacc.c  */ +#line 53 "src/tools/lcc/lburg/gram.y" +    { (yyval.tree) = tree((yyvsp[(1) - (4)].string), (yyvsp[(3) - (4)].tree),  0); } +    break; + +  case 19: + +/* Line 1464 of yacc.c  */ +#line 54 "src/tools/lcc/lburg/gram.y" +    { (yyval.tree) = tree((yyvsp[(1) - (6)].string), (yyvsp[(3) - (6)].tree), (yyvsp[(5) - (6)].tree)); } +    break; + +  case 20: + +/* Line 1464 of yacc.c  */ +#line 57 "src/tools/lcc/lburg/gram.y" +    { if (*(yyvsp[(1) - (1)].string) == 0) (yyval.string) = "0"; } +    break; + + + +/* Line 1464 of yacc.c  */ +#line 1461 "y.tab.c" +      default: break; +    } +  YY_SYMBOL_PRINT ("-> $$ =", yyr1[yyn], &yyval, &yyloc); + +  YYPOPSTACK (yylen); +  yylen = 0; +  YY_STACK_PRINT (yyss, yyssp); + +  *++yyvsp = yyval; + +  /* Now `shift' the result of the reduction.  Determine what state +     that goes to, based on the state we popped back to and the rule +     number reduced by.  */ + +  yyn = yyr1[yyn]; + +  yystate = yypgoto[yyn - YYNTOKENS] + *yyssp; +  if (0 <= yystate && yystate <= YYLAST && yycheck[yystate] == *yyssp) +    yystate = yytable[yystate]; +  else +    yystate = yydefgoto[yyn - YYNTOKENS]; + +  goto yynewstate; + + +/*------------------------------------. +| yyerrlab -- here on detecting error | +`------------------------------------*/ +yyerrlab: +  /* If not already recovering from an error, report this error.  */ +  if (!yyerrstatus) +    { +      ++yynerrs; +#if ! YYERROR_VERBOSE +      yyerror (YY_("syntax error")); +#else +      { +	YYSIZE_T yysize = yysyntax_error (0, yystate, yychar); +	if (yymsg_alloc < yysize && yymsg_alloc < YYSTACK_ALLOC_MAXIMUM) +	  { +	    YYSIZE_T yyalloc = 2 * yysize; +	    if (! (yysize <= yyalloc && yyalloc <= YYSTACK_ALLOC_MAXIMUM)) +	      yyalloc = YYSTACK_ALLOC_MAXIMUM; +	    if (yymsg != yymsgbuf) +	      YYSTACK_FREE (yymsg); +	    yymsg = (char *) YYSTACK_ALLOC (yyalloc); +	    if (yymsg) +	      yymsg_alloc = yyalloc; +	    else +	      { +		yymsg = yymsgbuf; +		yymsg_alloc = sizeof yymsgbuf; +	      } +	  } + +	if (0 < yysize && yysize <= yymsg_alloc) +	  { +	    (void) yysyntax_error (yymsg, yystate, yychar); +	    yyerror (yymsg); +	  } +	else +	  { +	    yyerror (YY_("syntax error")); +	    if (yysize != 0) +	      goto yyexhaustedlab; +	  } +      } +#endif +    } + + + +  if (yyerrstatus == 3) +    { +      /* If just tried and failed to reuse lookahead token after an +	 error, discard it.  */ + +      if (yychar <= YYEOF) +	{ +	  /* Return failure if at end of input.  */ +	  if (yychar == YYEOF) +	    YYABORT; +	} +      else +	{ +	  yydestruct ("Error: discarding", +		      yytoken, &yylval); +	  yychar = YYEMPTY; +	} +    } + +  /* Else will try to reuse lookahead token after shifting the error +     token.  */ +  goto yyerrlab1; + + +/*---------------------------------------------------. +| yyerrorlab -- error raised explicitly by YYERROR.  | +`---------------------------------------------------*/ +yyerrorlab: + +  /* Pacify compilers like GCC when the user code never invokes +     YYERROR and the label yyerrorlab therefore never appears in user +     code.  */ +  if (/*CONSTCOND*/ 0) +     goto yyerrorlab; + +  /* Do not reclaim the symbols of the rule which action triggered +     this YYERROR.  */ +  YYPOPSTACK (yylen); +  yylen = 0; +  YY_STACK_PRINT (yyss, yyssp); +  yystate = *yyssp; +  goto yyerrlab1; + + +/*-------------------------------------------------------------. +| yyerrlab1 -- common code for both syntax error and YYERROR.  | +`-------------------------------------------------------------*/ +yyerrlab1: +  yyerrstatus = 3;	/* Each real token shifted decrements this.  */ + +  for (;;) +    { +      yyn = yypact[yystate]; +      if (yyn != YYPACT_NINF) +	{ +	  yyn += YYTERROR; +	  if (0 <= yyn && yyn <= YYLAST && yycheck[yyn] == YYTERROR) +	    { +	      yyn = yytable[yyn]; +	      if (0 < yyn) +		break; +	    } +	} + +      /* Pop the current state because it cannot handle the error token.  */ +      if (yyssp == yyss) +	YYABORT; + + +      yydestruct ("Error: popping", +		  yystos[yystate], yyvsp); +      YYPOPSTACK (1); +      yystate = *yyssp; +      YY_STACK_PRINT (yyss, yyssp); +    } + +  *++yyvsp = yylval; + + +  /* Shift the error token.  */ +  YY_SYMBOL_PRINT ("Shifting", yystos[yyn], yyvsp, yylsp); + +  yystate = yyn; +  goto yynewstate; + + +/*-------------------------------------. +| yyacceptlab -- YYACCEPT comes here.  | +`-------------------------------------*/ +yyacceptlab: +  yyresult = 0; +  goto yyreturn; + +/*-----------------------------------. +| yyabortlab -- YYABORT comes here.  | +`-----------------------------------*/ +yyabortlab: +  yyresult = 1; +  goto yyreturn; + +#if !defined(yyoverflow) || YYERROR_VERBOSE +/*-------------------------------------------------. +| yyexhaustedlab -- memory exhaustion comes here.  | +`-------------------------------------------------*/ +yyexhaustedlab: +  yyerror (YY_("memory exhausted")); +  yyresult = 2; +  /* Fall through.  */ +#endif + +yyreturn: +  if (yychar != YYEMPTY) +     yydestruct ("Cleanup: discarding lookahead", +		 yytoken, &yylval); +  /* Do not reclaim the symbols of the rule which action triggered +     this YYABORT or YYACCEPT.  */ +  YYPOPSTACK (yylen); +  YY_STACK_PRINT (yyss, yyssp); +  while (yyssp != yyss) +    { +      yydestruct ("Cleanup: popping", +		  yystos[*yyssp], yyvsp); +      YYPOPSTACK (1); +    } +#ifndef yyoverflow +  if (yyss != yyssa) +    YYSTACK_FREE (yyss); +#endif +#if YYERROR_VERBOSE +  if (yymsg != yymsgbuf) +    YYSTACK_FREE (yymsg); +#endif +  /* Make sure YYID is used.  */ +  return YYID (yyresult); +} + + + +/* Line 1684 of yacc.c  */ +#line 59 "src/tools/lcc/lburg/gram.y" + +#include <assert.h> +#include <stdarg.h> +#include <ctype.h> +#include <string.h> +#include <limits.h> + +int errcnt = 0; +FILE *infp = NULL; +FILE *outfp = NULL; +static char buf[BUFSIZ], *bp = buf; +static int ppercent = 0; +static int code = 0; + +static int get(void) { +	if (*bp == 0) { +		bp = buf; +		*bp = 0; +		if (fgets(buf, sizeof buf, infp) == NULL) +			return EOF; +		yylineno++; +		while (buf[0] == '%' && buf[1] == '{' && buf[2] == '\n') { +			for (;;) { +				if (fgets(buf, sizeof buf, infp) == NULL) { +					yywarn("unterminated %{...%}\n"); +					return EOF; +				} +				yylineno++; +				if (strcmp(buf, "%}\n") == 0) +					break; +				fputs(buf, outfp); +			} +			if (fgets(buf, sizeof buf, infp) == NULL) +				return EOF; +			yylineno++; +		} +	} +	return *bp++; +} + +void yyerror(char *fmt, ...) { +	va_list ap; + +	va_start(ap, fmt); +	if (yylineno > 0) +		fprintf(stderr, "line %d: ", yylineno); +	vfprintf(stderr, fmt, ap); +	if (fmt[strlen(fmt)-1] != '\n') +		 fprintf(stderr, "\n"); +	errcnt++; +	va_end(ap); +} + +int yylex(void) { +	int c; + +	if (code) { +		char *p; +		bp += strspn(bp, " \t\f"); +		p = strchr(bp, '\n'); +		if (p == NULL) +			p = strchr(bp, '\n'); +		while (p > bp && isspace(p[-1])) +			p--; +		yylval.string = alloc(p - bp + 1); +		strncpy(yylval.string, bp, p - bp); +		yylval.string[p - bp] = 0; +		bp = p; +		code--; +		return CODE; +	} +	while ((c = get()) != EOF) { +		switch (c) { +		case ' ': case '\f': case '\t': +			continue; +		case '\n': +		case '(': case ')': case ',': +		case ':': case '=': +			return c; +		} +		if (c == '%' && *bp == '%') { +			bp++; +			return ppercent++ ? 0 : PPERCENT; +		} else if (c == '%' && strncmp(bp, "term", 4) == 0 +		&& isspace(bp[4])) { +			bp += 4; +			return TERMINAL; +		} else if (c == '%' && strncmp(bp, "start", 5) == 0 +		&& isspace(bp[5])) { +			bp += 5; +			return START; +		} else if (c == '"') { +			char *p = strchr(bp, '"'); +			if (p == NULL) { +				yyerror("missing \" in assembler template\n"); +				p = strchr(bp, '\n'); +				if (p == NULL) +					p = strchr(bp, '\0'); +			} +			assert(p); +			yylval.string = alloc(p - bp + 1); +			strncpy(yylval.string, bp, p - bp); +			yylval.string[p - bp] = 0; +			bp = *p == '"' ? p + 1 : p; +			code++; +			return TEMPLATE; +		} else if (isdigit(c)) { +			int n = 0; +			do { +				int d = c - '0'; +				if (n > (INT_MAX - d)/10) +					yyerror("integer greater than %d\n", INT_MAX); +				else +					n = 10*n + d; +				c = get(); +			} while (c != EOF && isdigit(c)); +			bp--; +			yylval.n = n; +			return INT; +		} else if (isalpha(c)) { +			char *p = bp - 1; +			while (isalpha(*bp) || isdigit(*bp) || *bp == '_') +				bp++; +			yylval.string = alloc(bp - p + 1); +			strncpy(yylval.string, p, bp - p); +			yylval.string[bp - p] = 0; +			return ID; +		} else if (isprint(c)) +			yyerror("invalid character `%c'\n", c); +		else +			yyerror("invalid character `\\%03o'\n", (unsigned char)c); +	} +	return 0; +} + +void yywarn(char *fmt, ...) { +	va_list ap; + +	va_start(ap, fmt); +	if (yylineno > 0) +		fprintf(stderr, "line %d: ", yylineno); +	fprintf(stderr, "warning: "); +	vfprintf(stderr, fmt, ap); +} + diff --git a/src/tools/lcc/lburg/gram.y b/src/tools/lcc/lburg/gram.y new file mode 100644 index 0000000..1ecd8a9 --- /dev/null +++ b/src/tools/lcc/lburg/gram.y @@ -0,0 +1,202 @@ +%{ +#include <stdio.h> +#include "lburg.h" +static char rcsid[] = "$Id: gram.y 145 2001-10-17 21:53:10Z timo $"; +/*lint -e616 -e527 -e652 -esym(552,yynerrs) -esym(563,yynewstate,yyerrlab) */ +static int yylineno = 0; +%} +%union { +	int n; +	char *string; +	Tree tree; +} +%term TERMINAL +%term START +%term PPERCENT + +%token  <string>        ID TEMPLATE CODE +%token  <n>             INT +%type	<string>	nonterm cost +%type   <tree>          tree +%% +spec	: decls PPERCENT rules		{ yylineno = 0; } +	| decls				{ yylineno = 0; } +	; + +decls	: /* lambda */ +	| decls decl +	; + +decl	: TERMINAL  blist '\n' +	| START nonterm '\n'		{ +		if (nonterm($2)->number != 1) +			yyerror("redeclaration of the start symbol\n"); +		} +	| '\n' +	| error '\n'			{ yyerrok; } +	; + +blist	: /* lambda */ +	| blist ID '=' INT      	{ term($2, $4); } +	; + +rules	: /* lambda */ +	| rules nonterm ':' tree TEMPLATE cost '\n'	{ rule($2, $4, $5, $6); } +	| rules '\n' +	| rules error '\n'		{ yyerrok; } +	; + +nonterm	: ID				{ nonterm($$ = $1); } +	; + +tree	: ID                            { $$ = tree($1,  0,  0); } +	| ID '(' tree ')'               { $$ = tree($1, $3,  0); } +	| ID '(' tree ',' tree ')'      { $$ = tree($1, $3, $5); } +	; + +cost	: CODE				{ if (*$1 == 0) $$ = "0"; } +	; +%% +#include <assert.h> +#include <stdarg.h> +#include <ctype.h> +#include <string.h> +#include <limits.h> + +int errcnt = 0; +FILE *infp = NULL; +FILE *outfp = NULL; +static char buf[BUFSIZ], *bp = buf; +static int ppercent = 0; +static int code = 0; + +static int get(void) { +	if (*bp == 0) { +		bp = buf; +		*bp = 0; +		if (fgets(buf, sizeof buf, infp) == NULL) +			return EOF; +		yylineno++; +		while (buf[0] == '%' && buf[1] == '{' && buf[2] == '\n') { +			for (;;) { +				if (fgets(buf, sizeof buf, infp) == NULL) { +					yywarn("unterminated %{...%}\n"); +					return EOF; +				} +				yylineno++; +				if (strcmp(buf, "%}\n") == 0) +					break; +				fputs(buf, outfp); +			} +			if (fgets(buf, sizeof buf, infp) == NULL) +				return EOF; +			yylineno++; +		} +	} +	return *bp++; +} + +void yyerror(char *fmt, ...) { +	va_list ap; + +	va_start(ap, fmt); +	if (yylineno > 0) +		fprintf(stderr, "line %d: ", yylineno); +	vfprintf(stderr, fmt, ap); +	if (fmt[strlen(fmt)-1] != '\n') +		 fprintf(stderr, "\n"); +	errcnt++; +	va_end(ap); +} + +int yylex(void) { +	int c; + +	if (code) { +		char *p; +		bp += strspn(bp, " \t\f"); +		p = strchr(bp, '\n'); +		if (p == NULL) +			p = strchr(bp, '\n'); +		while (p > bp && isspace(p[-1])) +			p--; +		yylval.string = alloc(p - bp + 1); +		strncpy(yylval.string, bp, p - bp); +		yylval.string[p - bp] = 0; +		bp = p; +		code--; +		return CODE; +	} +	while ((c = get()) != EOF) { +		switch (c) { +		case ' ': case '\f': case '\t': +			continue; +		case '\n': +		case '(': case ')': case ',': +		case ':': case '=': +			return c; +		} +		if (c == '%' && *bp == '%') { +			bp++; +			return ppercent++ ? 0 : PPERCENT; +		} else if (c == '%' && strncmp(bp, "term", 4) == 0 +		&& isspace(bp[4])) { +			bp += 4; +			return TERMINAL; +		} else if (c == '%' && strncmp(bp, "start", 5) == 0 +		&& isspace(bp[5])) { +			bp += 5; +			return START; +		} else if (c == '"') { +			char *p = strchr(bp, '"'); +			if (p == NULL) { +				yyerror("missing \" in assembler template\n"); +				p = strchr(bp, '\n'); +				if (p == NULL) +					p = strchr(bp, '\0'); +			} +			assert(p); +			yylval.string = alloc(p - bp + 1); +			strncpy(yylval.string, bp, p - bp); +			yylval.string[p - bp] = 0; +			bp = *p == '"' ? p + 1 : p; +			code++; +			return TEMPLATE; +		} else if (isdigit(c)) { +			int n = 0; +			do { +				int d = c - '0'; +				if (n > (INT_MAX - d)/10) +					yyerror("integer greater than %d\n", INT_MAX); +				else +					n = 10*n + d; +				c = get(); +			} while (c != EOF && isdigit(c)); +			bp--; +			yylval.n = n; +			return INT; +		} else if (isalpha(c)) { +			char *p = bp - 1; +			while (isalpha(*bp) || isdigit(*bp) || *bp == '_') +				bp++; +			yylval.string = alloc(bp - p + 1); +			strncpy(yylval.string, p, bp - p); +			yylval.string[bp - p] = 0; +			return ID; +		} else if (isprint(c)) +			yyerror("invalid character `%c'\n", c); +		else +			yyerror("invalid character `\\%03o'\n", (unsigned char)c); +	} +	return 0; +} + +void yywarn(char *fmt, ...) { +	va_list ap; + +	va_start(ap, fmt); +	if (yylineno > 0) +		fprintf(stderr, "line %d: ", yylineno); +	fprintf(stderr, "warning: "); +	vfprintf(stderr, fmt, ap); +} diff --git a/src/tools/lcc/lburg/lburg.1 b/src/tools/lcc/lburg/lburg.1 new file mode 100644 index 0000000..8cf7250 --- /dev/null +++ b/src/tools/lcc/lburg/lburg.1 @@ -0,0 +1,179 @@ +.TH LBURG 1 "local \- 11/30/94" +.\" $Id: lburg.1 145 2001-10-17 21:53:10Z timo $ +.SH NAME +lburg \- lcc's code-generator generator +.SH SYNOPSIS +.B lburg +[ +.I option +]... +[ [ +.I input +] +.I output +] +.br +.SH DESCRIPTION +.PP +.I lburg +reads an lcc-style BURG specification from +.I input +and writes a pattern-matching code generator to +.IR output . +If +.I input +is `\-' or is omitted, +.I lburg +reads the standard input; +If +.I output +is `\-' or is omitted, +.I lburg +writes to the standard output. +.PP +.I lburg +accepts specifications that conform to the following EBNF grammar. +Terminals are enclosed in single quotes or are +given in uppercase, all other symbols are nonterminals or English phrases, +{X} denotes zero or more instances of X, and [X] denotes an optional X. +.PP +.nf +.RS +.ft CW +spec:     `%{' configuration `%}' { dcl } `%%' { rule } +               [ `%%' C code ] + +dcl:      `%start' nonterm +          `%term' { ID `=' INT } + +rule:     nonterm `:' tree template [ C expression ] + +tree:     term `(' tree `,' tree `)' +          term `(' tree `)' +          term +          nonterm + +nonterm:  ID + +template: `"' { any character except double quote } `"' +.RE +.fi +.PP +Specifications are structurally similar to +.IR yacc 's. +Text between +`\f(CW%{\fP' +and +`\f(CW%}\fP' +is called the configuration section; there may be several such segments. +All are concatenated and copied verbatim into the head of the output. +Text after the second +`\f(CW%%\fP', +if any, is also copied verbatim into the output, at the end. +.PP +Specifications consist of declarations, a +`\f(CW%%\fP' +separator, and rules. +Input is line-oriented; each declaration and rule must appear on a separate line, +and declarations must begin in column 1. +Declarations declare terminals \(em the operators in subject +trees \(em and associate a unique, positive external symbol +number with each one. +Nonterminals are declared by their presence +on the left side of rules.  The +\f(CW%start\fP +declaration optionally declares a nonterminal as the start symbol. +In the grammar above, +\f(CWterm\fP +and +\f(CWnonterm\fP +denote identifiers that are terminals and nonterminals. +.PP +Rules define tree patterns in a fully parenthesized prefix +form. Every nonterminal denotes a tree. +Each operator has a fixed +arity, which is inferred from the rules in which it is used. +A chain rule is a rule whose pattern is another nonterminal. +If no start symbol is declared, the nonterminal defined by the first rule is used. +.PP +Each rule ends with an expression that computes the cost of matching +that rule; omitted costs +default to zero. Costs of chain rules must be constants. +.PP +The configuration section configures the output +for the trees being parsed and the client's environment. +As shown, this section must define +\f(CWNODEPTR_TYPE\fP +to be a visible typedef symbol for a pointer to a +node in the subject tree. +The labeller invokes +\f(CWOP_LABEL(p)\fP, +\f(CWLEFT\_CHILD(p)\fP, and +\f(CWRIGHT\_CHILD(p)\fP +to read the operator and children from the node pointed to by \f(CWp\fP. +If the configuration section defines these operations as macros, they are implemented in-line; +otherwise, they must be implemented as functions. +.PP +The matcher +computes and stores a single integral state in each node of the subject tree. +The configuration section must define a macro +\f(CWSTATE_LABEL(p)\fP +to access the state field of the node pointed to +by \f(CWp\fP. It must be large enough to hold a pointer, and +a macro is required because it is used as an lvalue. +.PP +.SH OPTIONS +.TP +.BI \-p \ prefix +.br +.ns +.TP +.BI \-p prefix +Use +.I prefix +as the disambiquating prefix for visible names and fields. +The default is `\f(CW_\fP'. +.TP +.B \-T +Arrange for +.sp +.nf +.ft CW +    void _trace(NODEPTR_TYPE p, int eruleno, +                    int cost, int bestcost); +.sp +.fi +.ft R +to be called at each successful match. +\f(CWp\fP +identifies the node and +\f(CWeruleno\fP +identifies the matching rule; the rules are numbered +beginning at 1 in the order they appear in the input. +\f(CWcost\fP +is the cost of the match and +\f(CWbestcost\fP +is the cost of the best previous match. The current match +wins only if +\f(CWcost\fP +is less than \f(CWbestcost\fP. +32767 represents the infinite cost of no previous match. +\f(CW_trace\fP must be declared in the configuration section. +.SH "SEE ALSO" +.IR lcc (1) +.PP +C. W. Fraser and D. R. Hanson, +.IR A Retargetable C Compiler: Design and Implementation , +Benjamin/Cummings, Redwood City, CA, 1995, +ISBN 0-8053-1670-1. Chapter 14. +.PP +C. W. Fraser, D. R. Hanson and T. A. Proebsting, +`Engineering a simple, efficient code generator generator,' +.I +ACM Letters on Programming Languages and Systems +.BR 1 , +3 (Sep. 1992), 213-226. +.br +.SH BUGS +Mail bug reports along with the shortest input +that exposes them to drh@cs.princeton.edu. diff --git a/src/tools/lcc/lburg/lburg.c b/src/tools/lcc/lburg/lburg.c new file mode 100644 index 0000000..c43c96a --- /dev/null +++ b/src/tools/lcc/lburg/lburg.c @@ -0,0 +1,671 @@ +#include <assert.h> +#include <ctype.h> +#include <stdarg.h> +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <time.h> +#include "lburg.h" + +static char rcsid[] = "lburg.c - faked rcsid"; + +static char *prefix = ""; +static int Tflag = 0; +static int ntnumber = 0; +static Nonterm start = 0; +static Term terms; +static Nonterm nts; +static Rule rules; +static int nrules; +static struct block { +	struct block *link; +} *memlist;			/* list of allocated blocks */ + +static char *stringf(char *fmt, ...); +static void print(char *fmt, ...); +static void ckreach(Nonterm p); +static void emitclosure(Nonterm nts); +static void emitcost(Tree t, char *v); +static void emitdefs(Nonterm nts, int ntnumber); +static void emitheader(void); +static void emitkids(Rule rules, int nrules); +static void emitnts(Rule rules, int nrules); +static void emitrecalc(char *pre, Term root, Term kid); +static void emitrecord(char *pre, Rule r, char *c, int cost); +static void emitrule(Nonterm nts); +static void emitlabel(Term terms, Nonterm start, int ntnumber); +static void emitstring(Rule rules); +static void emitstruct(Nonterm nts, int ntnumber); +static void emittest(Tree t, char *v, char *suffix); + +int main(int argc, char *argv[]) { +	int c, i; +	Nonterm p; +	 +	for (i = 1; i < argc; i++) +		if (strcmp(argv[i], "-T") == 0) +			Tflag = 1; +		else if (strncmp(argv[i], "-p", 2) == 0 && argv[i][2]) +			prefix = &argv[i][2]; +		else if (strncmp(argv[i], "-p", 2) == 0 && i + 1 < argc) +			prefix = argv[++i]; +		else if (*argv[i] == '-' && argv[i][1]) { +			yyerror("usage: %s [-T | -p prefix]... [ [ input ] output ] \n", +				argv[0]); +			exit(1); +		} else if (infp == NULL) { +			if (strcmp(argv[i], "-") == 0) +				infp = stdin; +			else if ((infp = fopen(argv[i], "r")) == NULL) { +				yyerror("%s: can't read `%s'\n", argv[0], argv[i]); +				exit(1); +			} +		} else if (outfp == NULL) { +			if (strcmp(argv[i], "-") == 0) +				outfp = stdout; +			if ((outfp = fopen(argv[i], "w")) == NULL) { +				yyerror("%s: can't write `%s'\n", argv[0], argv[i]); +				exit(1); +			} +		} +	if (infp == NULL) +		infp = stdin; +	if (outfp == NULL) +		outfp = stdout; +	yyparse(); +	if (start) +		ckreach(start); +	for (p = nts; p; p = p->link) { +		if (p->rules == NULL) +			yyerror("undefined nonterminal `%s'\n", p->name); +		if (!p->reached) +			yyerror("can't reach nonterminal `%s'\n", p->name); +	} +	emitheader(); +	emitdefs(nts, ntnumber); +	emitstruct(nts, ntnumber); +	emitnts(rules, nrules); +	emitstring(rules); +	emitrule(nts); +	emitclosure(nts); +	if (start) +		emitlabel(terms, start, ntnumber); +	emitkids(rules, nrules); +	if (!feof(infp)) +		while ((c = getc(infp)) != EOF) +			putc(c, outfp); +	while (memlist) {	/* for purify */ +		struct block *q = memlist->link; +		free(memlist); +		memlist = q; +	} +	return errcnt > 0; +} + +/* alloc - allocate nbytes or issue fatal error */ +void *alloc(int nbytes) { +	struct block *p = calloc(1, sizeof *p + nbytes); + +	if (p == NULL) { +		yyerror("out of memory\n"); +		exit(1); +	} +	p->link = memlist; +	memlist = p; +	return p + 1; +} + +/* stringf - format and save a string */ +static char *stringf(char *fmt, ...) { +	va_list ap; +	char buf[512]; + +	va_start(ap, fmt); +	vsprintf(buf, fmt, ap); +	va_end(ap); +	return strcpy(alloc(strlen(buf) + 1), buf); +}	 + +struct entry { +	union { +		char *name; +		struct term t; +		struct nonterm nt; +	} sym; +	struct entry *link; +} *table[211]; +#define HASHSIZE (sizeof table/sizeof table[0]) + +/* hash - return hash number for str */ +static unsigned hash(char *str) { +	unsigned h = 0; + +	while (*str) +		h = (h<<1) + *str++; +	return h; +} + +/* lookup - lookup symbol name */ +static void *lookup(char *name) { +	struct entry *p = table[hash(name)%HASHSIZE]; + +	for ( ; p; p = p->link) +		if (strcmp(name, p->sym.name) == 0) +			return &p->sym; +	return 0; +} + +/* install - install symbol name */ +static void *install(char *name) { +	struct entry *p = alloc(sizeof *p); +	int i = hash(name)%HASHSIZE; + +	p->sym.name = name; +	p->link = table[i]; +	table[i] = p; +	return &p->sym; +} + +/* nonterm - create a new terminal id, if necessary */ +Nonterm nonterm(char *id) { +	Nonterm p = lookup(id), *q = &nts; + +	if (p && p->kind == NONTERM) +		return p; +	if (p && p->kind == TERM) +		yyerror("`%s' is a terminal\n", id); +	p = install(id); +	p->kind = NONTERM; +	p->number = ++ntnumber; +	if (p->number == 1) +		start = p; +	while (*q && (*q)->number < p->number) +		q = &(*q)->link; +	assert(*q == 0 || (*q)->number != p->number); +	p->link = *q; +	*q = p; +	return p; +} + +/* term - create a new terminal id with external symbol number esn */ +Term term(char *id, int esn) { +	Term p = lookup(id), *q = &terms; + +	if (p) +		yyerror("redefinition of terminal `%s'\n", id); +	else +		p = install(id); +	p->kind = TERM; +	p->esn = esn; +	p->arity = -1; +	while (*q && (*q)->esn < p->esn) +		q = &(*q)->link; +	if (*q && (*q)->esn == p->esn) +		yyerror("duplicate external symbol number `%s=%d'\n", +			p->name, p->esn); +	p->link = *q; +	*q = p; +	return p; +} + +/* tree - create & initialize a tree node with the given fields */ +Tree tree(char *id, Tree left, Tree right) { +	Tree t = alloc(sizeof *t); +	Term p = lookup(id); +	int arity = 0; + +	if (left && right) +		arity = 2; +	else if (left) +		arity = 1; +	if (p == NULL && arity > 0) { +		yyerror("undefined terminal `%s'\n", id); +		p = term(id, -1); +	} else if (p == NULL && arity == 0) +		p = (Term)nonterm(id); +	else if (p && p->kind == NONTERM && arity > 0) { +		yyerror("`%s' is a nonterminal\n", id); +		p = term(id, -1); +	} +	if (p->kind == TERM && p->arity == -1) +		p->arity = arity; +	if (p->kind == TERM && arity != p->arity) +		yyerror("inconsistent arity for terminal `%s'\n", id); +	t->op = p; +	t->nterms = p->kind == TERM; +	if ((t->left = left) != NULL) +		t->nterms += left->nterms; +	if ((t->right = right) != NULL) +		t->nterms += right->nterms; +	return t; +} + +/* rule - create & initialize a rule with the given fields */ +Rule rule(char *id, Tree pattern, char *template, char *code) { +	Rule r = alloc(sizeof *r), *q; +	Term p = pattern->op; +	char *end; + +	r->lhs = nonterm(id); +	r->packed = ++r->lhs->lhscount; +	for (q = &r->lhs->rules; *q; q = &(*q)->decode) +		; +	*q = r; +	r->pattern = pattern; +	r->ern = ++nrules; +	r->template = template; +	r->code = code; +	r->cost = strtol(code, &end, 10); +	if (*end) { +		r->cost = -1; +		r->code = stringf("(%s)", code); +	} +	if (p->kind == TERM) { +		for (q = &p->rules; *q; q = &(*q)->next) +			; +		*q = r; +	} else if (pattern->left == NULL && pattern->right == NULL) { +		Nonterm p = pattern->op; +		r->chain = p->chain; +	        p->chain = r; +		if (r->cost == -1) +			yyerror("illegal nonconstant cost `%s'\n", code); +	} +	for (q = &rules; *q; q = &(*q)->link) +		; +	r->link = *q; +	*q = r; +	return r; +} + +/* print - formatted output */ +static void print(char *fmt, ...) { +	va_list ap; + +	va_start(ap, fmt); +	for ( ; *fmt; fmt++) +		if (*fmt == '%') +			switch (*++fmt) { +			case 'd': fprintf(outfp, "%d", va_arg(ap, int)); break; +			case 's': fputs(va_arg(ap, char *), outfp); break; +			case 'P': fprintf(outfp, "%s_", prefix); break; +			case 'T': { +				Tree t = va_arg(ap, Tree); +				print("%S", t->op); +				if (t->left && t->right) +					print("(%T,%T)", t->left, t->right); +				else if (t->left) +					print("(%T)", t->left); +				break; +				} +			case 'R': { +				Rule r = va_arg(ap, Rule); +				print("%S: %T", r->lhs, r->pattern); +				break; +				} +			case 'S': fputs(va_arg(ap, Term)->name, outfp); break; +			case '1': case '2': case '3': case '4': case '5': { +				int n = *fmt - '0'; +				while (n-- > 0) +					putc('\t', outfp); +				break; +				} +			default: putc(*fmt, outfp); break;			 +			} +		else +			putc(*fmt, outfp); +	va_end(ap); +} + +/* reach - mark all nonterminals in tree t as reachable */ +static void reach(Tree t) { +	Nonterm p = t->op; + +	if (p->kind == NONTERM) +		if (!p->reached) +			ckreach(p); +	if (t->left) +		reach(t->left); +	if (t->right) +		reach(t->right); +} + +/* ckreach - mark all nonterminals reachable from p */ +static void ckreach(Nonterm p) { +	Rule r; + +        p->reached = 1; +	for (r = p->rules; r; r = r->decode) +		reach(r->pattern); +} + +/* emitcase - emit one case in function state */ +static void emitcase(Term p, int ntnumber) { +	Rule r; + +	print("%1case %d: /* %S */\n", p->esn, p); +	switch (p->arity) { +	case 0: case -1: +		break; +	case 1: +		print("%2%Plabel(LEFT_CHILD(a));\n"); +		break; +	case 2: +		print("%2%Plabel(LEFT_CHILD(a));\n"); +		print("%2%Plabel(RIGHT_CHILD(a));\n"); +		break; +	default: assert(0); +	} +	for (r = p->rules; r; r = r->next) { +		char *indent = "\t\t\0"; +		switch (p->arity) { +		case 0: case -1: +			print("%2/* %R */\n", r); +			if (r->cost == -1) { +				print("%2c = %s;\n", r->code); +				emitrecord("\t\t", r, "c", 0); +			} else +				emitrecord("\t\t", r, r->code, 0); +			break; +		case 1: +			if (r->pattern->nterms > 1) { +				print("%2if (%1/* %R */\n", r); +				emittest(r->pattern->left, "LEFT_CHILD(a)", " "); +				print("%2) {\n"); +				indent = "\t\t\t"; +			} else +				print("%2/* %R */\n", r); +			if (r->pattern->nterms == 2 && r->pattern->left +			&&  r->pattern->right == NULL) +				emitrecalc(indent, r->pattern->op, r->pattern->left->op); +			print("%sc = ", indent); +			emitcost(r->pattern->left, "LEFT_CHILD(a)"); +			print("%s;\n", r->code); +			emitrecord(indent, r, "c", 0); +			if (indent[2]) +				print("%2}\n"); +			break; +		case 2: +			if (r->pattern->nterms > 1) { +				print("%2if (%1/* %R */\n", r); +				emittest(r->pattern->left,  "LEFT_CHILD(a)", +					r->pattern->right->nterms ? " && " : " "); +				emittest(r->pattern->right, "RIGHT_CHILD(a)", " "); +				print("%2) {\n"); +				indent = "\t\t\t"; +			} else +				print("%2/* %R */\n", r); +			print("%sc = ", indent); +			emitcost(r->pattern->left,  "LEFT_CHILD(a)"); +			emitcost(r->pattern->right, "RIGHT_CHILD(a)"); +			print("%s;\n", r->code); +			emitrecord(indent, r, "c", 0); +			if (indent[2]) +				print("%2}\n"); +			break; +		default: assert(0); +		} +	} +	print("%2break;\n"); +} + +/* emitclosure - emit the closure functions */ +static void emitclosure(Nonterm nts) { +	Nonterm p; + +	for (p = nts; p; p = p->link) +		if (p->chain) +			print("static void %Pclosure_%S(NODEPTR_TYPE, int);\n", p); +	print("\n"); +	for (p = nts; p; p = p->link) +		if (p->chain) { +			Rule r; +			print("static void %Pclosure_%S(NODEPTR_TYPE a, int c) {\n" +"%1struct %Pstate *p = STATE_LABEL(a);\n", p); +			for (r = p->chain; r; r = r->chain) +				emitrecord("\t", r, "c", r->cost); +			print("}\n\n"); +		} +} + +/* emitcost - emit cost computation for tree t */ +static void emitcost(Tree t, char *v) { +	Nonterm p = t->op; + +	if (p->kind == TERM) { +		if (t->left) +			emitcost(t->left,  stringf("LEFT_CHILD(%s)",  v)); +		if (t->right) +			emitcost(t->right, stringf("RIGHT_CHILD(%s)", v)); +	} else +		print("((struct %Pstate *)(%s->x.state))->cost[%P%S_NT] + ", v, p); +} + +/* emitdefs - emit nonterminal defines and data structures */ +static void emitdefs(Nonterm nts, int ntnumber) { +	Nonterm p; + +	for (p = nts; p; p = p->link) +		print("#define %P%S_NT %d\n", p, p->number); +	print("\n"); +	print("static char *%Pntname[] = {\n%10,\n"); +	for (p = nts; p; p = p->link) +		print("%1\"%S\",\n", p); +	print("%10\n};\n\n"); +} + +/* emitheader - emit initial definitions */ +static void emitheader(void) { +	time_t timer = time(NULL); + +	print("/*\ngenerated at %sby %s\n*/\n", ctime(&timer), rcsid); +	print("static void %Pkids(NODEPTR_TYPE, int, NODEPTR_TYPE[]);\n"); +	print("static void %Plabel(NODEPTR_TYPE);\n"); +	print("static int %Prule(void*, int);\n\n"); +} + +/* computekids - compute paths to kids in tree t */ +static char *computekids(Tree t, char *v, char *bp, int *ip) { +	Term p = t->op; + +	if (p->kind == NONTERM) { +		sprintf(bp, "\t\tkids[%d] = %s;\n", (*ip)++, v); +		bp += strlen(bp); +	} else if (p->arity > 0) { +		bp = computekids(t->left, stringf("LEFT_CHILD(%s)", v), bp, ip); +		if (p->arity == 2) +			bp = computekids(t->right, stringf("RIGHT_CHILD(%s)", v), bp, ip); +	} +	return bp; +} + +/* emitkids - emit _kids */ +static void emitkids(Rule rules, int nrules) { +	int i; +	Rule r, *rc = alloc((nrules + 1 + 1)*sizeof *rc); +	char **str  = alloc((nrules + 1 + 1)*sizeof *str); + +	for (i = 0, r = rules; r; r = r->link) { +		int j = 0; +		char buf[1024], *bp = buf; +		*computekids(r->pattern, "p", bp, &j) = 0; +		for (j = 0; str[j] && strcmp(str[j], buf); j++) +			; +		if (str[j] == NULL) +			str[j] = strcpy(alloc(strlen(buf) + 1), buf); +		r->kids = rc[j]; +		rc[j] = r; +	} +	print("static void %Pkids(NODEPTR_TYPE p, int eruleno, NODEPTR_TYPE kids[]) {\n" +"%1if (!p)\n%2fatal(\"%Pkids\", \"Null tree\\n\", 0);\n" +"%1if (!kids)\n%2fatal(\"%Pkids\", \"Null kids\\n\", 0);\n" +"%1switch (eruleno) {\n"); +	for (i = 0; (r = rc[i]) != NULL; i++) { +		for ( ; r; r = r->kids) +			print("%1case %d: /* %R */\n", r->ern, r); +		print("%s%2break;\n", str[i]); +	} +	print("%1default:\n%2fatal(\"%Pkids\", \"Bad rule number %%d\\n\", eruleno);\n%1}\n}\n\n"); +} + +/* emitlabel - emit label function */ +static void emitlabel(Term terms, Nonterm start, int ntnumber) { +	int i; +	Term p; + +	print("static void %Plabel(NODEPTR_TYPE a) {\n%1int c;\n" +"%1struct %Pstate *p;\n\n" +"%1if (!a)\n%2fatal(\"%Plabel\", \"Null tree\\n\", 0);\n"); +	print("%1STATE_LABEL(a) = p = allocate(sizeof *p, FUNC);\n" +"%1p->rule._stmt = 0;\n"); +	for (i = 1; i <= ntnumber; i++) +		print("%1p->cost[%d] =\n", i); +	print("%20x7fff;\n%1switch (OP_LABEL(a)) {\n"); +	for (p = terms; p; p = p->link) +		emitcase(p, ntnumber); +	print("%1default:\n" +"%2fatal(\"%Plabel\", \"Bad terminal %%d\\n\", OP_LABEL(a));\n%1}\n}\n\n"); +} + +/* computents - fill in bp with _nts vector for tree t */ +static char *computents(Tree t, char *bp) { +	if (t) { +		Nonterm p = t->op; +		if (p->kind == NONTERM) { +			sprintf(bp, "%s_%s_NT, ", prefix, p->name); +			bp += strlen(bp); +		} else +			bp = computents(t->right, computents(t->left,  bp)); +	} +	return bp; +} + +/* emitnts - emit _nts ragged array */ +static void emitnts(Rule rules, int nrules) { +	Rule r; +	int i, j, *nts = alloc((nrules + 1)*sizeof *nts); +	char **str = alloc((nrules + 1)*sizeof *str); + +	for (i = 0, r = rules; r; r = r->link) { +		char buf[1024]; +		*computents(r->pattern, buf) = 0; +		for (j = 0; str[j] && strcmp(str[j], buf); j++) +			; +		if (str[j] == NULL) { +			print("static short %Pnts_%d[] = { %s0 };\n", j, buf); +			str[j] = strcpy(alloc(strlen(buf) + 1), buf); +		} +		nts[i++] = j; +	} +	print("\nstatic short *%Pnts[] = {\n"); +	for (i = j = 0, r = rules; r; r = r->link) { +		for ( ; j < r->ern; j++) +			print("%10,%1/* %d */\n", j); +		print("%1%Pnts_%d,%1/* %d */\n", nts[i++], j++); +	} +	print("};\n\n"); +} + +/* emitrecalc - emit code that tests for recalculation of INDIR?(VREGP) */ +static void emitrecalc(char *pre, Term root, Term kid) { +	if (root->kind == TERM && strncmp(root->name, "INDIR", 5) == 0 +	&&   kid->kind == TERM &&  strcmp(kid->name,  "VREGP"   ) == 0) { +		Nonterm p; +		print("%sif (mayrecalc(a)) {\n", pre); +		print("%s%1struct %Pstate *q = a->syms[RX]->u.t.cse->x.state;\n", pre); +		for (p = nts; p; p = p->link) { +			print("%s%1if (q->cost[%P%S_NT] == 0) {\n", pre, p); +			print("%s%2p->cost[%P%S_NT] = 0;\n", pre, p); +			print("%s%2p->rule.%P%S = q->rule.%P%S;\n", pre, p, p); +			print("%s%1}\n", pre); +		} +		print("%s}\n", pre); +	} +} + +/* emitrecord - emit code that tests for a winning match of rule r */ +static void emitrecord(char *pre, Rule r, char *c, int cost) { +	if (Tflag) +		print("%s%Ptrace(a, %d, %s + %d, p->cost[%P%S_NT]);\n", +			pre, r->ern, c, cost, r->lhs); +	print("%sif (", pre); +	print("%s + %d < p->cost[%P%S_NT]) {\n" +"%s%1p->cost[%P%S_NT] = %s + %d;\n%s%1p->rule.%P%S = %d;\n", +		c, cost, r->lhs, pre, r->lhs, c, cost, pre, r->lhs, +		r->packed); +	if (r->lhs->chain) +		print("%s%1%Pclosure_%S(a, %s + %d);\n", pre, r->lhs, c, cost); +	print("%s}\n", pre); +} + +/* emitrule - emit decoding vectors and _rule */ +static void emitrule(Nonterm nts) { +	Nonterm p; + +	for (p = nts; p; p = p->link) { +		Rule r; +		print("static short %Pdecode_%S[] = {\n%10,\n", p); +		for (r = p->rules; r; r = r->decode) +			print("%1%d,\n", r->ern); +		print("};\n\n"); +	} +	print("static int %Prule(void *state, int goalnt) {\n" +"%1if (goalnt < 1 || goalnt > %d)\n%2fatal(\"%Prule\", \"Bad goal nonterminal %%d\\n\", goalnt);\n" +"%1if (!state)\n%2return 0;\n%1switch (goalnt) {\n", ntnumber); +	for (p = nts; p; p = p->link) +		print("%1case %P%S_NT:" +"%1return %Pdecode_%S[((struct %Pstate *)state)->rule.%P%S];\n", p, p, p); +	print("%1default:\n%2fatal(\"%Prule\", \"Bad goal nonterminal %%d\\n\", goalnt);\n%2return 0;\n%1}\n}\n\n"); +} + +/* emitstring - emit arrays of templates, instruction flags, and rules */ +static void emitstring(Rule rules) { +	Rule r; + +	print("static char *%Ptemplates[] = {\n"); +	print("/* 0 */%10,\n"); +	for (r = rules; r; r = r->link) +		print("/* %d */%1\"%s\",%1/* %R */\n", r->ern, r->template, r); +	print("};\n"); +	print("\nstatic char %Pisinstruction[] = {\n"); +	print("/* 0 */%10,\n"); +	for (r = rules; r; r = r->link) { +		int len = strlen(r->template); +		print("/* %d */%1%d,%1/* %s */\n", r->ern, +			len >= 2 && r->template[len-2] == '\\' && r->template[len-1] == 'n', +			r->template); +	} +	print("};\n"); +	print("\nstatic char *%Pstring[] = {\n"); +	print("/* 0 */%10,\n"); +	for (r = rules; r; r = r->link) +		print("/* %d */%1\"%R\",\n", r->ern, r); +	print("};\n\n"); +} + +/* emitstruct - emit the definition of the state structure */ +static void emitstruct(Nonterm nts, int ntnumber) { +	print("struct %Pstate {\n%1short cost[%d];\n%1struct {\n", ntnumber + 1); +	for ( ; nts; nts = nts->link) { +		int n = 1, m = nts->lhscount; +		while ((m >>= 1) != 0) +			n++;		 +		print("%2unsigned int %P%S:%d;\n", nts, n); +	} +	print("%1} rule;\n};\n\n"); +} + +/* emittest - emit clause for testing a match */ +static void emittest(Tree t, char *v, char *suffix) { +	Term p = t->op; + +	if (p->kind == TERM) { +		print("%3%s->op == %d%s/* %S */\n", v, p->esn, +			t->nterms > 1 ? " && " : suffix, p); +		if (t->left) +			emittest(t->left, stringf("LEFT_CHILD(%s)",  v), +				t->right && t->right->nterms ? " && " : suffix); +		if (t->right) +			emittest(t->right, stringf("RIGHT_CHILD(%s)", v), suffix); +	} +} diff --git a/src/tools/lcc/lburg/lburg.h b/src/tools/lcc/lburg/lburg.h new file mode 100644 index 0000000..b67e802 --- /dev/null +++ b/src/tools/lcc/lburg/lburg.h @@ -0,0 +1,65 @@ +#ifndef BURG_INCLUDED +#define BURG_INCLUDED + +/* iburg.c: */ +extern void *alloc(int nbytes); + +typedef enum { TERM=1, NONTERM } Kind; +typedef struct rule *Rule; +typedef struct term *Term; +struct term {		/* terminals: */ +	char *name;		/* terminal name */ +	Kind kind;		/* TERM */ +	int esn;		/* external symbol number */ +	int arity;		/* operator arity */ +	Term link;		/* next terminal in esn order */ +	Rule rules;		/* rules whose pattern starts with term */ +}; + +typedef struct nonterm *Nonterm; +struct nonterm {	/* nonterminals: */ +	char *name;		/* nonterminal name */ +	Kind kind;		/* NONTERM */ +	int number;		/* identifying number */ +	int lhscount;		/* # times nt appears in a rule lhs */ +	int reached;		/* 1 iff reached from start nonterminal */ +	Rule rules;		/* rules w/nonterminal on lhs */ +	Rule chain;		/* chain rules w/nonterminal on rhs */ +	Nonterm link;		/* next terminal in number order */ +}; +extern Nonterm nonterm(char *id); +extern Term term(char *id, int esn); + +typedef struct tree *Tree; +struct tree {		/* tree patterns: */ +	void *op;		/* a terminal or nonterminal */ +	Tree left, right;	/* operands */ +	int nterms;		/* number of terminal nodes in this tree */ +}; +extern Tree tree(char *op, Tree left, Tree right); + +struct rule {		/* rules: */ +	Nonterm lhs;		/* lefthand side nonterminal */ +	Tree pattern;		/* rule pattern */ +	int ern;		/* external rule number */ +	int packed;		/* packed external rule number */ +	int cost;		/* cost, if a constant */ +	char *code;		/* cost, if an expression */ +	char *template;		/* assembler template */ +	Rule link;		/* next rule in ern order */ +	Rule next;		/* next rule with same pattern root */ +	Rule chain;		/* next chain rule with same rhs */ +	Rule decode;		/* next rule with same lhs */ +	Rule kids;		/* next rule with same _kids pattern */ +}; +extern Rule rule(char *id, Tree pattern, char *template, char *code); + +/* gram.y: */ +void yyerror(char *fmt, ...); +int yyparse(void); +void yywarn(char *fmt, ...); +extern int errcnt; +extern FILE *infp; +extern FILE *outfp; + +#endif  | 
