| 1 | /* $Id: cgram.y,v 1.319 2011/01/27 18:00:32 ragge Exp $ */
|
|---|
| 2 |
|
|---|
| 3 | /*
|
|---|
| 4 | * Copyright (c) 2003 Anders Magnusson (ragge@ludd.luth.se).
|
|---|
| 5 | * All rights reserved.
|
|---|
| 6 | *
|
|---|
| 7 | * Redistribution and use in source and binary forms, with or without
|
|---|
| 8 | * modification, are permitted provided that the following conditions
|
|---|
| 9 | * are met:
|
|---|
| 10 | * 1. Redistributions of source code must retain the above copyright
|
|---|
| 11 | * notice, this list of conditions and the following disclaimer.
|
|---|
| 12 | * 2. Redistributions in binary form must reproduce the above copyright
|
|---|
| 13 | * notice, this list of conditions and the following disclaimer in the
|
|---|
| 14 | * documentation and/or other materials provided with the distribution.
|
|---|
| 15 | * 3. The name of the author may not be used to endorse or promote products
|
|---|
| 16 | * derived from this software without specific prior written permission
|
|---|
| 17 | *
|
|---|
| 18 | * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|---|
| 19 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|---|
| 20 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|---|
| 21 | * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|---|
| 22 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|---|
| 23 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|---|
| 24 | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|---|
| 25 | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|---|
| 26 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|---|
| 27 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|---|
| 28 | */
|
|---|
| 29 |
|
|---|
| 30 | /*
|
|---|
| 31 | * Copyright(C) Caldera International Inc. 2001-2002. All rights reserved.
|
|---|
| 32 | *
|
|---|
| 33 | * Redistribution and use in source and binary forms, with or without
|
|---|
| 34 | * modification, are permitted provided that the following conditions
|
|---|
| 35 | * are met:
|
|---|
| 36 | *
|
|---|
| 37 | * Redistributions of source code and documentation must retain the above
|
|---|
| 38 | * copyright notice, this list of conditions and the following disclaimer.
|
|---|
| 39 | * Redistributions in binary form must reproduce the above copyright
|
|---|
| 40 | * notice, this list of conditions and the following disclaimer in the
|
|---|
| 41 | * documentation and/or other materials provided with the distribution.
|
|---|
| 42 | * All advertising materials mentioning features or use of this software
|
|---|
| 43 | * must display the following acknowledgement:
|
|---|
| 44 | * This product includes software developed or owned by Caldera
|
|---|
| 45 | * International, Inc.
|
|---|
| 46 | * Neither the name of Caldera International, Inc. nor the names of other
|
|---|
| 47 | * contributors may be used to endorse or promote products derived from
|
|---|
| 48 | * this software without specific prior written permission.
|
|---|
| 49 | *
|
|---|
| 50 | * USE OF THE SOFTWARE PROVIDED FOR UNDER THIS LICENSE BY CALDERA
|
|---|
| 51 | * INTERNATIONAL, INC. AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR
|
|---|
| 52 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|---|
| 53 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|---|
| 54 | * DISCLAIMED. IN NO EVENT SHALL CALDERA INTERNATIONAL, INC. BE LIABLE
|
|---|
| 55 | * FOR ANY DIRECT, INDIRECT INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|---|
| 56 | * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|---|
| 57 | * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|---|
| 58 | * HOWEVER CAUSED AND ON ANY THEORY OFLIABILITY, WHETHER IN CONTRACT,
|
|---|
| 59 | * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|---|
| 60 | * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|---|
| 61 | * POSSIBILITY OF SUCH DAMAGE.
|
|---|
| 62 | */
|
|---|
| 63 |
|
|---|
| 64 | /*
|
|---|
| 65 | * Comments for this grammar file. Ragge 021123
|
|---|
| 66 | *
|
|---|
| 67 | * ANSI support required rewrite of the function header and declaration
|
|---|
| 68 | * rules almost totally.
|
|---|
| 69 | *
|
|---|
| 70 | * The lex/yacc shared keywords are now split from the keywords used
|
|---|
| 71 | * in the rest of the compiler, to simplify use of other frontends.
|
|---|
| 72 | */
|
|---|
| 73 |
|
|---|
| 74 | /*
|
|---|
| 75 | * At last count, there were 5 shift/reduce and no reduce/reduce conflicts
|
|---|
| 76 | * Four are accounted for;
|
|---|
| 77 | * One is "dangling else"
|
|---|
| 78 | * Two is in attribute parsing
|
|---|
| 79 | * One is in ({ }) parsing
|
|---|
| 80 | */
|
|---|
| 81 |
|
|---|
| 82 | /*
|
|---|
| 83 | * Token used in C lex/yacc communications.
|
|---|
| 84 | */
|
|---|
| 85 | %token C_STRING /* a string constant */
|
|---|
| 86 | %token C_ICON /* an integer constant */
|
|---|
| 87 | %token C_FCON /* a floating point constant */
|
|---|
| 88 | %token C_NAME /* an identifier */
|
|---|
| 89 | %token C_TYPENAME /* a typedef'd name */
|
|---|
| 90 | %token C_ANDAND /* && */
|
|---|
| 91 | %token C_OROR /* || */
|
|---|
| 92 | %token C_GOTO /* unconditional goto */
|
|---|
| 93 | %token C_RETURN /* return from function */
|
|---|
| 94 | %token C_TYPE /* a type */
|
|---|
| 95 | %token C_CLASS /* a storage class */
|
|---|
| 96 | %token C_ASOP /* assignment ops */
|
|---|
| 97 | %token C_RELOP /* <=, <, >=, > */
|
|---|
| 98 | %token C_EQUOP /* ==, != */
|
|---|
| 99 | %token C_DIVOP /* /, % */
|
|---|
| 100 | %token C_SHIFTOP /* <<, >> */
|
|---|
| 101 | %token C_INCOP /* ++, -- */
|
|---|
| 102 | %token C_UNOP /* !, ~ */
|
|---|
| 103 | %token C_STROP /* ., -> */
|
|---|
| 104 | %token C_STRUCT
|
|---|
| 105 | %token C_IF
|
|---|
| 106 | %token C_ELSE
|
|---|
| 107 | %token C_SWITCH
|
|---|
| 108 | %token C_BREAK
|
|---|
| 109 | %token C_CONTINUE
|
|---|
| 110 | %token C_WHILE
|
|---|
| 111 | %token C_DO
|
|---|
| 112 | %token C_FOR
|
|---|
| 113 | %token C_DEFAULT
|
|---|
| 114 | %token C_CASE
|
|---|
| 115 | %token C_SIZEOF
|
|---|
| 116 | %token C_ALIGNOF
|
|---|
| 117 | %token C_ENUM
|
|---|
| 118 | %token C_ELLIPSIS
|
|---|
| 119 | %token C_QUALIFIER
|
|---|
| 120 | %token C_FUNSPEC
|
|---|
| 121 | %token C_ASM
|
|---|
| 122 | %token NOMATCH
|
|---|
| 123 | %token C_TYPEOF /* COMPAT_GCC */
|
|---|
| 124 | %token C_ATTRIBUTE /* COMPAT_GCC */
|
|---|
| 125 | %token PCC_OFFSETOF
|
|---|
| 126 |
|
|---|
| 127 | /*
|
|---|
| 128 | * Precedence
|
|---|
| 129 | */
|
|---|
| 130 | %left ','
|
|---|
| 131 | %right '=' C_ASOP
|
|---|
| 132 | %right '?' ':'
|
|---|
| 133 | %left C_OROR
|
|---|
| 134 | %left C_ANDAND
|
|---|
| 135 | %left '|'
|
|---|
| 136 | %left '^'
|
|---|
| 137 | %left '&'
|
|---|
| 138 | %left C_EQUOP
|
|---|
| 139 | %left C_RELOP
|
|---|
| 140 | %left C_SHIFTOP
|
|---|
| 141 | %left '+' '-'
|
|---|
| 142 | %left '*' C_DIVOP
|
|---|
| 143 | %right C_UNOP
|
|---|
| 144 | %right C_INCOP C_SIZEOF
|
|---|
| 145 | %left '[' '(' C_STROP
|
|---|
| 146 | %{
|
|---|
| 147 | # include "pass1.h"
|
|---|
| 148 | # include <stdarg.h>
|
|---|
| 149 | # include <string.h>
|
|---|
| 150 | # include <stdlib.h>
|
|---|
| 151 |
|
|---|
| 152 | int fun_inline; /* Reading an inline function */
|
|---|
| 153 | int oldstyle; /* Current function being defined */
|
|---|
| 154 | static struct symtab *xnf;
|
|---|
| 155 | extern int enummer, tvaloff, inattr;
|
|---|
| 156 | extern struct rstack *rpole;
|
|---|
| 157 | static int widestr, alwinl;
|
|---|
| 158 | NODE *cftnod;
|
|---|
| 159 | static int attrwarn = 1;
|
|---|
| 160 |
|
|---|
| 161 | #define NORETYP SNOCREAT /* no return type, save in unused field in symtab */
|
|---|
| 162 |
|
|---|
| 163 | NODE *bdty(int op, ...);
|
|---|
| 164 | static void fend(void);
|
|---|
| 165 | static void fundef(NODE *tp, NODE *p);
|
|---|
| 166 | static void olddecl(NODE *p, NODE *a);
|
|---|
| 167 | static struct symtab *init_declarator(NODE *tn, NODE *p, int assign, NODE *a);
|
|---|
| 168 | static void resetbc(int mask);
|
|---|
| 169 | static void swend(void);
|
|---|
| 170 | static void addcase(NODE *p);
|
|---|
| 171 | #ifdef GCC_COMPAT
|
|---|
| 172 | static void gcccase(NODE *p, NODE *);
|
|---|
| 173 | #endif
|
|---|
| 174 | static void adddef(void);
|
|---|
| 175 | static void savebc(void);
|
|---|
| 176 | static void swstart(int, TWORD);
|
|---|
| 177 | static void genswitch(int, TWORD, struct swents **, int);
|
|---|
| 178 | static char *mkpstr(char *str);
|
|---|
| 179 | static struct symtab *clbrace(NODE *);
|
|---|
| 180 | static NODE *cmop(NODE *l, NODE *r);
|
|---|
| 181 | static NODE *xcmop(NODE *out, NODE *in, NODE *str);
|
|---|
| 182 | static void mkxasm(char *str, NODE *p);
|
|---|
| 183 | static NODE *xasmop(char *str, NODE *p);
|
|---|
| 184 | static int maxstlen(char *str);
|
|---|
| 185 | static char *stradd(char *old, char *new);
|
|---|
| 186 | static NODE *biop(int op, NODE *l, NODE *r);
|
|---|
| 187 | static void flend(void);
|
|---|
| 188 | static char * simname(char *s);
|
|---|
| 189 | #ifdef GCC_COMPAT
|
|---|
| 190 | static NODE *tyof(NODE *); /* COMPAT_GCC */
|
|---|
| 191 | static NODE *voidcon(void); /* COMPAT_GCC */
|
|---|
| 192 | #endif
|
|---|
| 193 | static NODE *funargs(NODE *p);
|
|---|
| 194 | static void oldargs(NODE *p);
|
|---|
| 195 | static void uawarn(NODE *p, char *s);
|
|---|
| 196 | static int con_e(NODE *p);
|
|---|
| 197 | static void dainit(NODE *d, NODE *a);
|
|---|
| 198 | static NODE *tymfix(NODE *p);
|
|---|
| 199 | static NODE *namekill(NODE *p, int clr);
|
|---|
| 200 | static NODE *aryfix(NODE *p);
|
|---|
| 201 |
|
|---|
| 202 | #define TYMFIX(inp) { \
|
|---|
| 203 | NODE *pp = inp; \
|
|---|
| 204 | inp = tymerge(pp->n_left, pp->n_right); \
|
|---|
| 205 | nfree(pp->n_left); nfree(pp); }
|
|---|
| 206 | /*
|
|---|
| 207 | * State for saving current switch state (when nested switches).
|
|---|
| 208 | */
|
|---|
| 209 | struct savbc {
|
|---|
| 210 | struct savbc *next;
|
|---|
| 211 | int brklab;
|
|---|
| 212 | int contlab;
|
|---|
| 213 | int flostat;
|
|---|
| 214 | int swx;
|
|---|
| 215 | } *savbc, *savctx;
|
|---|
| 216 |
|
|---|
| 217 | %}
|
|---|
| 218 |
|
|---|
| 219 | %union {
|
|---|
| 220 | int intval;
|
|---|
| 221 | NODE *nodep;
|
|---|
| 222 | struct symtab *symp;
|
|---|
| 223 | struct rstack *rp;
|
|---|
| 224 | char *strp;
|
|---|
| 225 | }
|
|---|
| 226 |
|
|---|
| 227 | /* define types */
|
|---|
| 228 | %start ext_def_list
|
|---|
| 229 |
|
|---|
| 230 | %type <intval> ifelprefix ifprefix whprefix forprefix doprefix switchpart
|
|---|
| 231 | xbegin
|
|---|
| 232 | %type <nodep> e .e term enum_dcl struct_dcl cast_type declarator
|
|---|
| 233 | elist type_sq cf_spec merge_attribs
|
|---|
| 234 | parameter_declaration abstract_declarator initializer
|
|---|
| 235 | parameter_type_list parameter_list addrlbl
|
|---|
| 236 | declaration_specifiers designation
|
|---|
| 237 | specifier_qualifier_list merge_specifiers
|
|---|
| 238 | identifier_list arg_param_list type_qualifier_list
|
|---|
| 239 | designator_list designator xasm oplist oper cnstr funtype
|
|---|
| 240 | typeof attribute attribute_specifier /* COMPAT_GCC */
|
|---|
| 241 | attribute_list attr_spec_list attr_var /* COMPAT_GCC */
|
|---|
| 242 | %type <strp> string C_STRING
|
|---|
| 243 | %type <rp> str_head
|
|---|
| 244 | %type <symp> xnfdeclarator clbrace enum_head
|
|---|
| 245 |
|
|---|
| 246 | %type <intval> C_STRUCT C_RELOP C_DIVOP C_SHIFTOP
|
|---|
| 247 | C_ANDAND C_OROR C_STROP C_INCOP C_UNOP C_ASOP C_EQUOP
|
|---|
| 248 |
|
|---|
| 249 | %type <nodep> C_TYPE C_QUALIFIER C_ICON C_FCON C_CLASS
|
|---|
| 250 | %type <strp> C_NAME C_TYPENAME
|
|---|
| 251 | %%
|
|---|
| 252 |
|
|---|
| 253 | ext_def_list: ext_def_list external_def
|
|---|
| 254 | | { ftnend(); }
|
|---|
| 255 | ;
|
|---|
| 256 |
|
|---|
| 257 | external_def: funtype kr_args compoundstmt { fend(); }
|
|---|
| 258 | | declaration { blevel = 0; symclear(0); }
|
|---|
| 259 | | asmstatement ';'
|
|---|
| 260 | | ';'
|
|---|
| 261 | | error { blevel = 0; }
|
|---|
| 262 | ;
|
|---|
| 263 |
|
|---|
| 264 | funtype: /* no type given */ declarator {
|
|---|
| 265 | fundef(mkty(INT, 0, MKAP(INT)), $1);
|
|---|
| 266 | cftnsp->sflags |= NORETYP;
|
|---|
| 267 | }
|
|---|
| 268 | | declaration_specifiers declarator { fundef($1,$2); }
|
|---|
| 269 | ;
|
|---|
| 270 |
|
|---|
| 271 | kr_args: /* empty */
|
|---|
| 272 | | arg_dcl_list
|
|---|
| 273 | ;
|
|---|
| 274 |
|
|---|
| 275 | /*
|
|---|
| 276 | * Returns a node pointer or NULL, if no types at all given.
|
|---|
| 277 | * Type trees are checked for correctness and merged into one
|
|---|
| 278 | * type node in typenode().
|
|---|
| 279 | */
|
|---|
| 280 | declaration_specifiers:
|
|---|
| 281 | merge_attribs { $$ = typenode($1); }
|
|---|
| 282 | ;
|
|---|
| 283 |
|
|---|
| 284 | merge_attribs: type_sq { $$ = $1; }
|
|---|
| 285 | | type_sq merge_attribs { $$ = cmop($2, $1); }
|
|---|
| 286 | | cf_spec { $$ = $1; }
|
|---|
| 287 | | cf_spec merge_attribs { $$ = cmop($2, $1); }
|
|---|
| 288 | ;
|
|---|
| 289 |
|
|---|
| 290 | type_sq: C_TYPE { $$ = $1; }
|
|---|
| 291 | | C_TYPENAME {
|
|---|
| 292 | struct symtab *sp = lookup($1, 0);
|
|---|
| 293 | if (sp->stype == ENUMTY) {
|
|---|
| 294 | sp->stype = strmemb(sp->sap)->stype;
|
|---|
| 295 | }
|
|---|
| 296 | $$ = mkty(sp->stype, sp->sdf, sp->sap);
|
|---|
| 297 | $$->n_sp = sp;
|
|---|
| 298 | }
|
|---|
| 299 | | struct_dcl { $$ = $1; }
|
|---|
| 300 | | enum_dcl { $$ = $1; }
|
|---|
| 301 | | C_QUALIFIER { $$ = $1; }
|
|---|
| 302 | | attribute_specifier { $$ = biop(ATTRIB, $1, 0); }
|
|---|
| 303 | | typeof { $$ = $1; }
|
|---|
| 304 | ;
|
|---|
| 305 |
|
|---|
| 306 | cf_spec: C_CLASS { $$ = $1; }
|
|---|
| 307 | | C_FUNSPEC { fun_inline = 1; /* XXX - hack */
|
|---|
| 308 | $$ = block(QUALIFIER, NIL, NIL, 0, 0, 0); }
|
|---|
| 309 | ;
|
|---|
| 310 |
|
|---|
| 311 | typeof: C_TYPEOF '(' term ')' { $$ = tyof(eve($3)); }
|
|---|
| 312 | | C_TYPEOF '(' cast_type ')' { TYMFIX($3); $$ = tyof($3); }
|
|---|
| 313 | ;
|
|---|
| 314 |
|
|---|
| 315 | attribute_specifier :
|
|---|
| 316 | C_ATTRIBUTE '(' '(' attribute_list ')' ')' { $$ = $4; }
|
|---|
| 317 | /*COMPAT_GCC*/ ;
|
|---|
| 318 |
|
|---|
| 319 | attribute_list: attribute
|
|---|
| 320 | | attribute ',' attribute_list { $$ = cmop($3, $1); }
|
|---|
| 321 | ;
|
|---|
| 322 |
|
|---|
| 323 | attribute: {
|
|---|
| 324 | #ifdef GCC_COMPAT
|
|---|
| 325 | $$ = voidcon();
|
|---|
| 326 | #endif
|
|---|
| 327 | }
|
|---|
| 328 | | C_NAME { $$ = bdty(NAME, $1); }
|
|---|
| 329 | | C_NAME '(' elist ')' {
|
|---|
| 330 | $$ = bdty($3 == NIL ? UCALL : CALL, bdty(NAME, $1), $3);
|
|---|
| 331 | }
|
|---|
| 332 | ;
|
|---|
| 333 |
|
|---|
| 334 | /*
|
|---|
| 335 | * Adds a pointer list to front of the declarators.
|
|---|
| 336 | */
|
|---|
| 337 | declarator: '*' declarator { $$ = bdty(UMUL, $2); }
|
|---|
| 338 | | '*' type_qualifier_list declarator {
|
|---|
| 339 | $$ = $2;
|
|---|
| 340 | $$->n_left = $3;
|
|---|
| 341 | }
|
|---|
| 342 | | C_NAME { $$ = bdty(NAME, $1); }
|
|---|
| 343 | | '(' attr_spec_list declarator ')' {
|
|---|
| 344 | $$ = $3;
|
|---|
| 345 | $$->n_ap = attr_add($$->n_ap, gcc_attr_parse($2));
|
|---|
| 346 | }
|
|---|
| 347 | | '(' declarator ')' { $$ = $2; }
|
|---|
| 348 | | declarator '[' e ']' { $$ = biop(LB, $1, $3); }
|
|---|
| 349 | | declarator '[' C_CLASS e ']' {
|
|---|
| 350 | if ($3->n_type != STATIC)
|
|---|
| 351 | uerror("bad class keyword");
|
|---|
| 352 | tfree($3); /* XXX - handle */
|
|---|
| 353 | $$ = biop(LB, $1, $4);
|
|---|
| 354 | }
|
|---|
| 355 | | declarator '[' ']' { $$ = biop(LB, $1, bcon(NOOFFSET)); }
|
|---|
| 356 | | declarator '[' '*' ']' { $$ = biop(LB, $1, bcon(NOOFFSET)); }
|
|---|
| 357 | | declarator '(' parameter_type_list ')' {
|
|---|
| 358 | $$ = bdty(CALL, $1, $3);
|
|---|
| 359 | }
|
|---|
| 360 | | declarator '(' identifier_list ')' {
|
|---|
| 361 | $$ = bdty(CALL, $1, $3);
|
|---|
| 362 | oldstyle = 1;
|
|---|
| 363 | }
|
|---|
| 364 | | declarator '(' ')' { $$ = bdty(UCALL, $1); }
|
|---|
| 365 | ;
|
|---|
| 366 |
|
|---|
| 367 | type_qualifier_list:
|
|---|
| 368 | C_QUALIFIER { $$ = $1; $$->n_op = UMUL; }
|
|---|
| 369 | | type_qualifier_list C_QUALIFIER {
|
|---|
| 370 | $$ = $1;
|
|---|
| 371 | $$->n_qual |= $2->n_qual;
|
|---|
| 372 | nfree($2);
|
|---|
| 373 | }
|
|---|
| 374 | | attribute_specifier {
|
|---|
| 375 | $$ = block(UMUL, NIL, NIL, 0, 0, gcc_attr_parse($1));
|
|---|
| 376 | }
|
|---|
| 377 | | type_qualifier_list attribute_specifier {
|
|---|
| 378 | $1->n_ap = attr_add($1->n_ap, gcc_attr_parse($2));
|
|---|
| 379 | }
|
|---|
| 380 | ;
|
|---|
| 381 |
|
|---|
| 382 | identifier_list: C_NAME { $$ = bdty(NAME, $1); oldargs($$); }
|
|---|
| 383 | | identifier_list ',' C_NAME {
|
|---|
| 384 | $$ = cmop($1, bdty(NAME, $3));
|
|---|
| 385 | oldargs($$->n_right);
|
|---|
| 386 | }
|
|---|
| 387 | ;
|
|---|
| 388 |
|
|---|
| 389 | /*
|
|---|
| 390 | * Returns as parameter_list, but can add an additional ELLIPSIS node.
|
|---|
| 391 | */
|
|---|
| 392 | parameter_type_list:
|
|---|
| 393 | parameter_list { $$ = $1; }
|
|---|
| 394 | | parameter_list ',' C_ELLIPSIS {
|
|---|
| 395 | $$ = cmop($1, biop(ELLIPSIS, NIL, NIL));
|
|---|
| 396 | }
|
|---|
| 397 | ;
|
|---|
| 398 |
|
|---|
| 399 | /*
|
|---|
| 400 | * Returns a linked lists of nodes of op CM with parameters on
|
|---|
| 401 | * its right and additional CM nodes of its left pointer.
|
|---|
| 402 | * No CM nodes if only one parameter.
|
|---|
| 403 | */
|
|---|
| 404 | parameter_list: parameter_declaration { $$ = $1; }
|
|---|
| 405 | | parameter_list ',' parameter_declaration {
|
|---|
| 406 | $$ = cmop($1, $3);
|
|---|
| 407 | }
|
|---|
| 408 | ;
|
|---|
| 409 |
|
|---|
| 410 | /*
|
|---|
| 411 | * Returns a node pointer to the declaration.
|
|---|
| 412 | */
|
|---|
| 413 | parameter_declaration:
|
|---|
| 414 | declaration_specifiers declarator attr_var {
|
|---|
| 415 | if ($1->n_lval != SNULL && $1->n_lval != REGISTER)
|
|---|
| 416 | uerror("illegal parameter class");
|
|---|
| 417 | $$ = block(TYMERGE, $1, $2, INT, 0, gcc_attr_parse($3));
|
|---|
| 418 | }
|
|---|
| 419 | | declaration_specifiers abstract_declarator {
|
|---|
| 420 | $$ = block(TYMERGE, $1, $2, INT, 0, 0);
|
|---|
| 421 | }
|
|---|
| 422 | | declaration_specifiers {
|
|---|
| 423 | $$ = block(TYMERGE, $1, bdty(NAME, NULL), INT, 0, 0);
|
|---|
| 424 | }
|
|---|
| 425 | ;
|
|---|
| 426 |
|
|---|
| 427 | abstract_declarator:
|
|---|
| 428 | '*' { $$ = bdty(UMUL, bdty(NAME, NULL)); }
|
|---|
| 429 | | '*' type_qualifier_list {
|
|---|
| 430 | $$ = $2;
|
|---|
| 431 | $$->n_left = bdty(NAME, NULL);
|
|---|
| 432 | }
|
|---|
| 433 | | '*' abstract_declarator { $$ = bdty(UMUL, $2); }
|
|---|
| 434 | | '*' type_qualifier_list abstract_declarator {
|
|---|
| 435 | $$ = $2;
|
|---|
| 436 | $$->n_left = $3;
|
|---|
| 437 | }
|
|---|
| 438 | | '(' abstract_declarator ')' { $$ = $2; }
|
|---|
| 439 | | '[' ']' attr_var {
|
|---|
| 440 | $$ = block(LB, bdty(NAME, NULL), bcon(NOOFFSET),
|
|---|
| 441 | INT, 0, gcc_attr_parse($3));
|
|---|
| 442 | }
|
|---|
| 443 | | '[' e ']' attr_var {
|
|---|
| 444 | $$ = block(LB, bdty(NAME, NULL), $2,
|
|---|
| 445 | INT, 0, gcc_attr_parse($4));
|
|---|
| 446 | }
|
|---|
| 447 | | abstract_declarator '[' ']' attr_var {
|
|---|
| 448 | $$ = block(LB, $1, bcon(NOOFFSET),
|
|---|
| 449 | INT, 0, gcc_attr_parse($4));
|
|---|
| 450 | }
|
|---|
| 451 | | abstract_declarator '[' e ']' attr_var {
|
|---|
| 452 | $$ = block(LB, $1, $3, INT, 0, gcc_attr_parse($5));
|
|---|
| 453 | }
|
|---|
| 454 | | '(' ')' { $$ = bdty(UCALL, bdty(NAME, NULL)); }
|
|---|
| 455 | | '(' ib2 parameter_type_list ')' {
|
|---|
| 456 | $$ = bdty(CALL, bdty(NAME, NULL), $3);
|
|---|
| 457 | }
|
|---|
| 458 | | abstract_declarator '(' ')' {
|
|---|
| 459 | $$ = bdty(UCALL, $1);
|
|---|
| 460 | }
|
|---|
| 461 | | abstract_declarator '(' ib2 parameter_type_list ')' {
|
|---|
| 462 | $$ = bdty(CALL, $1, $4);
|
|---|
| 463 | }
|
|---|
| 464 | ;
|
|---|
| 465 |
|
|---|
| 466 | ib2: { }
|
|---|
| 467 | ;
|
|---|
| 468 | /*
|
|---|
| 469 | * K&R arg declaration, between ) and {
|
|---|
| 470 | */
|
|---|
| 471 | arg_dcl_list: arg_declaration
|
|---|
| 472 | | arg_dcl_list arg_declaration
|
|---|
| 473 | ;
|
|---|
| 474 |
|
|---|
| 475 |
|
|---|
| 476 | arg_declaration: declaration_specifiers arg_param_list ';' {
|
|---|
| 477 | nfree($1);
|
|---|
| 478 | }
|
|---|
| 479 | ;
|
|---|
| 480 |
|
|---|
| 481 | arg_param_list: declarator attr_var {
|
|---|
| 482 | olddecl(block(TYMERGE, ccopy($<nodep>0), $1,
|
|---|
| 483 | INT, 0, 0), $2);
|
|---|
| 484 | }
|
|---|
| 485 | | arg_param_list ',' declarator attr_var {
|
|---|
| 486 | olddecl(block(TYMERGE, ccopy($<nodep>0), $3,
|
|---|
| 487 | INT, 0, 0), $4);
|
|---|
| 488 | }
|
|---|
| 489 | ;
|
|---|
| 490 |
|
|---|
| 491 | /*
|
|---|
| 492 | * Declarations in beginning of blocks.
|
|---|
| 493 | */
|
|---|
| 494 | block_item_list: block_item
|
|---|
| 495 | | block_item_list block_item
|
|---|
| 496 | ;
|
|---|
| 497 |
|
|---|
| 498 | block_item: declaration
|
|---|
| 499 | | statement
|
|---|
| 500 | ;
|
|---|
| 501 |
|
|---|
| 502 | /*
|
|---|
| 503 | * Here starts the old YACC code.
|
|---|
| 504 | */
|
|---|
| 505 |
|
|---|
| 506 | /*
|
|---|
| 507 | * Variables are declared in init_declarator.
|
|---|
| 508 | */
|
|---|
| 509 | declaration: declaration_specifiers ';' { tfree($1); fun_inline = 0; }
|
|---|
| 510 | | declaration_specifiers init_declarator_list ';' {
|
|---|
| 511 | tfree($1);
|
|---|
| 512 | fun_inline = 0;
|
|---|
| 513 | }
|
|---|
| 514 | ;
|
|---|
| 515 |
|
|---|
| 516 | /*
|
|---|
| 517 | * Normal declaration of variables. curtype contains the current type node.
|
|---|
| 518 | * Returns nothing, variables are declared in init_declarator.
|
|---|
| 519 | */
|
|---|
| 520 | init_declarator_list:
|
|---|
| 521 | init_declarator { symclear(blevel); }
|
|---|
| 522 | | init_declarator_list ',' attr_var { $<nodep>$ = $<nodep>0; } init_declarator {
|
|---|
| 523 | uawarn($3, "init_declarator");
|
|---|
| 524 | symclear(blevel);
|
|---|
| 525 | }
|
|---|
| 526 | ;
|
|---|
| 527 |
|
|---|
| 528 | enum_dcl: enum_head '{' moe_list optcomma '}' { $$ = enumdcl($1); }
|
|---|
| 529 | | C_ENUM C_NAME { $$ = enumref($2); }
|
|---|
| 530 | ;
|
|---|
| 531 |
|
|---|
| 532 | enum_head: C_ENUM { $$ = enumhd(NULL); }
|
|---|
| 533 | | C_ENUM C_NAME { $$ = enumhd($2); }
|
|---|
| 534 | ;
|
|---|
| 535 |
|
|---|
| 536 | moe_list: moe
|
|---|
| 537 | | moe_list ',' moe
|
|---|
| 538 | ;
|
|---|
| 539 |
|
|---|
| 540 | moe: C_NAME { moedef($1); }
|
|---|
| 541 | | C_TYPENAME { moedef($1); }
|
|---|
| 542 | | C_NAME '=' e { enummer = con_e($3); moedef($1); }
|
|---|
| 543 | | C_TYPENAME '=' e { enummer = con_e($3); moedef($1); }
|
|---|
| 544 | ;
|
|---|
| 545 |
|
|---|
| 546 | struct_dcl: str_head '{' struct_dcl_list '}' {
|
|---|
| 547 | NODE *p;
|
|---|
| 548 |
|
|---|
| 549 | $$ = dclstruct($1);
|
|---|
| 550 | if (pragma_allpacked) {
|
|---|
| 551 | p = bdty(CALL, bdty(NAME, "packed"),
|
|---|
| 552 | bcon(pragma_allpacked));
|
|---|
| 553 | $$ = cmop(biop(ATTRIB, p, 0), $$);
|
|---|
| 554 | }
|
|---|
| 555 | }
|
|---|
| 556 | | C_STRUCT attr_var C_NAME {
|
|---|
| 557 | $$ = rstruct($3,$1);
|
|---|
| 558 | uawarn($2, "struct_dcl");
|
|---|
| 559 | }
|
|---|
| 560 | /*COMPAT_GCC*/ | str_head '{' '}' { $$ = dclstruct($1); }
|
|---|
| 561 | ;
|
|---|
| 562 |
|
|---|
| 563 | attr_var: {
|
|---|
| 564 | NODE *q, *p;
|
|---|
| 565 |
|
|---|
| 566 | p = pragma_aligned ? bdty(CALL, bdty(NAME, "aligned"),
|
|---|
| 567 | bcon(pragma_aligned)) : NIL;
|
|---|
| 568 | if (pragma_packed) {
|
|---|
| 569 | q = bdty(NAME, "packed");
|
|---|
| 570 | p = (p == NIL ? q : cmop(p, q));
|
|---|
| 571 | }
|
|---|
| 572 | pragma_aligned = pragma_packed = 0;
|
|---|
| 573 | $$ = p;
|
|---|
| 574 | }
|
|---|
| 575 | /*COMPAT_GCC*/ | attr_spec_list
|
|---|
| 576 | ;
|
|---|
| 577 |
|
|---|
| 578 | attr_spec_list: attribute_specifier
|
|---|
| 579 | | attr_spec_list attribute_specifier { $$ = cmop($1, $2); }
|
|---|
| 580 | ;
|
|---|
| 581 |
|
|---|
| 582 | str_head: C_STRUCT attr_var { $$ = bstruct(NULL, $1, $2); }
|
|---|
| 583 | | C_STRUCT attr_var C_NAME { $$ = bstruct($3, $1, $2); }
|
|---|
| 584 | ;
|
|---|
| 585 |
|
|---|
| 586 | struct_dcl_list: struct_declaration
|
|---|
| 587 | | struct_dcl_list struct_declaration
|
|---|
| 588 | ;
|
|---|
| 589 |
|
|---|
| 590 | struct_declaration:
|
|---|
| 591 | specifier_qualifier_list struct_declarator_list optsemi {
|
|---|
| 592 | tfree($1);
|
|---|
| 593 | }
|
|---|
| 594 | ;
|
|---|
| 595 |
|
|---|
| 596 | optsemi: ';' { }
|
|---|
| 597 | | optsemi ';' { werror("extra ; in struct"); }
|
|---|
| 598 | ;
|
|---|
| 599 |
|
|---|
| 600 | specifier_qualifier_list:
|
|---|
| 601 | merge_specifiers { $$ = typenode($1); }
|
|---|
| 602 | ;
|
|---|
| 603 |
|
|---|
| 604 | merge_specifiers: type_sq merge_specifiers { $$ = cmop($2, $1); }
|
|---|
| 605 | | type_sq { $$ = $1; }
|
|---|
| 606 | ;
|
|---|
| 607 |
|
|---|
| 608 | struct_declarator_list:
|
|---|
| 609 | struct_declarator { symclear(blevel); }
|
|---|
| 610 | | struct_declarator_list ',' { $<nodep>$=$<nodep>0; }
|
|---|
| 611 | struct_declarator { symclear(blevel); }
|
|---|
| 612 | ;
|
|---|
| 613 |
|
|---|
| 614 | struct_declarator: declarator attr_var {
|
|---|
| 615 | NODE *p;
|
|---|
| 616 |
|
|---|
| 617 | $1 = aryfix($1);
|
|---|
| 618 | p = tymerge($<nodep>0, tymfix($1));
|
|---|
| 619 | if ($2)
|
|---|
| 620 | p->n_ap = attr_add(p->n_ap, gcc_attr_parse($2));
|
|---|
| 621 | soumemb(p, (char *)$1->n_sp, 0);
|
|---|
| 622 | tfree(p);
|
|---|
| 623 | }
|
|---|
| 624 | | ':' e {
|
|---|
| 625 | int ie = con_e($2);
|
|---|
| 626 | if (fldchk(ie))
|
|---|
| 627 | ie = 1;
|
|---|
| 628 | falloc(NULL, ie, $<nodep>0);
|
|---|
| 629 | }
|
|---|
| 630 | | declarator ':' e {
|
|---|
| 631 | int ie = con_e($3);
|
|---|
| 632 | if (fldchk(ie))
|
|---|
| 633 | ie = 1;
|
|---|
| 634 | if ($1->n_op == NAME) {
|
|---|
| 635 | /* XXX - tymfix() may alter $1 */
|
|---|
| 636 | tymerge($<nodep>0, tymfix($1));
|
|---|
| 637 | soumemb($1, (char *)$1->n_sp, FIELD | ie);
|
|---|
| 638 | nfree($1);
|
|---|
| 639 | } else
|
|---|
| 640 | uerror("illegal declarator");
|
|---|
| 641 | }
|
|---|
| 642 | | declarator ':' e attr_spec_list {
|
|---|
| 643 | int ie = con_e($3);
|
|---|
| 644 | if (fldchk(ie))
|
|---|
| 645 | ie = 1;
|
|---|
| 646 | if ($1->n_op == NAME) {
|
|---|
| 647 | /* XXX - tymfix() may alter $1 */
|
|---|
| 648 | tymerge($<nodep>0, tymfix($1));
|
|---|
| 649 | if ($4)
|
|---|
| 650 | $1->n_ap = attr_add($1->n_ap,
|
|---|
| 651 | gcc_attr_parse($4));
|
|---|
| 652 | soumemb($1, (char *)$1->n_sp, FIELD | ie);
|
|---|
| 653 | nfree($1);
|
|---|
| 654 | } else
|
|---|
| 655 | uerror("illegal declarator");
|
|---|
| 656 | }
|
|---|
| 657 | | /* unnamed member */ {
|
|---|
| 658 | NODE *p = $<nodep>0;
|
|---|
| 659 | char *c = permalloc(10);
|
|---|
| 660 |
|
|---|
| 661 | if (p->n_type != STRTY && p->n_type != UNIONTY)
|
|---|
| 662 | uerror("bad unnamed member type");
|
|---|
| 663 | snprintf(c, 10, "*%dFAKE", getlab());
|
|---|
| 664 | soumemb(p, c, 0);
|
|---|
| 665 | }
|
|---|
| 666 | ;
|
|---|
| 667 |
|
|---|
| 668 | /* always preceeded by attributes */
|
|---|
| 669 | xnfdeclarator: declarator attr_var {
|
|---|
| 670 | $$ = xnf = init_declarator($<nodep>0, $1, 1, $2);
|
|---|
| 671 | }
|
|---|
| 672 | | declarator C_ASM '(' string ')' {
|
|---|
| 673 | pragma_renamed = newstring($4, strlen($4));
|
|---|
| 674 | $$ = xnf = init_declarator($<nodep>0, $1, 1, NULL);
|
|---|
| 675 | }
|
|---|
| 676 | ;
|
|---|
| 677 |
|
|---|
| 678 | /*
|
|---|
| 679 | * Handles declarations and assignments.
|
|---|
| 680 | * Returns nothing.
|
|---|
| 681 | */
|
|---|
| 682 | init_declarator: declarator attr_var { init_declarator($<nodep>0, $1, 0, $2);}
|
|---|
| 683 | | declarator C_ASM '(' string ')' attr_var {
|
|---|
| 684 | #ifdef GCC_COMPAT
|
|---|
| 685 | pragma_renamed = newstring($4, strlen($4));
|
|---|
| 686 | init_declarator($<nodep>0, $1, 0, $6);
|
|---|
| 687 | #else
|
|---|
| 688 | werror("gcc extension");
|
|---|
| 689 | init_declarator($<nodep>0, $1, 0, $6);
|
|---|
| 690 | #endif
|
|---|
| 691 | }
|
|---|
| 692 | | xnfdeclarator '=' e { simpleinit($1, eve($3)); xnf = NULL; }
|
|---|
| 693 | | xnfdeclarator '=' begbr init_list optcomma '}' {
|
|---|
| 694 | endinit();
|
|---|
| 695 | xnf = NULL;
|
|---|
| 696 | }
|
|---|
| 697 | /*COMPAT_GCC*/ | xnfdeclarator '=' begbr '}' { endinit(); xnf = NULL; }
|
|---|
| 698 | | xnfdeclarator '=' addrlbl { simpleinit($1, $3); xnf = NULL; }
|
|---|
| 699 | ;
|
|---|
| 700 |
|
|---|
| 701 | begbr: '{' { beginit($<symp>-1); }
|
|---|
| 702 | ;
|
|---|
| 703 |
|
|---|
| 704 | initializer: e %prec ',' { $$ = eve($1); }
|
|---|
| 705 | | addrlbl { $$ = $1; }
|
|---|
| 706 | | ibrace init_list optcomma '}' { $$ = NULL; }
|
|---|
| 707 | | ibrace '}' { asginit(bcon(0)); $$ = NULL; }
|
|---|
| 708 | ;
|
|---|
| 709 |
|
|---|
| 710 | init_list: designation initializer { dainit($1, $2); }
|
|---|
| 711 | | init_list ',' designation initializer { dainit($3, $4); }
|
|---|
| 712 | ;
|
|---|
| 713 |
|
|---|
| 714 | designation: designator_list '=' { desinit($1); $$ = NIL; }
|
|---|
| 715 | | '[' e C_ELLIPSIS e ']' '=' { $$ = biop(CM, $2, $4); }
|
|---|
| 716 | | { $$ = NIL; }
|
|---|
| 717 | ;
|
|---|
| 718 |
|
|---|
| 719 | designator_list: designator { $$ = $1; }
|
|---|
| 720 | | designator_list designator { $$ = $2; $$->n_left = $1; }
|
|---|
| 721 | ;
|
|---|
| 722 |
|
|---|
| 723 | designator: '[' e ']' {
|
|---|
| 724 | int ie = con_e($2);
|
|---|
| 725 | if (ie < 0) {
|
|---|
| 726 | uerror("designator must be non-negative");
|
|---|
| 727 | ie = 0;
|
|---|
| 728 | }
|
|---|
| 729 | $$ = biop(LB, NIL, bcon(ie));
|
|---|
| 730 | }
|
|---|
| 731 | | C_STROP C_TYPENAME {
|
|---|
| 732 | if ($1 != DOT)
|
|---|
| 733 | uerror("invalid designator");
|
|---|
| 734 | $$ = bdty(NAME, $2);
|
|---|
| 735 | }
|
|---|
| 736 | | C_STROP C_NAME {
|
|---|
| 737 | if ($1 != DOT)
|
|---|
| 738 | uerror("invalid designator");
|
|---|
| 739 | $$ = bdty(NAME, $2);
|
|---|
| 740 | }
|
|---|
| 741 | ;
|
|---|
| 742 |
|
|---|
| 743 | optcomma : /* VOID */
|
|---|
| 744 | | ','
|
|---|
| 745 | ;
|
|---|
| 746 |
|
|---|
| 747 | ibrace: '{' { ilbrace(); }
|
|---|
| 748 | ;
|
|---|
| 749 |
|
|---|
| 750 | /* STATEMENTS */
|
|---|
| 751 |
|
|---|
| 752 | compoundstmt: begin block_item_list '}' { flend(); }
|
|---|
| 753 | | begin '}' { flend(); }
|
|---|
| 754 | ;
|
|---|
| 755 |
|
|---|
| 756 | begin: '{' {
|
|---|
| 757 | struct savbc *bc = tmpalloc(sizeof(struct savbc));
|
|---|
| 758 | if (blevel == 1) {
|
|---|
| 759 | #ifdef STABS
|
|---|
| 760 | if (gflag)
|
|---|
| 761 | stabs_line(lineno);
|
|---|
| 762 | #endif
|
|---|
| 763 | dclargs();
|
|---|
| 764 | }
|
|---|
| 765 | #ifdef STABS
|
|---|
| 766 | if (gflag && blevel > 1)
|
|---|
| 767 | stabs_lbrac(blevel+1);
|
|---|
| 768 | #endif
|
|---|
| 769 | ++blevel;
|
|---|
| 770 | oldstyle = 0;
|
|---|
| 771 | bc->contlab = autooff;
|
|---|
| 772 | bc->next = savctx;
|
|---|
| 773 | savctx = bc;
|
|---|
| 774 | bccode();
|
|---|
| 775 | if (!isinlining && sspflag && blevel == 2)
|
|---|
| 776 | sspstart();
|
|---|
| 777 | }
|
|---|
| 778 | ;
|
|---|
| 779 |
|
|---|
| 780 | statement: e ';' { ecomp(eve($1)); symclear(blevel); }
|
|---|
| 781 | | compoundstmt
|
|---|
| 782 | | ifprefix statement { plabel($1); reached = 1; }
|
|---|
| 783 | | ifelprefix statement {
|
|---|
| 784 | if ($1 != NOLAB) {
|
|---|
| 785 | plabel( $1);
|
|---|
| 786 | reached = 1;
|
|---|
| 787 | }
|
|---|
| 788 | }
|
|---|
| 789 | | whprefix statement {
|
|---|
| 790 | branch(contlab);
|
|---|
| 791 | plabel( brklab );
|
|---|
| 792 | if( (flostat&FBRK) || !(flostat&FLOOP))
|
|---|
| 793 | reached = 1;
|
|---|
| 794 | else
|
|---|
| 795 | reached = 0;
|
|---|
| 796 | resetbc(0);
|
|---|
| 797 | }
|
|---|
| 798 | | doprefix statement C_WHILE '(' e ')' ';' {
|
|---|
| 799 | plabel(contlab);
|
|---|
| 800 | if (flostat & FCONT)
|
|---|
| 801 | reached = 1;
|
|---|
| 802 | if (reached)
|
|---|
| 803 | cbranch(buildtree(NE, eve($5), bcon(0)),
|
|---|
| 804 | bcon($1));
|
|---|
| 805 | else
|
|---|
| 806 | tfree(eve($5));
|
|---|
| 807 | plabel( brklab);
|
|---|
| 808 | reached = 1;
|
|---|
| 809 | resetbc(0);
|
|---|
| 810 | }
|
|---|
| 811 | | forprefix .e ')' statement
|
|---|
| 812 | { plabel( contlab );
|
|---|
| 813 | if( flostat&FCONT ) reached = 1;
|
|---|
| 814 | if( $2 ) ecomp( $2 );
|
|---|
| 815 | branch($1);
|
|---|
| 816 | plabel( brklab );
|
|---|
| 817 | if( (flostat&FBRK) || !(flostat&FLOOP) ) reached = 1;
|
|---|
| 818 | else reached = 0;
|
|---|
| 819 | resetbc(0);
|
|---|
| 820 | }
|
|---|
| 821 | | switchpart statement
|
|---|
| 822 | { if( reached ) branch( brklab );
|
|---|
| 823 | plabel( $1 );
|
|---|
| 824 | swend();
|
|---|
| 825 | plabel( brklab);
|
|---|
| 826 | if( (flostat&FBRK) || !(flostat&FDEF) ) reached = 1;
|
|---|
| 827 | resetbc(FCONT);
|
|---|
| 828 | }
|
|---|
| 829 | | C_BREAK ';' {
|
|---|
| 830 | if (brklab == NOLAB)
|
|---|
| 831 | uerror("illegal break");
|
|---|
| 832 | else if (reached)
|
|---|
| 833 | branch(brklab);
|
|---|
| 834 | flostat |= FBRK;
|
|---|
| 835 | reached = 0;
|
|---|
| 836 | }
|
|---|
| 837 | | C_CONTINUE ';' {
|
|---|
| 838 | if (contlab == NOLAB)
|
|---|
| 839 | uerror("illegal continue");
|
|---|
| 840 | else
|
|---|
| 841 | branch(contlab);
|
|---|
| 842 | flostat |= FCONT;
|
|---|
| 843 | goto rch;
|
|---|
| 844 | }
|
|---|
| 845 | | C_RETURN ';' {
|
|---|
| 846 | branch(retlab);
|
|---|
| 847 | if (cftnsp->stype != VOID &&
|
|---|
| 848 | (cftnsp->sflags & NORETYP) == 0 &&
|
|---|
| 849 | cftnsp->stype != VOID+FTN)
|
|---|
| 850 | uerror("return value required");
|
|---|
| 851 | rch:
|
|---|
| 852 | if (!reached)
|
|---|
| 853 | warner(Wunreachable_code, NULL);
|
|---|
| 854 | reached = 0;
|
|---|
| 855 | }
|
|---|
| 856 | | C_RETURN e ';' {
|
|---|
| 857 | NODE *p, *q;
|
|---|
| 858 |
|
|---|
| 859 | p = nametree(cftnsp);
|
|---|
| 860 | p->n_type = DECREF(p->n_type);
|
|---|
| 861 | q = eve($2);
|
|---|
| 862 | #ifndef NO_COMPLEX
|
|---|
| 863 | if (ANYCX(q) || ANYCX(p))
|
|---|
| 864 | q = cxret(q, p);
|
|---|
| 865 | #endif
|
|---|
| 866 | p = buildtree(RETURN, p, q);
|
|---|
| 867 | if (p->n_type == VOID) {
|
|---|
| 868 | ecomp(p->n_right);
|
|---|
| 869 | } else {
|
|---|
| 870 | if (cftnod == NIL)
|
|---|
| 871 | cftnod = tempnode(0, p->n_type,
|
|---|
| 872 | p->n_df, p->n_ap);
|
|---|
| 873 | ecomp(buildtree(ASSIGN,
|
|---|
| 874 | ccopy(cftnod), p->n_right));
|
|---|
| 875 | }
|
|---|
| 876 | tfree(p->n_left);
|
|---|
| 877 | nfree(p);
|
|---|
| 878 | branch(retlab);
|
|---|
| 879 | reached = 0;
|
|---|
| 880 | }
|
|---|
| 881 | | C_GOTO C_NAME ';' { gotolabel($2); goto rch; }
|
|---|
| 882 | | C_GOTO '*' e ';' { ecomp(biop(GOTO, eve($3), NIL)); }
|
|---|
| 883 | | asmstatement ';'
|
|---|
| 884 | | ';'
|
|---|
| 885 | | error ';'
|
|---|
| 886 | | error '}'
|
|---|
| 887 | | label statement
|
|---|
| 888 | ;
|
|---|
| 889 |
|
|---|
| 890 | asmstatement: C_ASM mvol '(' string ')' { send_passt(IP_ASM, mkpstr($4)); }
|
|---|
| 891 | | C_ASM mvol '(' string xasm ')' { mkxasm($4, $5); }
|
|---|
| 892 | ;
|
|---|
| 893 |
|
|---|
| 894 | mvol: /* empty */
|
|---|
| 895 | | C_QUALIFIER { nfree($1); }
|
|---|
| 896 | ;
|
|---|
| 897 |
|
|---|
| 898 | xasm: ':' oplist { $$ = xcmop($2, NIL, NIL); }
|
|---|
| 899 | | ':' oplist ':' oplist { $$ = xcmop($2, $4, NIL); }
|
|---|
| 900 | | ':' oplist ':' oplist ':' cnstr { $$ = xcmop($2, $4, $6); }
|
|---|
| 901 | ;
|
|---|
| 902 |
|
|---|
| 903 | oplist: /* nothing */ { $$ = NIL; }
|
|---|
| 904 | | oper { $$ = $1; }
|
|---|
| 905 | ;
|
|---|
| 906 |
|
|---|
| 907 | oper: string '(' e ')' { $$ = xasmop($1, eve($3)); }
|
|---|
| 908 | | oper ',' string '(' e ')' {
|
|---|
| 909 | $$ = cmop($1, xasmop($3, eve($5)));
|
|---|
| 910 | }
|
|---|
| 911 | ;
|
|---|
| 912 |
|
|---|
| 913 | cnstr: string { $$ = xasmop($1, bcon(0)); }
|
|---|
| 914 | | cnstr ',' string { $$ = cmop($1, xasmop($3, bcon(0))); }
|
|---|
| 915 | ;
|
|---|
| 916 |
|
|---|
| 917 | label: C_NAME ':' attr_var { deflabel($1, $3); reached = 1; }
|
|---|
| 918 | | C_TYPENAME ':' attr_var { deflabel($1, $3); reached = 1; }
|
|---|
| 919 | | C_CASE e ':' { addcase(eve($2)); reached = 1; }
|
|---|
| 920 | /* COMPAT_GCC */| C_CASE e C_ELLIPSIS e ':' {
|
|---|
| 921 | #ifdef GCC_COMPAT
|
|---|
| 922 | gcccase(eve($2), eve($4)); reached = 1;
|
|---|
| 923 | #endif
|
|---|
| 924 | }
|
|---|
| 925 | | C_DEFAULT ':' { reached = 1; adddef(); flostat |= FDEF; }
|
|---|
| 926 | ;
|
|---|
| 927 |
|
|---|
| 928 | doprefix: C_DO {
|
|---|
| 929 | savebc();
|
|---|
| 930 | brklab = getlab();
|
|---|
| 931 | contlab = getlab();
|
|---|
| 932 | plabel( $$ = getlab());
|
|---|
| 933 | reached = 1;
|
|---|
| 934 | }
|
|---|
| 935 | ;
|
|---|
| 936 | ifprefix: C_IF '(' e ')' {
|
|---|
| 937 | cbranch(buildtree(NOT, eve($3), NIL), bcon($$ = getlab()));
|
|---|
| 938 | reached = 1;
|
|---|
| 939 | }
|
|---|
| 940 | ;
|
|---|
| 941 | ifelprefix: ifprefix statement C_ELSE {
|
|---|
| 942 | if (reached)
|
|---|
| 943 | branch($$ = getlab());
|
|---|
| 944 | else
|
|---|
| 945 | $$ = NOLAB;
|
|---|
| 946 | plabel( $1);
|
|---|
| 947 | reached = 1;
|
|---|
| 948 | }
|
|---|
| 949 | ;
|
|---|
| 950 |
|
|---|
| 951 | whprefix: C_WHILE '(' e ')' {
|
|---|
| 952 | savebc();
|
|---|
| 953 | $3 = eve($3);
|
|---|
| 954 | if ($3->n_op == ICON && $3->n_lval != 0)
|
|---|
| 955 | flostat = FLOOP;
|
|---|
| 956 | plabel( contlab = getlab());
|
|---|
| 957 | reached = 1;
|
|---|
| 958 | brklab = getlab();
|
|---|
| 959 | if (flostat == FLOOP)
|
|---|
| 960 | tfree($3);
|
|---|
| 961 | else
|
|---|
| 962 | cbranch(buildtree(NOT, $3, NIL), bcon(brklab));
|
|---|
| 963 | }
|
|---|
| 964 | ;
|
|---|
| 965 | forprefix: C_FOR '(' .e ';' .e ';' {
|
|---|
| 966 | if ($3)
|
|---|
| 967 | ecomp($3);
|
|---|
| 968 | savebc();
|
|---|
| 969 | contlab = getlab();
|
|---|
| 970 | brklab = getlab();
|
|---|
| 971 | plabel( $$ = getlab());
|
|---|
| 972 | reached = 1;
|
|---|
| 973 | if ($5)
|
|---|
| 974 | cbranch(buildtree(NOT, $5, NIL), bcon(brklab));
|
|---|
| 975 | else
|
|---|
| 976 | flostat |= FLOOP;
|
|---|
| 977 | }
|
|---|
| 978 | | C_FOR '(' { ++blevel; } declaration .e ';' {
|
|---|
| 979 | blevel--;
|
|---|
| 980 | savebc();
|
|---|
| 981 | contlab = getlab();
|
|---|
| 982 | brklab = getlab();
|
|---|
| 983 | plabel( $$ = getlab());
|
|---|
| 984 | reached = 1;
|
|---|
| 985 | if ($5)
|
|---|
| 986 | cbranch(buildtree(NOT, $5, NIL), bcon(brklab));
|
|---|
| 987 | else
|
|---|
| 988 | flostat |= FLOOP;
|
|---|
| 989 | }
|
|---|
| 990 | ;
|
|---|
| 991 |
|
|---|
| 992 | switchpart: C_SWITCH '(' e ')' {
|
|---|
| 993 | NODE *p;
|
|---|
| 994 | int num;
|
|---|
| 995 | TWORD t;
|
|---|
| 996 |
|
|---|
| 997 | savebc();
|
|---|
| 998 | brklab = getlab();
|
|---|
| 999 | $3 = eve($3);
|
|---|
| 1000 | if (($3->n_type != BOOL && $3->n_type > ULONGLONG) ||
|
|---|
| 1001 | $3->n_type < CHAR) {
|
|---|
| 1002 | uerror("switch expression must have integer "
|
|---|
| 1003 | "type");
|
|---|
| 1004 | t = INT;
|
|---|
| 1005 | } else {
|
|---|
| 1006 | $3 = intprom($3);
|
|---|
| 1007 | t = $3->n_type;
|
|---|
| 1008 | }
|
|---|
| 1009 | p = tempnode(0, t, 0, MKAP(t));
|
|---|
| 1010 | num = regno(p);
|
|---|
| 1011 | ecomp(buildtree(ASSIGN, p, $3));
|
|---|
| 1012 | branch( $$ = getlab());
|
|---|
| 1013 | swstart(num, t);
|
|---|
| 1014 | reached = 0;
|
|---|
| 1015 | }
|
|---|
| 1016 | ;
|
|---|
| 1017 | /* EXPRESSIONS */
|
|---|
| 1018 | .e: e { $$ = eve($1); }
|
|---|
| 1019 | | { $$=0; }
|
|---|
| 1020 | ;
|
|---|
| 1021 |
|
|---|
| 1022 | elist: { $$ = NIL; }
|
|---|
| 1023 | | e %prec ','
|
|---|
| 1024 | | elist ',' e { $$ = biop(CM, $1, $3); }
|
|---|
| 1025 | | elist ',' cast_type { /* hack for stdarg */
|
|---|
| 1026 | TYMFIX($3);
|
|---|
| 1027 | $3->n_op = TYPE;
|
|---|
| 1028 | $$ = biop(CM, $1, $3);
|
|---|
| 1029 | }
|
|---|
| 1030 | ;
|
|---|
| 1031 |
|
|---|
| 1032 | /*
|
|---|
| 1033 | * Precedence order of operators.
|
|---|
| 1034 | */
|
|---|
| 1035 | e: e ',' e { $$ = biop(COMOP, $1, $3); }
|
|---|
| 1036 | | e '=' e { $$ = biop(ASSIGN, $1, $3); }
|
|---|
| 1037 | | e C_ASOP e { $$ = biop($2, $1, $3); }
|
|---|
| 1038 | | e '?' e ':' e {
|
|---|
| 1039 | $$=biop(QUEST, $1, biop(COLON, $3, $5));
|
|---|
| 1040 | }
|
|---|
| 1041 | | e '?' ':' e {
|
|---|
| 1042 | NODE *p = tempnode(0, $1->n_type, $1->n_df, $1->n_ap);
|
|---|
| 1043 | $$ = biop(COLON, ccopy(p), $4);
|
|---|
| 1044 | $$=biop(QUEST, biop(ASSIGN, p, $1), $$);
|
|---|
| 1045 | }
|
|---|
| 1046 | | e C_OROR e { $$ = biop($2, $1, $3); }
|
|---|
| 1047 | | e C_ANDAND e { $$ = biop($2, $1, $3); }
|
|---|
| 1048 | | e '|' e { $$ = biop(OR, $1, $3); }
|
|---|
| 1049 | | e '^' e { $$ = biop(ER, $1, $3); }
|
|---|
| 1050 | | e '&' e { $$ = biop(AND, $1, $3); }
|
|---|
| 1051 | | e C_EQUOP e { $$ = biop($2, $1, $3); }
|
|---|
| 1052 | | e C_RELOP e { $$ = biop($2, $1, $3); }
|
|---|
| 1053 | | e C_SHIFTOP e { $$ = biop($2, $1, $3); }
|
|---|
| 1054 | | e '+' e { $$ = biop(PLUS, $1, $3); }
|
|---|
| 1055 | | e '-' e { $$ = biop(MINUS, $1, $3); }
|
|---|
| 1056 | | e C_DIVOP e { $$ = biop($2, $1, $3); }
|
|---|
| 1057 | | e '*' e { $$ = biop(MUL, $1, $3); }
|
|---|
| 1058 | | e '=' addrlbl { $$ = biop(ASSIGN, $1, $3); }
|
|---|
| 1059 | | term
|
|---|
| 1060 | ;
|
|---|
| 1061 |
|
|---|
| 1062 | xbegin: begin {
|
|---|
| 1063 | $$ = getlab(); getlab(); getlab();
|
|---|
| 1064 | branch($$); plabel(($$)+1); }
|
|---|
| 1065 | ;
|
|---|
| 1066 |
|
|---|
| 1067 | addrlbl: C_ANDAND C_NAME {
|
|---|
| 1068 | #ifdef GCC_COMPAT
|
|---|
| 1069 | struct symtab *s = lookup($2, SLBLNAME);
|
|---|
| 1070 | if (s->soffset == 0)
|
|---|
| 1071 | s->soffset = -getlab();
|
|---|
| 1072 | $$ = buildtree(ADDROF, nametree(s), NIL);
|
|---|
| 1073 | #else
|
|---|
| 1074 | uerror("gcc extension");
|
|---|
| 1075 | #endif
|
|---|
| 1076 | }
|
|---|
| 1077 | ;
|
|---|
| 1078 |
|
|---|
| 1079 | term: term C_INCOP { $$ = biop($2, $1, bcon(1)); }
|
|---|
| 1080 | | '*' term { $$ = biop(UMUL, $2, NIL); }
|
|---|
| 1081 | | '&' term { $$ = biop(ADDROF, $2, NIL); }
|
|---|
| 1082 | | '-' term { $$ = biop(UMINUS, $2, NIL ); }
|
|---|
| 1083 | | '+' term { $$ = biop(PLUS, $2, bcon(0)); }
|
|---|
| 1084 | | C_UNOP term { $$ = biop($1, $2, NIL); }
|
|---|
| 1085 | | C_INCOP term {
|
|---|
| 1086 | $$ = biop($1 == INCR ? PLUSEQ : MINUSEQ, $2, bcon(1));
|
|---|
| 1087 | }
|
|---|
| 1088 | | C_SIZEOF xa term { $$ = biop(SZOF, $3, bcon(0)); inattr = $<intval>2; }
|
|---|
| 1089 | | '(' cast_type ')' term %prec C_INCOP {
|
|---|
| 1090 | TYMFIX($2);
|
|---|
| 1091 | $$ = biop(CAST, $2, $4);
|
|---|
| 1092 | }
|
|---|
| 1093 | | C_SIZEOF xa '(' cast_type ')' %prec C_SIZEOF {
|
|---|
| 1094 | $$ = biop(SZOF, $4, bcon(1));
|
|---|
| 1095 | inattr = $<intval>2;
|
|---|
| 1096 | }
|
|---|
| 1097 | | C_ALIGNOF xa '(' cast_type ')' {
|
|---|
| 1098 | TYMFIX($4);
|
|---|
| 1099 | int al = talign($4->n_type, $4->n_ap);
|
|---|
| 1100 | $$ = bcon(al/SZCHAR);
|
|---|
| 1101 | inattr = $<intval>2;
|
|---|
| 1102 | tfree($4);
|
|---|
| 1103 | }
|
|---|
| 1104 | | '(' cast_type ')' clbrace init_list optcomma '}' {
|
|---|
| 1105 | endinit();
|
|---|
| 1106 | $$ = bdty(NAME, $4);
|
|---|
| 1107 | $$->n_op = CLOP;
|
|---|
| 1108 | }
|
|---|
| 1109 | | '(' cast_type ')' clbrace '}' {
|
|---|
| 1110 | endinit();
|
|---|
| 1111 | $$ = bdty(NAME, $4);
|
|---|
| 1112 | $$->n_op = CLOP;
|
|---|
| 1113 | }
|
|---|
| 1114 | | term '[' e ']' { $$ = biop(LB, $1, $3); }
|
|---|
| 1115 | | C_NAME '(' elist ')' {
|
|---|
| 1116 | $$ = biop($3 ? CALL : UCALL, bdty(NAME, $1), $3);
|
|---|
| 1117 | }
|
|---|
| 1118 | | term '(' elist ')' { $$ = biop($3 ? CALL : UCALL, $1, $3); }
|
|---|
| 1119 | | term C_STROP C_NAME { $$ = biop($2, $1, bdty(NAME, $3)); }
|
|---|
| 1120 | | term C_STROP C_TYPENAME { $$ = biop($2, $1, bdty(NAME, $3));}
|
|---|
| 1121 | | C_NAME %prec C_SIZEOF /* below ( */{ $$ = bdty(NAME, $1); }
|
|---|
| 1122 | | PCC_OFFSETOF '(' cast_type ',' term ')' {
|
|---|
| 1123 | TYMFIX($3);
|
|---|
| 1124 | $3->n_type = INCREF($3->n_type);
|
|---|
| 1125 | $3 = biop(CAST, $3, bcon(0));
|
|---|
| 1126 | if ($5->n_op == NAME) {
|
|---|
| 1127 | $$ = biop(STREF, $3, $5);
|
|---|
| 1128 | } else {
|
|---|
| 1129 | NODE *p = $5;
|
|---|
| 1130 | while (p->n_left->n_op != NAME)
|
|---|
| 1131 | p = p->n_left;
|
|---|
| 1132 | p->n_left = biop(STREF, $3, p->n_left);
|
|---|
| 1133 | $$ = $5;
|
|---|
| 1134 | }
|
|---|
| 1135 | $$ = biop(ADDROF, $$, NIL);
|
|---|
| 1136 | $3 = block(NAME, NIL, NIL, ENUNSIGN(INTPTR), 0,
|
|---|
| 1137 | MKAP(ENUNSIGN(INTPTR)));
|
|---|
| 1138 | $$ = biop(CAST, $3, $$);
|
|---|
| 1139 | }
|
|---|
| 1140 | | C_ICON { $$ = $1; }
|
|---|
| 1141 | | C_FCON { $$ = $1; }
|
|---|
| 1142 | | string { $$ = bdty(STRING, $1, widestr); }
|
|---|
| 1143 | | '(' e ')' { $$=$2; }
|
|---|
| 1144 | | '(' xbegin block_item_list e ';' '}' ')' {
|
|---|
| 1145 | /* XXX - check recursive ({ }) statements */
|
|---|
| 1146 | branch(($2)+2);
|
|---|
| 1147 | plabel($2);
|
|---|
| 1148 | $$ = buildtree(COMOP,
|
|---|
| 1149 | biop(GOTO, bcon(($2)+1), NIL), eve($4));
|
|---|
| 1150 | flend();
|
|---|
| 1151 | }
|
|---|
| 1152 | | '(' xbegin block_item_list '}' ')' {
|
|---|
| 1153 | /* XXX - check recursive ({ }) statements */
|
|---|
| 1154 | branch(($2)+2);
|
|---|
| 1155 | plabel($2);
|
|---|
| 1156 | $$ = buildtree(COMOP,
|
|---|
| 1157 | biop(GOTO, bcon(($2)+1), NIL), voidcon());
|
|---|
| 1158 | flend();
|
|---|
| 1159 | }
|
|---|
| 1160 | ;
|
|---|
| 1161 |
|
|---|
| 1162 | xa: { $<intval>$ = inattr; inattr = 0; }
|
|---|
| 1163 | ;
|
|---|
| 1164 |
|
|---|
| 1165 | clbrace: '{' { NODE *q = $<nodep>-1; TYMFIX(q); $$ = clbrace(q); }
|
|---|
| 1166 | ;
|
|---|
| 1167 |
|
|---|
| 1168 | string: C_STRING { widestr = 0; $$ = stradd("", $1); }
|
|---|
| 1169 | | string C_STRING { $$ = stradd($1, $2); }
|
|---|
| 1170 | ;
|
|---|
| 1171 |
|
|---|
| 1172 | cast_type: specifier_qualifier_list {
|
|---|
| 1173 | $$ = biop(TYMERGE, $1, bdty(NAME, NULL));
|
|---|
| 1174 | }
|
|---|
| 1175 | | specifier_qualifier_list abstract_declarator {
|
|---|
| 1176 | $$ = biop(TYMERGE, $1, aryfix($2));
|
|---|
| 1177 | }
|
|---|
| 1178 | ;
|
|---|
| 1179 |
|
|---|
| 1180 | %%
|
|---|
| 1181 |
|
|---|
| 1182 | NODE *
|
|---|
| 1183 | mkty(TWORD t, union dimfun *d, struct attr *sue)
|
|---|
| 1184 | {
|
|---|
| 1185 | return block(TYPE, NIL, NIL, t, d, sue);
|
|---|
| 1186 | }
|
|---|
| 1187 |
|
|---|
| 1188 | NODE *
|
|---|
| 1189 | bdty(int op, ...)
|
|---|
| 1190 | {
|
|---|
| 1191 | va_list ap;
|
|---|
| 1192 | int val;
|
|---|
| 1193 | register NODE *q;
|
|---|
| 1194 |
|
|---|
| 1195 | va_start(ap, op);
|
|---|
| 1196 | q = biop(op, NIL, NIL);
|
|---|
| 1197 |
|
|---|
| 1198 | switch (op) {
|
|---|
| 1199 | case UMUL:
|
|---|
| 1200 | case UCALL:
|
|---|
| 1201 | q->n_left = va_arg(ap, NODE *);
|
|---|
| 1202 | q->n_rval = 0;
|
|---|
| 1203 | break;
|
|---|
| 1204 |
|
|---|
| 1205 | case CALL:
|
|---|
| 1206 | q->n_left = va_arg(ap, NODE *);
|
|---|
| 1207 | q->n_right = va_arg(ap, NODE *);
|
|---|
| 1208 | break;
|
|---|
| 1209 |
|
|---|
| 1210 | case LB:
|
|---|
| 1211 | q->n_left = va_arg(ap, NODE *);
|
|---|
| 1212 | if ((val = va_arg(ap, int)) <= 0) {
|
|---|
| 1213 | uerror("array size must be positive");
|
|---|
| 1214 | val = 1;
|
|---|
| 1215 | }
|
|---|
| 1216 | q->n_right = bcon(val);
|
|---|
| 1217 | break;
|
|---|
| 1218 |
|
|---|
| 1219 | case NAME:
|
|---|
| 1220 | q->n_sp = va_arg(ap, struct symtab *); /* XXX survive tymerge */
|
|---|
| 1221 | break;
|
|---|
| 1222 |
|
|---|
| 1223 | case STRING:
|
|---|
| 1224 | q->n_name = va_arg(ap, char *);
|
|---|
| 1225 | q->n_lval = va_arg(ap, int);
|
|---|
| 1226 | break;
|
|---|
| 1227 |
|
|---|
| 1228 | default:
|
|---|
| 1229 | cerror("bad bdty");
|
|---|
| 1230 | }
|
|---|
| 1231 | va_end(ap);
|
|---|
| 1232 |
|
|---|
| 1233 | return q;
|
|---|
| 1234 | }
|
|---|
| 1235 |
|
|---|
| 1236 | static void
|
|---|
| 1237 | flend(void)
|
|---|
| 1238 | {
|
|---|
| 1239 | if (!isinlining && sspflag && blevel == 2)
|
|---|
| 1240 | sspend();
|
|---|
| 1241 | #ifdef STABS
|
|---|
| 1242 | if (gflag && blevel > 2)
|
|---|
| 1243 | stabs_rbrac(blevel);
|
|---|
| 1244 | #endif
|
|---|
| 1245 | --blevel;
|
|---|
| 1246 | if( blevel == 1 )
|
|---|
| 1247 | blevel = 0;
|
|---|
| 1248 | symclear(blevel); /* Clean ut the symbol table */
|
|---|
| 1249 | if (autooff > maxautooff)
|
|---|
| 1250 | maxautooff = autooff;
|
|---|
| 1251 | autooff = savctx->contlab;
|
|---|
| 1252 | savctx = savctx->next;
|
|---|
| 1253 | }
|
|---|
| 1254 |
|
|---|
| 1255 | static void
|
|---|
| 1256 | savebc(void)
|
|---|
| 1257 | {
|
|---|
| 1258 | struct savbc *bc = tmpalloc(sizeof(struct savbc));
|
|---|
| 1259 |
|
|---|
| 1260 | bc->brklab = brklab;
|
|---|
| 1261 | bc->contlab = contlab;
|
|---|
| 1262 | bc->flostat = flostat;
|
|---|
| 1263 | bc->next = savbc;
|
|---|
| 1264 | savbc = bc;
|
|---|
| 1265 | flostat = 0;
|
|---|
| 1266 | }
|
|---|
| 1267 |
|
|---|
| 1268 | static void
|
|---|
| 1269 | resetbc(int mask)
|
|---|
| 1270 | {
|
|---|
| 1271 | flostat = savbc->flostat | (flostat&mask);
|
|---|
| 1272 | contlab = savbc->contlab;
|
|---|
| 1273 | brklab = savbc->brklab;
|
|---|
| 1274 | savbc = savbc->next;
|
|---|
| 1275 | }
|
|---|
| 1276 |
|
|---|
| 1277 | struct swdef {
|
|---|
| 1278 | struct swdef *next; /* Next in list */
|
|---|
| 1279 | int deflbl; /* Label for "default" */
|
|---|
| 1280 | struct swents *ents; /* Linked sorted list of case entries */
|
|---|
| 1281 | int nents; /* # of entries in list */
|
|---|
| 1282 | int num; /* Node value will end up in */
|
|---|
| 1283 | TWORD type; /* Type of switch expression */
|
|---|
| 1284 | } *swpole;
|
|---|
| 1285 |
|
|---|
| 1286 | /*
|
|---|
| 1287 | * add case to switch
|
|---|
| 1288 | */
|
|---|
| 1289 | static void
|
|---|
| 1290 | addcase(NODE *p)
|
|---|
| 1291 | {
|
|---|
| 1292 | struct swents **put, *w, *sw = tmpalloc(sizeof(struct swents));
|
|---|
| 1293 | CONSZ val;
|
|---|
| 1294 |
|
|---|
| 1295 | p = optim(p); /* change enum to ints */
|
|---|
| 1296 | if (p->n_op != ICON || p->n_sp != NULL) {
|
|---|
| 1297 | uerror( "non-constant case expression");
|
|---|
| 1298 | return;
|
|---|
| 1299 | }
|
|---|
| 1300 | if (swpole == NULL) {
|
|---|
| 1301 | uerror("case not in switch");
|
|---|
| 1302 | return;
|
|---|
| 1303 | }
|
|---|
| 1304 |
|
|---|
| 1305 | if (DEUNSIGN(swpole->type) != DEUNSIGN(p->n_type)) {
|
|---|
| 1306 | val = p->n_lval;
|
|---|
| 1307 | p = makety(p, swpole->type, 0, 0, MKAP(swpole->type));
|
|---|
| 1308 | if (p->n_op != ICON)
|
|---|
| 1309 | cerror("could not cast case value to type of switch "
|
|---|
| 1310 | "expression");
|
|---|
| 1311 | if (p->n_lval != val)
|
|---|
| 1312 | werror("case expression truncated");
|
|---|
| 1313 | }
|
|---|
| 1314 | sw->sval = p->n_lval;
|
|---|
| 1315 | tfree(p);
|
|---|
| 1316 | put = &swpole->ents;
|
|---|
| 1317 | if (ISUNSIGNED(swpole->type)) {
|
|---|
| 1318 | for (w = swpole->ents;
|
|---|
| 1319 | w != NULL && (U_CONSZ)w->sval < (U_CONSZ)sw->sval;
|
|---|
| 1320 | w = w->next)
|
|---|
| 1321 | put = &w->next;
|
|---|
| 1322 | } else {
|
|---|
| 1323 | for (w = swpole->ents; w != NULL && w->sval < sw->sval;
|
|---|
| 1324 | w = w->next)
|
|---|
| 1325 | put = &w->next;
|
|---|
| 1326 | }
|
|---|
| 1327 | if (w != NULL && w->sval == sw->sval) {
|
|---|
| 1328 | uerror("duplicate case in switch");
|
|---|
| 1329 | return;
|
|---|
| 1330 | }
|
|---|
| 1331 | plabel(sw->slab = getlab());
|
|---|
| 1332 | *put = sw;
|
|---|
| 1333 | sw->next = w;
|
|---|
| 1334 | swpole->nents++;
|
|---|
| 1335 | }
|
|---|
| 1336 |
|
|---|
| 1337 | #ifdef GCC_COMPAT
|
|---|
| 1338 | void
|
|---|
| 1339 | gcccase(NODE *ln, NODE *hn)
|
|---|
| 1340 | {
|
|---|
| 1341 | CONSZ i, l, h;
|
|---|
| 1342 |
|
|---|
| 1343 | l = icons(optim(ln));
|
|---|
| 1344 | h = icons(optim(hn));
|
|---|
| 1345 |
|
|---|
| 1346 | if (h < l)
|
|---|
| 1347 | i = l, l = h, h = i;
|
|---|
| 1348 |
|
|---|
| 1349 | for (i = l; i <= h; i++)
|
|---|
| 1350 | addcase(xbcon(i, NULL, hn->n_type));
|
|---|
| 1351 | }
|
|---|
| 1352 | #endif
|
|---|
| 1353 |
|
|---|
| 1354 | /*
|
|---|
| 1355 | * add default case to switch
|
|---|
| 1356 | */
|
|---|
| 1357 | static void
|
|---|
| 1358 | adddef(void)
|
|---|
| 1359 | {
|
|---|
| 1360 | if (swpole == NULL)
|
|---|
| 1361 | uerror("default not inside switch");
|
|---|
| 1362 | else if (swpole->deflbl != 0)
|
|---|
| 1363 | uerror("duplicate default in switch");
|
|---|
| 1364 | else
|
|---|
| 1365 | plabel( swpole->deflbl = getlab());
|
|---|
| 1366 | }
|
|---|
| 1367 |
|
|---|
| 1368 | static void
|
|---|
| 1369 | swstart(int num, TWORD type)
|
|---|
| 1370 | {
|
|---|
| 1371 | struct swdef *sw = tmpalloc(sizeof(struct swdef));
|
|---|
| 1372 |
|
|---|
| 1373 | sw->deflbl = sw->nents = 0;
|
|---|
| 1374 | sw->ents = NULL;
|
|---|
| 1375 | sw->next = swpole;
|
|---|
| 1376 | sw->num = num;
|
|---|
| 1377 | sw->type = type;
|
|---|
| 1378 | swpole = sw;
|
|---|
| 1379 | }
|
|---|
| 1380 |
|
|---|
| 1381 | /*
|
|---|
| 1382 | * end a switch block
|
|---|
| 1383 | */
|
|---|
| 1384 | static void
|
|---|
| 1385 | swend(void)
|
|---|
| 1386 | {
|
|---|
| 1387 | struct swents *sw, **swp;
|
|---|
| 1388 | int i;
|
|---|
| 1389 |
|
|---|
| 1390 | sw = tmpalloc(sizeof(struct swents));
|
|---|
| 1391 | swp = tmpalloc(sizeof(struct swents *) * (swpole->nents+1));
|
|---|
| 1392 |
|
|---|
| 1393 | sw->slab = swpole->deflbl;
|
|---|
| 1394 | swp[0] = sw;
|
|---|
| 1395 |
|
|---|
| 1396 | for (i = 1; i <= swpole->nents; i++) {
|
|---|
| 1397 | swp[i] = swpole->ents;
|
|---|
| 1398 | swpole->ents = swpole->ents->next;
|
|---|
| 1399 | }
|
|---|
| 1400 | genswitch(swpole->num, swpole->type, swp, swpole->nents);
|
|---|
| 1401 |
|
|---|
| 1402 | swpole = swpole->next;
|
|---|
| 1403 | }
|
|---|
| 1404 |
|
|---|
| 1405 | /*
|
|---|
| 1406 | * num: tempnode the value of the switch expression is in
|
|---|
| 1407 | * type: type of the switch expression
|
|---|
| 1408 | *
|
|---|
| 1409 | * p points to an array of structures, each consisting
|
|---|
| 1410 | * of a constant value and a label.
|
|---|
| 1411 | * The first is >=0 if there is a default label;
|
|---|
| 1412 | * its value is the label number
|
|---|
| 1413 | * The entries p[1] to p[n] are the nontrivial cases
|
|---|
| 1414 | * n is the number of case statements (length of list)
|
|---|
| 1415 | */
|
|---|
| 1416 | static void
|
|---|
| 1417 | genswitch(int num, TWORD type, struct swents **p, int n)
|
|---|
| 1418 | {
|
|---|
| 1419 | NODE *r, *q;
|
|---|
| 1420 | int i;
|
|---|
| 1421 |
|
|---|
| 1422 | if (mygenswitch(num, type, p, n))
|
|---|
| 1423 | return;
|
|---|
| 1424 |
|
|---|
| 1425 | /* simple switch code */
|
|---|
| 1426 | for (i = 1; i <= n; ++i) {
|
|---|
| 1427 | /* already in 1 */
|
|---|
| 1428 | r = tempnode(num, type, 0, MKAP(type));
|
|---|
| 1429 | q = xbcon(p[i]->sval, NULL, type);
|
|---|
| 1430 | r = buildtree(NE, r, clocal(q));
|
|---|
| 1431 | cbranch(buildtree(NOT, r, NIL), bcon(p[i]->slab));
|
|---|
| 1432 | }
|
|---|
| 1433 | if (p[0]->slab > 0)
|
|---|
| 1434 | branch(p[0]->slab);
|
|---|
| 1435 | }
|
|---|
| 1436 |
|
|---|
| 1437 | /*
|
|---|
| 1438 | * Declare a variable or prototype.
|
|---|
| 1439 | */
|
|---|
| 1440 | static struct symtab *
|
|---|
| 1441 | init_declarator(NODE *tn, NODE *p, int assign, NODE *a)
|
|---|
| 1442 | {
|
|---|
| 1443 | int class = tn->n_lval;
|
|---|
| 1444 | struct symtab *sp;
|
|---|
| 1445 |
|
|---|
| 1446 | p = aryfix(p);
|
|---|
| 1447 | p = tymerge(tn, p);
|
|---|
| 1448 | if (a) {
|
|---|
| 1449 | struct attr *ap = gcc_attr_parse(a);
|
|---|
| 1450 | p->n_ap = attr_add(p->n_ap, ap);
|
|---|
| 1451 | }
|
|---|
| 1452 |
|
|---|
| 1453 | p->n_sp = sp = lookup((char *)p->n_sp, 0); /* XXX */
|
|---|
| 1454 |
|
|---|
| 1455 | if (fun_inline && ISFTN(p->n_type))
|
|---|
| 1456 | sp->sflags |= SINLINE;
|
|---|
| 1457 |
|
|---|
| 1458 | if (ISFTN(p->n_type) == 0) {
|
|---|
| 1459 | if (assign) {
|
|---|
| 1460 | defid(p, class);
|
|---|
| 1461 | sp = p->n_sp;
|
|---|
| 1462 | sp->sflags |= SASG;
|
|---|
| 1463 | if (sp->sflags & SDYNARRAY)
|
|---|
| 1464 | uerror("can't initialize dynamic arrays");
|
|---|
| 1465 | lcommdel(sp);
|
|---|
| 1466 | } else
|
|---|
| 1467 | nidcl(p, class);
|
|---|
| 1468 | } else {
|
|---|
| 1469 | extern NODE *parlink;
|
|---|
| 1470 | if (assign)
|
|---|
| 1471 | uerror("cannot initialise function");
|
|---|
| 1472 | defid(p, uclass(class));
|
|---|
| 1473 | sp = p->n_sp;
|
|---|
| 1474 | if (parlink) {
|
|---|
| 1475 | /* dynamic sized arrays in prototypes */
|
|---|
| 1476 | tfree(parlink); /* Free delayed tree */
|
|---|
| 1477 | parlink = NIL;
|
|---|
| 1478 | }
|
|---|
| 1479 | }
|
|---|
| 1480 | tfree(p);
|
|---|
| 1481 | return sp;
|
|---|
| 1482 | }
|
|---|
| 1483 |
|
|---|
| 1484 | /*
|
|---|
| 1485 | * Declare old-stype function arguments.
|
|---|
| 1486 | */
|
|---|
| 1487 | static void
|
|---|
| 1488 | oldargs(NODE *p)
|
|---|
| 1489 | {
|
|---|
| 1490 | blevel++;
|
|---|
| 1491 | p->n_op = TYPE;
|
|---|
| 1492 | p->n_type = FARG;
|
|---|
| 1493 | p->n_sp = lookup((char *)p->n_sp, 0);/* XXX */
|
|---|
| 1494 | defid(p, PARAM);
|
|---|
| 1495 | blevel--;
|
|---|
| 1496 | }
|
|---|
| 1497 |
|
|---|
| 1498 | /*
|
|---|
| 1499 | * Set NAME nodes to a null name and index of LB nodes to NOOFFSET
|
|---|
| 1500 | * unless clr is one, in that case preserve variable name.
|
|---|
| 1501 | */
|
|---|
| 1502 | static NODE *
|
|---|
| 1503 | namekill(NODE *p, int clr)
|
|---|
| 1504 | {
|
|---|
| 1505 | NODE *q;
|
|---|
| 1506 | int o = p->n_op;
|
|---|
| 1507 |
|
|---|
| 1508 | switch (coptype(o)) {
|
|---|
| 1509 | case LTYPE:
|
|---|
| 1510 | if (o == NAME) {
|
|---|
| 1511 | if (clr)
|
|---|
| 1512 | p->n_sp = NULL;
|
|---|
| 1513 | else
|
|---|
| 1514 | p->n_sp = lookup((char *)p->n_sp, 0);/* XXX */
|
|---|
| 1515 | }
|
|---|
| 1516 | break;
|
|---|
| 1517 |
|
|---|
| 1518 | case UTYPE:
|
|---|
| 1519 | p->n_left = namekill(p->n_left, clr);
|
|---|
| 1520 | break;
|
|---|
| 1521 |
|
|---|
| 1522 | case BITYPE:
|
|---|
| 1523 | p->n_left = namekill(p->n_left, clr);
|
|---|
| 1524 | if (o == LB) {
|
|---|
| 1525 | if (clr) {
|
|---|
| 1526 | tfree(p->n_right);
|
|---|
| 1527 | p->n_right = bcon(NOOFFSET);
|
|---|
| 1528 | } else
|
|---|
| 1529 | p->n_right = eve(p->n_right);
|
|---|
| 1530 | } else if (o == CALL)
|
|---|
| 1531 | p->n_right = namekill(p->n_right, 1);
|
|---|
| 1532 | else
|
|---|
| 1533 | p->n_right = namekill(p->n_right, clr);
|
|---|
| 1534 | if (o == TYMERGE) {
|
|---|
| 1535 | q = tymerge(p->n_left, p->n_right);
|
|---|
| 1536 | q->n_ap = attr_add(q->n_ap, p->n_ap);
|
|---|
| 1537 | tfree(p->n_left);
|
|---|
| 1538 | nfree(p);
|
|---|
| 1539 | p = q;
|
|---|
| 1540 | }
|
|---|
| 1541 | break;
|
|---|
| 1542 | }
|
|---|
| 1543 | return p;
|
|---|
| 1544 | }
|
|---|
| 1545 |
|
|---|
| 1546 | /*
|
|---|
| 1547 | * Declare function arguments.
|
|---|
| 1548 | */
|
|---|
| 1549 | static NODE *
|
|---|
| 1550 | funargs(NODE *p)
|
|---|
| 1551 | {
|
|---|
| 1552 | extern NODE *arrstk[10];
|
|---|
| 1553 |
|
|---|
| 1554 | if (p->n_op == ELLIPSIS)
|
|---|
| 1555 | return p;
|
|---|
| 1556 |
|
|---|
| 1557 | p = namekill(p, 0);
|
|---|
| 1558 | if (ISFTN(p->n_type))
|
|---|
| 1559 | p->n_type = INCREF(p->n_type);
|
|---|
| 1560 | if (ISARY(p->n_type)) {
|
|---|
| 1561 | p->n_type += (PTR-ARY);
|
|---|
| 1562 | if (p->n_df->ddim == -1)
|
|---|
| 1563 | tfree(arrstk[0]), arrstk[0] = NIL;
|
|---|
| 1564 | p->n_df++;
|
|---|
| 1565 | }
|
|---|
| 1566 | if (p->n_type == VOID && p->n_sp->sname == NULL)
|
|---|
| 1567 | return p; /* sanitycheck later */
|
|---|
| 1568 | else if (p->n_sp->sname == NULL)
|
|---|
| 1569 | uerror("argument missing");
|
|---|
| 1570 | else
|
|---|
| 1571 | defid(p, PARAM);
|
|---|
| 1572 | return p;
|
|---|
| 1573 | }
|
|---|
| 1574 |
|
|---|
| 1575 | static NODE *
|
|---|
| 1576 | listfw(NODE *p, NODE * (*f)(NODE *))
|
|---|
| 1577 | {
|
|---|
| 1578 | if (p->n_op == CM) {
|
|---|
| 1579 | p->n_left = listfw(p->n_left, f);
|
|---|
| 1580 | p->n_right = (*f)(p->n_right);
|
|---|
| 1581 | } else
|
|---|
| 1582 | p = (*f)(p);
|
|---|
| 1583 | return p;
|
|---|
| 1584 | }
|
|---|
| 1585 |
|
|---|
| 1586 |
|
|---|
| 1587 | /*
|
|---|
| 1588 | * Declare a function.
|
|---|
| 1589 | */
|
|---|
| 1590 | static void
|
|---|
| 1591 | fundef(NODE *tp, NODE *p)
|
|---|
| 1592 | {
|
|---|
| 1593 | extern int prolab;
|
|---|
| 1594 | struct symtab *s;
|
|---|
| 1595 | NODE *q, *typ;
|
|---|
| 1596 | int class = tp->n_lval, oclass, ctval;
|
|---|
| 1597 | char *c;
|
|---|
| 1598 |
|
|---|
| 1599 | /*
|
|---|
| 1600 | * We discard all names except for those needed for
|
|---|
| 1601 | * parameter declaration. While doing that, also change
|
|---|
| 1602 | * non-constant array sizes to unknown.
|
|---|
| 1603 | */
|
|---|
| 1604 | ctval = tvaloff;
|
|---|
| 1605 | for (q = p; coptype(q->n_op) != LTYPE &&
|
|---|
| 1606 | q->n_left->n_op != NAME; q = q->n_left) {
|
|---|
| 1607 | if (q->n_op == CALL)
|
|---|
| 1608 | q->n_right = namekill(q->n_right, 1);
|
|---|
| 1609 | }
|
|---|
| 1610 | if (q->n_op != CALL && q->n_op != UCALL) {
|
|---|
| 1611 | uerror("invalid function definition");
|
|---|
| 1612 | p = bdty(UCALL, p);
|
|---|
| 1613 | } else if (q->n_op == CALL) {
|
|---|
| 1614 | blevel = 1;
|
|---|
| 1615 | argoff = ARGINIT;
|
|---|
| 1616 | if (oldstyle == 0)
|
|---|
| 1617 | q->n_right = listfw(q->n_right, funargs);
|
|---|
| 1618 | ftnarg(q);
|
|---|
| 1619 | blevel = 0;
|
|---|
| 1620 | }
|
|---|
| 1621 |
|
|---|
| 1622 | p = typ = tymerge(tp, p);
|
|---|
| 1623 | s = typ->n_sp = lookup((char *)typ->n_sp, 0); /* XXX */
|
|---|
| 1624 |
|
|---|
| 1625 | oclass = s->sclass;
|
|---|
| 1626 | if (class == STATIC && oclass == EXTERN)
|
|---|
| 1627 | werror("%s was first declared extern, then static", s->sname);
|
|---|
| 1628 |
|
|---|
| 1629 | if (fun_inline) {
|
|---|
| 1630 | /* special syntax for inline functions */
|
|---|
| 1631 | if (! strcmp(s->sname,"main"))
|
|---|
| 1632 | uerror("cannot inline main()");
|
|---|
| 1633 |
|
|---|
| 1634 | s->sflags |= SINLINE;
|
|---|
| 1635 | inline_start(s);
|
|---|
| 1636 | if (class == EXTERN)
|
|---|
| 1637 | class = EXTDEF;
|
|---|
| 1638 | } else if (class == EXTERN)
|
|---|
| 1639 | class = SNULL; /* same result */
|
|---|
| 1640 |
|
|---|
| 1641 | cftnsp = s;
|
|---|
| 1642 | defid(p, class);
|
|---|
| 1643 | #ifdef GCC_COMPAT
|
|---|
| 1644 | if (attr_find(p->n_ap, GCC_ATYP_ALW_INL)) {
|
|---|
| 1645 | /* Temporary turn on temps to make always_inline work */
|
|---|
| 1646 | alwinl = 1;
|
|---|
| 1647 | if (xtemps == 0) alwinl |= 2;
|
|---|
| 1648 | xtemps = 1;
|
|---|
| 1649 | }
|
|---|
| 1650 | #endif
|
|---|
| 1651 | prolab = getlab();
|
|---|
| 1652 | if ((c = cftnsp->soname) == NULL)
|
|---|
| 1653 | c = addname(exname(cftnsp->sname));
|
|---|
| 1654 | send_passt(IP_PROLOG, -1, c, cftnsp->stype,
|
|---|
| 1655 | cftnsp->sclass == EXTDEF, prolab, ctval);
|
|---|
| 1656 | blevel++;
|
|---|
| 1657 | #ifdef STABS
|
|---|
| 1658 | if (gflag)
|
|---|
| 1659 | stabs_func(s);
|
|---|
| 1660 | #endif
|
|---|
| 1661 | tfree(tp);
|
|---|
| 1662 | tfree(p);
|
|---|
| 1663 |
|
|---|
| 1664 | }
|
|---|
| 1665 |
|
|---|
| 1666 | static void
|
|---|
| 1667 | fend(void)
|
|---|
| 1668 | {
|
|---|
| 1669 | if (blevel)
|
|---|
| 1670 | cerror("function level error");
|
|---|
| 1671 | ftnend();
|
|---|
| 1672 | fun_inline = 0;
|
|---|
| 1673 | if (alwinl & 2) xtemps = 0;
|
|---|
| 1674 | alwinl = 0;
|
|---|
| 1675 | cftnsp = NULL;
|
|---|
| 1676 | }
|
|---|
| 1677 |
|
|---|
| 1678 | NODE *
|
|---|
| 1679 | structref(NODE *p, int f, char *name)
|
|---|
| 1680 | {
|
|---|
| 1681 | NODE *r;
|
|---|
| 1682 |
|
|---|
| 1683 | if (f == DOT)
|
|---|
| 1684 | p = buildtree(ADDROF, p, NIL);
|
|---|
| 1685 | r = biop(NAME, NIL, NIL);
|
|---|
| 1686 | r->n_name = name;
|
|---|
| 1687 | r = buildtree(STREF, p, r);
|
|---|
| 1688 | return r;
|
|---|
| 1689 | }
|
|---|
| 1690 |
|
|---|
| 1691 | static void
|
|---|
| 1692 | olddecl(NODE *p, NODE *a)
|
|---|
| 1693 | {
|
|---|
| 1694 | struct symtab *s;
|
|---|
| 1695 |
|
|---|
| 1696 | p = namekill(p, 0);
|
|---|
| 1697 | s = p->n_sp;
|
|---|
| 1698 | if (s->slevel != 1 || s->stype == UNDEF)
|
|---|
| 1699 | uerror("parameter '%s' not defined", s->sname);
|
|---|
| 1700 | else if (s->stype != FARG)
|
|---|
| 1701 | uerror("parameter '%s' redefined", s->sname);
|
|---|
| 1702 |
|
|---|
| 1703 | s->stype = p->n_type;
|
|---|
| 1704 | s->sdf = p->n_df;
|
|---|
| 1705 | s->sap = p->n_ap;
|
|---|
| 1706 | if (ISARY(s->stype)) {
|
|---|
| 1707 | s->stype += (PTR-ARY);
|
|---|
| 1708 | s->sdf++;
|
|---|
| 1709 | }
|
|---|
| 1710 | if (a)
|
|---|
| 1711 | attr_add(s->sap, gcc_attr_parse(a));
|
|---|
| 1712 | nfree(p);
|
|---|
| 1713 | }
|
|---|
| 1714 |
|
|---|
| 1715 | void
|
|---|
| 1716 | branch(int lbl)
|
|---|
| 1717 | {
|
|---|
| 1718 | int r = reached++;
|
|---|
| 1719 | ecomp(biop(GOTO, bcon(lbl), NIL));
|
|---|
| 1720 | reached = r;
|
|---|
| 1721 | }
|
|---|
| 1722 |
|
|---|
| 1723 | /*
|
|---|
| 1724 | * Create a printable string based on an encoded string.
|
|---|
| 1725 | */
|
|---|
| 1726 | static char *
|
|---|
| 1727 | mkpstr(char *str)
|
|---|
| 1728 | {
|
|---|
| 1729 | char *s, *os;
|
|---|
| 1730 | int v, l = strlen(str)+3; /* \t + \n + \0 */
|
|---|
| 1731 |
|
|---|
| 1732 | os = s = inlalloc(l);
|
|---|
| 1733 | *s++ = '\t';
|
|---|
| 1734 | for (; *str; ) {
|
|---|
| 1735 | if (*str++ == '\\')
|
|---|
| 1736 | v = esccon(&str);
|
|---|
| 1737 | else
|
|---|
| 1738 | v = str[-1];
|
|---|
| 1739 | *s++ = v;
|
|---|
| 1740 | }
|
|---|
| 1741 | *s++ = '\n';
|
|---|
| 1742 | *s = 0;
|
|---|
| 1743 | return os;
|
|---|
| 1744 | }
|
|---|
| 1745 |
|
|---|
| 1746 | /*
|
|---|
| 1747 | * Estimate the max length a string will have in its internal
|
|---|
| 1748 | * representation based on number of \ characters.
|
|---|
| 1749 | */
|
|---|
| 1750 | static int
|
|---|
| 1751 | maxstlen(char *str)
|
|---|
| 1752 | {
|
|---|
| 1753 | int i;
|
|---|
| 1754 |
|
|---|
| 1755 | for (i = 0; *str; str++, i++)
|
|---|
| 1756 | if (*str == '\\' || *str < 32 || *str > 0176)
|
|---|
| 1757 | i += 3;
|
|---|
| 1758 | return i;
|
|---|
| 1759 | }
|
|---|
| 1760 |
|
|---|
| 1761 | static char *
|
|---|
| 1762 | voct(char *d, unsigned int v)
|
|---|
| 1763 | {
|
|---|
| 1764 | v &= (1 << SZCHAR) - 1;
|
|---|
| 1765 | *d++ = '\\';
|
|---|
| 1766 | *d++ = v/64 + '0'; v &= 077;
|
|---|
| 1767 | *d++ = v/8 + '0'; v &= 7;
|
|---|
| 1768 | *d++ = v + '0';
|
|---|
| 1769 | return d;
|
|---|
| 1770 | }
|
|---|
| 1771 |
|
|---|
| 1772 |
|
|---|
| 1773 | /*
|
|---|
| 1774 | * Convert a string to internal format. The resulting string may be no
|
|---|
| 1775 | * more than len characters long.
|
|---|
| 1776 | */
|
|---|
| 1777 | static void
|
|---|
| 1778 | fixstr(char *d, char *s, int len)
|
|---|
| 1779 | {
|
|---|
| 1780 | unsigned int v;
|
|---|
| 1781 |
|
|---|
| 1782 | while (*s) {
|
|---|
| 1783 | if (len <= 0)
|
|---|
| 1784 | cerror("fixstr");
|
|---|
| 1785 | if (*s == '\\') {
|
|---|
| 1786 | s++;
|
|---|
| 1787 | v = esccon(&s);
|
|---|
| 1788 | d = voct(d, v);
|
|---|
| 1789 | len -= 4;
|
|---|
| 1790 | } else if (*s < ' ' || *s > 0176) {
|
|---|
| 1791 | d = voct(d, *s++);
|
|---|
| 1792 | len -= 4;
|
|---|
| 1793 | } else
|
|---|
| 1794 | *d++ = *s++, len--;
|
|---|
| 1795 | }
|
|---|
| 1796 | *d = 0;
|
|---|
| 1797 | }
|
|---|
| 1798 |
|
|---|
| 1799 | /*
|
|---|
| 1800 | * Add "raw" string new to cleaned string old.
|
|---|
| 1801 | */
|
|---|
| 1802 | static char *
|
|---|
| 1803 | stradd(char *old, char *new)
|
|---|
| 1804 | {
|
|---|
| 1805 | char *rv;
|
|---|
| 1806 | int len;
|
|---|
| 1807 |
|
|---|
| 1808 | if (*new == 'L' && new[1] == '\"')
|
|---|
| 1809 | widestr = 1, new++;
|
|---|
| 1810 | if (*new == '\"') {
|
|---|
| 1811 | new++; /* remove first " */
|
|---|
| 1812 | new[strlen(new) - 1] = 0;/* remove last " */
|
|---|
| 1813 | }
|
|---|
| 1814 | len = strlen(old) + maxstlen(new) + 1;
|
|---|
| 1815 | rv = tmpalloc(len);
|
|---|
| 1816 | strlcpy(rv, old, len);
|
|---|
| 1817 | fixstr(rv + strlen(old), new, maxstlen(new) + 1);
|
|---|
| 1818 | return rv;
|
|---|
| 1819 | }
|
|---|
| 1820 |
|
|---|
| 1821 | /*
|
|---|
| 1822 | * Fake a symtab entry for compound literals.
|
|---|
| 1823 | */
|
|---|
| 1824 | static struct symtab *
|
|---|
| 1825 | clbrace(NODE *p)
|
|---|
| 1826 | {
|
|---|
| 1827 | struct symtab *sp;
|
|---|
| 1828 |
|
|---|
| 1829 | sp = getsymtab(simname("cl"), STEMP);
|
|---|
| 1830 | sp->stype = p->n_type;
|
|---|
| 1831 | sp->squal = p->n_qual;
|
|---|
| 1832 | sp->sdf = p->n_df;
|
|---|
| 1833 | sp->sap = p->n_ap;
|
|---|
| 1834 | tfree(p);
|
|---|
| 1835 | if (blevel == 0 && xnf != NULL) {
|
|---|
| 1836 | sp->sclass = STATIC;
|
|---|
| 1837 | sp->slevel = 2;
|
|---|
| 1838 | sp->soffset = getlab();
|
|---|
| 1839 | } else {
|
|---|
| 1840 | sp->sclass = blevel ? AUTO : STATIC;
|
|---|
| 1841 | if (!ISARY(sp->stype) || sp->sdf->ddim != NOOFFSET) {
|
|---|
| 1842 | sp->soffset = NOOFFSET;
|
|---|
| 1843 | oalloc(sp, &autooff);
|
|---|
| 1844 | }
|
|---|
| 1845 | }
|
|---|
| 1846 | beginit(sp);
|
|---|
| 1847 | return sp;
|
|---|
| 1848 | }
|
|---|
| 1849 |
|
|---|
| 1850 | char *
|
|---|
| 1851 | simname(char *s)
|
|---|
| 1852 | {
|
|---|
| 1853 | int len = strlen(s) + 10 + 1;
|
|---|
| 1854 | char *w = tmpalloc(len);
|
|---|
| 1855 |
|
|---|
| 1856 | snprintf(w, len, "%s%d", s, getlab());
|
|---|
| 1857 | return w;
|
|---|
| 1858 | }
|
|---|
| 1859 |
|
|---|
| 1860 | NODE *
|
|---|
| 1861 | biop(int op, NODE *l, NODE *r)
|
|---|
| 1862 | {
|
|---|
| 1863 | return block(op, l, r, INT, 0, MKAP(INT));
|
|---|
| 1864 | }
|
|---|
| 1865 |
|
|---|
| 1866 | static NODE *
|
|---|
| 1867 | cmop(NODE *l, NODE *r)
|
|---|
| 1868 | {
|
|---|
| 1869 | return biop(CM, l, r);
|
|---|
| 1870 | }
|
|---|
| 1871 |
|
|---|
| 1872 | static NODE *
|
|---|
| 1873 | voidcon(void)
|
|---|
| 1874 | {
|
|---|
| 1875 | return block(ICON, NIL, NIL, STRTY, 0, MKAP(VOID));
|
|---|
| 1876 | }
|
|---|
| 1877 |
|
|---|
| 1878 | /* Support for extended assembler a' la' gcc style follows below */
|
|---|
| 1879 |
|
|---|
| 1880 | static NODE *
|
|---|
| 1881 | xmrg(NODE *out, NODE *in)
|
|---|
| 1882 | {
|
|---|
| 1883 | NODE *p = in;
|
|---|
| 1884 |
|
|---|
| 1885 | if (p->n_op == XARG) {
|
|---|
| 1886 | in = cmop(out, p);
|
|---|
| 1887 | } else {
|
|---|
| 1888 | while (p->n_left->n_op == CM)
|
|---|
| 1889 | p = p->n_left;
|
|---|
| 1890 | p->n_left = cmop(out, p->n_left);
|
|---|
| 1891 | }
|
|---|
| 1892 | return in;
|
|---|
| 1893 | }
|
|---|
| 1894 |
|
|---|
| 1895 | /*
|
|---|
| 1896 | * Put together in and out node lists in one list, and balance it with
|
|---|
| 1897 | * the constraints on the right side of a CM node.
|
|---|
| 1898 | */
|
|---|
| 1899 | static NODE *
|
|---|
| 1900 | xcmop(NODE *out, NODE *in, NODE *str)
|
|---|
| 1901 | {
|
|---|
| 1902 | NODE *p, *q;
|
|---|
| 1903 |
|
|---|
| 1904 | if (out) {
|
|---|
| 1905 | /* D out-list sanity check */
|
|---|
| 1906 | for (p = out; p->n_op == CM; p = p->n_left) {
|
|---|
| 1907 | q = p->n_right;
|
|---|
| 1908 | if (q->n_name[0] != '=' && q->n_name[0] != '+')
|
|---|
| 1909 | uerror("output missing =");
|
|---|
| 1910 | }
|
|---|
| 1911 | if (p->n_name[0] != '=' && p->n_name[0] != '+')
|
|---|
| 1912 | uerror("output missing =");
|
|---|
| 1913 | if (in == NIL)
|
|---|
| 1914 | p = out;
|
|---|
| 1915 | else
|
|---|
| 1916 | p = xmrg(out, in);
|
|---|
| 1917 | } else if (in) {
|
|---|
| 1918 | p = in;
|
|---|
| 1919 | } else
|
|---|
| 1920 | p = voidcon();
|
|---|
| 1921 |
|
|---|
| 1922 | if (str == NIL)
|
|---|
| 1923 | str = voidcon();
|
|---|
| 1924 | return cmop(p, str);
|
|---|
| 1925 | }
|
|---|
| 1926 |
|
|---|
| 1927 | /*
|
|---|
| 1928 | * Generate a XARG node based on a string and an expression.
|
|---|
| 1929 | */
|
|---|
| 1930 | static NODE *
|
|---|
| 1931 | xasmop(char *str, NODE *p)
|
|---|
| 1932 | {
|
|---|
| 1933 |
|
|---|
| 1934 | p = biop(XARG, p, NIL);
|
|---|
| 1935 | p->n_name = isinlining ? newstring(str, strlen(str)+1) : str;
|
|---|
| 1936 | return p;
|
|---|
| 1937 | }
|
|---|
| 1938 |
|
|---|
| 1939 | /*
|
|---|
| 1940 | * Generate a XASM node based on a string and an expression.
|
|---|
| 1941 | */
|
|---|
| 1942 | static void
|
|---|
| 1943 | mkxasm(char *str, NODE *p)
|
|---|
| 1944 | {
|
|---|
| 1945 | NODE *q;
|
|---|
| 1946 |
|
|---|
| 1947 | q = biop(XASM, p->n_left, p->n_right);
|
|---|
| 1948 | q->n_name = isinlining ? newstring(str, strlen(str)+1) : str;
|
|---|
| 1949 | nfree(p);
|
|---|
| 1950 | ecomp(q);
|
|---|
| 1951 | }
|
|---|
| 1952 |
|
|---|
| 1953 | #ifdef GCC_COMPAT
|
|---|
| 1954 | static NODE *
|
|---|
| 1955 | tyof(NODE *p)
|
|---|
| 1956 | {
|
|---|
| 1957 | static struct symtab spp;
|
|---|
| 1958 | NODE *q = block(TYPE, NIL, NIL, p->n_type, p->n_df, p->n_ap);
|
|---|
| 1959 | q->n_qual = p->n_qual;
|
|---|
| 1960 | q->n_sp = &spp; /* for typenode */
|
|---|
| 1961 | tfree(p);
|
|---|
| 1962 | return q;
|
|---|
| 1963 | }
|
|---|
| 1964 | #endif
|
|---|
| 1965 |
|
|---|
| 1966 | /*
|
|---|
| 1967 | * Traverse an unhandled expression tree bottom-up and call buildtree()
|
|---|
| 1968 | * or equivalent as needed.
|
|---|
| 1969 | */
|
|---|
| 1970 | NODE *
|
|---|
| 1971 | eve(NODE *p)
|
|---|
| 1972 | {
|
|---|
| 1973 | struct symtab *sp;
|
|---|
| 1974 | NODE *r, *p1, *p2;
|
|---|
| 1975 | int x;
|
|---|
| 1976 |
|
|---|
| 1977 | p1 = p->n_left;
|
|---|
| 1978 | p2 = p->n_right;
|
|---|
| 1979 | switch (p->n_op) {
|
|---|
| 1980 | case NAME:
|
|---|
| 1981 | sp = lookup((char *)p->n_sp, 0);
|
|---|
| 1982 | if (sp->sflags & SINLINE)
|
|---|
| 1983 | inline_ref(sp);
|
|---|
| 1984 | r = nametree(sp);
|
|---|
| 1985 | if (sp->sflags & SDYNARRAY)
|
|---|
| 1986 | r = buildtree(UMUL, r, NIL);
|
|---|
| 1987 | #ifdef GCC_COMPAT
|
|---|
| 1988 | if (attr_find(sp->sap, GCC_ATYP_DEPRECATED))
|
|---|
| 1989 | werror("`%s' is deprecated", sp->sname);
|
|---|
| 1990 | #endif
|
|---|
| 1991 | break;
|
|---|
| 1992 |
|
|---|
| 1993 | case DOT:
|
|---|
| 1994 | case STREF:
|
|---|
| 1995 | r = structref(eve(p1), p->n_op, (char *)p2->n_sp);
|
|---|
| 1996 | nfree(p2);
|
|---|
| 1997 | break;
|
|---|
| 1998 |
|
|---|
| 1999 | case CAST:
|
|---|
| 2000 | p1 = buildtree(CAST, p1, eve(p2));
|
|---|
| 2001 | nfree(p1->n_left);
|
|---|
| 2002 | r = p1->n_right;
|
|---|
| 2003 | nfree(p1);
|
|---|
| 2004 | break;
|
|---|
| 2005 |
|
|---|
| 2006 |
|
|---|
| 2007 | case SZOF:
|
|---|
| 2008 | x = xinline; xinline = 0; /* XXX hack */
|
|---|
| 2009 | if (p2->n_lval == 0)
|
|---|
| 2010 | p1 = eve(p1);
|
|---|
| 2011 | else
|
|---|
| 2012 | TYMFIX(p1);
|
|---|
| 2013 | nfree(p2);
|
|---|
| 2014 | r = doszof(p1);
|
|---|
| 2015 | xinline = x;
|
|---|
| 2016 | break;
|
|---|
| 2017 |
|
|---|
| 2018 | case LB:
|
|---|
| 2019 | p1 = eve(p->n_left);
|
|---|
| 2020 | r = buildtree(UMUL, buildtree(PLUS, p1, eve(p2)), NIL);
|
|---|
| 2021 | break;
|
|---|
| 2022 |
|
|---|
| 2023 | case COMPL:
|
|---|
| 2024 | #ifndef NO_COMPLEX
|
|---|
| 2025 | p1 = eve(p1);
|
|---|
| 2026 | if (ANYCX(p1))
|
|---|
| 2027 | r = cxconj(p1);
|
|---|
| 2028 | else
|
|---|
| 2029 | r = buildtree(COMPL, p1, NIL);
|
|---|
| 2030 | break;
|
|---|
| 2031 | #endif
|
|---|
| 2032 | case UMINUS:
|
|---|
| 2033 | case NOT:
|
|---|
| 2034 | case UMUL:
|
|---|
| 2035 | r = buildtree(p->n_op, eve(p->n_left), NIL);
|
|---|
| 2036 | break;
|
|---|
| 2037 |
|
|---|
| 2038 | case ADDROF:
|
|---|
| 2039 | r = eve(p1);
|
|---|
| 2040 | if (ISFTN(p->n_type)/* || ISARY(p->n_type) */){
|
|---|
| 2041 | #ifdef notdef
|
|---|
| 2042 | werror( "& before array or function: ignored" );
|
|---|
| 2043 | #endif
|
|---|
| 2044 | } else
|
|---|
| 2045 | r = buildtree(ADDROF, r, NIL);
|
|---|
| 2046 | break;
|
|---|
| 2047 |
|
|---|
| 2048 | case CALL:
|
|---|
| 2049 | p2 = eve(p2);
|
|---|
| 2050 | /* FALLTHROUGH */
|
|---|
| 2051 | case UCALL:
|
|---|
| 2052 | if (p1->n_op == NAME) {
|
|---|
| 2053 | sp = lookup((char *)p1->n_sp, 0);
|
|---|
| 2054 | if (sp->stype == UNDEF) {
|
|---|
| 2055 | p1->n_type = FTN|INT;
|
|---|
| 2056 | p1->n_sp = sp;
|
|---|
| 2057 | p1->n_ap = MKAP(INT);
|
|---|
| 2058 | defid(p1, EXTERN);
|
|---|
| 2059 | }
|
|---|
| 2060 | nfree(p1);
|
|---|
| 2061 | #ifdef GCC_COMPAT
|
|---|
| 2062 | if (attr_find(sp->sap, GCC_ATYP_DEPRECATED))
|
|---|
| 2063 | werror("`%s' is deprecated", sp->sname);
|
|---|
| 2064 | #endif
|
|---|
| 2065 | r = doacall(sp, nametree(sp), p2);
|
|---|
| 2066 | } else
|
|---|
| 2067 | r = doacall(NULL, eve(p1), p2);
|
|---|
| 2068 | break;
|
|---|
| 2069 |
|
|---|
| 2070 | #ifndef NO_COMPLEX
|
|---|
| 2071 | case XREAL:
|
|---|
| 2072 | case XIMAG:
|
|---|
| 2073 | p1 = eve(p1);
|
|---|
| 2074 | r = cxelem(p->n_op, p1);
|
|---|
| 2075 | break;
|
|---|
| 2076 | #endif
|
|---|
| 2077 |
|
|---|
| 2078 | case MUL:
|
|---|
| 2079 | case DIV:
|
|---|
| 2080 | case PLUS:
|
|---|
| 2081 | case MINUS:
|
|---|
| 2082 | case ASSIGN:
|
|---|
| 2083 | case EQ:
|
|---|
| 2084 | case NE:
|
|---|
| 2085 | #ifndef NO_COMPLEX
|
|---|
| 2086 | p1 = eve(p1);
|
|---|
| 2087 | p2 = eve(p2);
|
|---|
| 2088 | if (ANYCX(p1) || ANYCX(p2)) {
|
|---|
| 2089 | r = cxop(p->n_op, p1, p2);
|
|---|
| 2090 | } else if (ISITY(p1->n_type) || ISITY(p2->n_type)) {
|
|---|
| 2091 | r = imop(p->n_op, p1, p2);
|
|---|
| 2092 | } else
|
|---|
| 2093 | r = buildtree(p->n_op, p1, p2);
|
|---|
| 2094 | break;
|
|---|
| 2095 | #endif
|
|---|
| 2096 | case MOD:
|
|---|
| 2097 | case INCR:
|
|---|
| 2098 | case DECR:
|
|---|
| 2099 | case CM:
|
|---|
| 2100 | case GT:
|
|---|
| 2101 | case GE:
|
|---|
| 2102 | case LT:
|
|---|
| 2103 | case LE:
|
|---|
| 2104 | case RS:
|
|---|
| 2105 | case LS:
|
|---|
| 2106 | case RSEQ:
|
|---|
| 2107 | case LSEQ:
|
|---|
| 2108 | case AND:
|
|---|
| 2109 | case OR:
|
|---|
| 2110 | case ER:
|
|---|
| 2111 | case OROR:
|
|---|
| 2112 | case ANDAND:
|
|---|
| 2113 | case EREQ:
|
|---|
| 2114 | case OREQ:
|
|---|
| 2115 | case ANDEQ:
|
|---|
| 2116 | case MINUSEQ:
|
|---|
| 2117 | case PLUSEQ:
|
|---|
| 2118 | case MULEQ:
|
|---|
| 2119 | case DIVEQ:
|
|---|
| 2120 | case MODEQ:
|
|---|
| 2121 | case QUEST:
|
|---|
| 2122 | case COLON:
|
|---|
| 2123 | p1 = eve(p1);
|
|---|
| 2124 | r = buildtree(p->n_op, p1, eve(p2));
|
|---|
| 2125 | break;
|
|---|
| 2126 |
|
|---|
| 2127 | case STRING:
|
|---|
| 2128 | r = strend(p->n_lval, p->n_name);
|
|---|
| 2129 | break;
|
|---|
| 2130 |
|
|---|
| 2131 | case COMOP:
|
|---|
| 2132 | if (p1->n_op == GOTO) {
|
|---|
| 2133 | /* inside ({ }), eve already called */
|
|---|
| 2134 | r = buildtree(p->n_op, p1, p2);
|
|---|
| 2135 | } else {
|
|---|
| 2136 | p1 = eve(p1);
|
|---|
| 2137 | r = buildtree(p->n_op, p1, eve(p2));
|
|---|
| 2138 | }
|
|---|
| 2139 | break;
|
|---|
| 2140 |
|
|---|
| 2141 | case TYPE:
|
|---|
| 2142 | case ICON:
|
|---|
| 2143 | case FCON:
|
|---|
| 2144 | case TEMP:
|
|---|
| 2145 | return p;
|
|---|
| 2146 |
|
|---|
| 2147 | case CLOP:
|
|---|
| 2148 | r = nametree(p->n_sp);
|
|---|
| 2149 | break;
|
|---|
| 2150 |
|
|---|
| 2151 | default:
|
|---|
| 2152 | #ifdef PCC_DEBUG
|
|---|
| 2153 | fwalk(p, eprint, 0);
|
|---|
| 2154 | #endif
|
|---|
| 2155 | cerror("eve");
|
|---|
| 2156 | r = NIL;
|
|---|
| 2157 | }
|
|---|
| 2158 | nfree(p);
|
|---|
| 2159 | return r;
|
|---|
| 2160 | }
|
|---|
| 2161 |
|
|---|
| 2162 | int
|
|---|
| 2163 | con_e(NODE *p)
|
|---|
| 2164 | {
|
|---|
| 2165 | return icons(eve(p));
|
|---|
| 2166 | }
|
|---|
| 2167 |
|
|---|
| 2168 | void
|
|---|
| 2169 | uawarn(NODE *p, char *s)
|
|---|
| 2170 | {
|
|---|
| 2171 | if (p == 0)
|
|---|
| 2172 | return;
|
|---|
| 2173 | if (attrwarn)
|
|---|
| 2174 | werror("unhandled %s attribute", s);
|
|---|
| 2175 | tfree(p);
|
|---|
| 2176 | }
|
|---|
| 2177 |
|
|---|
| 2178 | static void
|
|---|
| 2179 | dainit(NODE *d, NODE *a)
|
|---|
| 2180 | {
|
|---|
| 2181 | if (d == NULL) {
|
|---|
| 2182 | asginit(a);
|
|---|
| 2183 | } else if (d->n_op == CM) {
|
|---|
| 2184 | int is = con_e(d->n_left);
|
|---|
| 2185 | int ie = con_e(d->n_right);
|
|---|
| 2186 | int i;
|
|---|
| 2187 |
|
|---|
| 2188 | nfree(d);
|
|---|
| 2189 | if (ie < is)
|
|---|
| 2190 | uerror("negative initializer range");
|
|---|
| 2191 | desinit(biop(LB, NIL, bcon(is)));
|
|---|
| 2192 | for (i = is; i < ie; i++)
|
|---|
| 2193 | asginit(ccopy(a));
|
|---|
| 2194 | asginit(a);
|
|---|
| 2195 | } else {
|
|---|
| 2196 | cerror("dainit");
|
|---|
| 2197 | }
|
|---|
| 2198 | }
|
|---|
| 2199 |
|
|---|
| 2200 | /*
|
|---|
| 2201 | * Traverse down and tymerge() where appropriate.
|
|---|
| 2202 | */
|
|---|
| 2203 | static NODE *
|
|---|
| 2204 | tymfix(NODE *p)
|
|---|
| 2205 | {
|
|---|
| 2206 | NODE *q;
|
|---|
| 2207 | int o = coptype(p->n_op);
|
|---|
| 2208 |
|
|---|
| 2209 | switch (o) {
|
|---|
| 2210 | case LTYPE:
|
|---|
| 2211 | break;
|
|---|
| 2212 | case UTYPE:
|
|---|
| 2213 | p->n_left = tymfix(p->n_left);
|
|---|
| 2214 | break;
|
|---|
| 2215 | case BITYPE:
|
|---|
| 2216 | p->n_left = tymfix(p->n_left);
|
|---|
| 2217 | p->n_right = tymfix(p->n_right);
|
|---|
| 2218 | if (p->n_op == TYMERGE) {
|
|---|
| 2219 | q = tymerge(p->n_left, p->n_right);
|
|---|
| 2220 | q->n_ap = attr_add(q->n_ap, p->n_ap);
|
|---|
| 2221 | tfree(p->n_left);
|
|---|
| 2222 | nfree(p);
|
|---|
| 2223 | p = q;
|
|---|
| 2224 | }
|
|---|
| 2225 | break;
|
|---|
| 2226 | }
|
|---|
| 2227 | return p;
|
|---|
| 2228 | }
|
|---|
| 2229 |
|
|---|
| 2230 | static NODE *
|
|---|
| 2231 | aryfix(NODE *p)
|
|---|
| 2232 | {
|
|---|
| 2233 | NODE *q;
|
|---|
| 2234 |
|
|---|
| 2235 | for (q = p; q->n_op != NAME; q = q->n_left) {
|
|---|
| 2236 | if (q->n_op == LB) {
|
|---|
| 2237 | q->n_right = optim(eve(q->n_right));
|
|---|
| 2238 | if ((blevel == 0 || rpole != NULL) &&
|
|---|
| 2239 | !nncon(q->n_right))
|
|---|
| 2240 | uerror("array size not constant");
|
|---|
| 2241 | /*
|
|---|
| 2242 | * Checks according to 6.7.5.2 clause 1:
|
|---|
| 2243 | * "...the expression shall have an integer type."
|
|---|
| 2244 | * "If the expression is a constant expression,
|
|---|
| 2245 | * it shall have a value greater than zero."
|
|---|
| 2246 | */
|
|---|
| 2247 | if (!ISINTEGER(q->n_right->n_type))
|
|---|
| 2248 | werror("array size is not an integer");
|
|---|
| 2249 | else if (q->n_right->n_op == ICON &&
|
|---|
| 2250 | q->n_right->n_lval < 0 &&
|
|---|
| 2251 | q->n_right->n_lval != NOOFFSET) {
|
|---|
| 2252 | uerror("array size cannot be negative");
|
|---|
| 2253 | q->n_right->n_lval = 1;
|
|---|
| 2254 | }
|
|---|
| 2255 | } else if (q->n_op == CALL)
|
|---|
| 2256 | q->n_right = namekill(q->n_right, 1);
|
|---|
| 2257 | }
|
|---|
| 2258 | return p;
|
|---|
| 2259 | }
|
|---|