Annotation of GNUtools/cc/c-lex.c, revision 1.1.1.1

1.1       root        1: /* Lexical analyzer for C and Objective C.
                      2:    Copyright (C) 1987, 1988, 1989, 1992 Free Software Foundation, Inc.
                      3: 
                      4: This file is part of GNU CC.
                      5: 
                      6: GNU CC is free software; you can redistribute it and/or modify
                      7: it under the terms of the GNU General Public License as published by
                      8: the Free Software Foundation; either version 2, or (at your option)
                      9: any later version.
                     10: 
                     11: GNU CC is distributed in the hope that it will be useful,
                     12: but WITHOUT ANY WARRANTY; without even the implied warranty of
                     13: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     14: GNU General Public License for more details.
                     15: 
                     16: You should have received a copy of the GNU General Public License
                     17: along with GNU CC; see the file COPYING.  If not, write to
                     18: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
                     19: 
                     20: 
                     21: #include <stdio.h>
                     22: #include <errno.h>
                     23: #include <setjmp.h>
                     24: 
                     25: #include "config.h"
                     26: #include "rtl.h"
                     27: #include "tree.h"
                     28: #include "input.h"
                     29: #include "c-lex.h"
                     30: #include "c-tree.h"
                     31: #include "flags.h"
                     32: #include "c-parse.h"
                     33: 
                     34: #include <ctype.h>
                     35: 
                     36: #ifdef MULTIBYTE_CHARS
                     37: #include <stdlib.h>
                     38: #include <locale.h>
                     39: #endif
                     40: 
                     41: #ifndef errno
                     42: extern int errno;
                     43: #endif
                     44: 
                     45: /* The elements of `ridpointers' are identifier nodes
                     46:    for the reserved type names and storage classes.
                     47:    It is indexed by a RID_... value.  */
                     48: tree ridpointers[(int) RID_MAX];
                     49: 
                     50: /* Cause the `yydebug' variable to be defined.  */
                     51: #define YYDEBUG 1
                     52: 
                     53: /* the declaration found for the last IDENTIFIER token read in.
                     54:    yylex must look this up to detect typedefs, which get token type TYPENAME,
                     55:    so it is left around in case the identifier is not a typedef but is
                     56:    used in a context which makes it a reference to a variable.  */
                     57: tree lastiddecl;
                     58: 
                     59: /* Nonzero enables objc features.  */
                     60: 
                     61: int doing_objc_thang;
                     62: 
                     63: extern tree is_class_name ();
                     64: 
                     65: extern int yydebug;
                     66: 
                     67: /* File used for outputting assembler code.  */
                     68: extern FILE *asm_out_file;
                     69: 
                     70: #ifndef WCHAR_TYPE_SIZE
                     71: #ifdef INT_TYPE_SIZE
                     72: #define WCHAR_TYPE_SIZE INT_TYPE_SIZE
                     73: #else
                     74: #define WCHAR_TYPE_SIZE        BITS_PER_WORD
                     75: #endif
                     76: #endif
                     77: 
                     78: /* Number of bytes in a wide character.  */
                     79: #define WCHAR_BYTES (WCHAR_TYPE_SIZE / BITS_PER_UNIT)
                     80: 
                     81: static int maxtoken;           /* Current nominal length of token buffer.  */
                     82: char *token_buffer;    /* Pointer to token buffer.
                     83:                           Actual allocated length is maxtoken + 2.
                     84:                           This is not static because objc-parse.y uses it.  */
                     85: 
                     86: /* Nonzero if end-of-file has been seen on input.  */
                     87: static int end_of_file;
                     88: 
                     89: /* Buffered-back input character; faster than using ungetc.  */
                     90: static int nextchar = -1;
                     91: 
                     92: int check_newline ();
                     93: 
                     94: /* C code produced by gperf version 2.5 (GNU C++ version) */
                     95: /* Command-line: gperf -p -j1 -i 1 -g -o -t -G -N is_reserved_word -k1,3,$ c-parse.gperf  */
                     96: struct resword { char *name; short token; enum rid rid; };
                     97: 
                     98: #define TOTAL_KEYWORDS 79
                     99: #define MIN_WORD_LENGTH 2
                    100: #define MAX_WORD_LENGTH 20
                    101: #define MIN_HASH_VALUE 10
                    102: #define MAX_HASH_VALUE 144
                    103: /* maximum key range = 135, duplicates = 0 */
                    104: 
                    105: #ifdef __GNUC__
                    106: __inline
                    107: #endif
                    108: static unsigned int
                    109: hash (str, len)
                    110:      register char *str;
                    111:      register int unsigned len;
                    112: {
                    113:   static unsigned char asso_values[] =
                    114:     {
                    115:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    116:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    117:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    118:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    119:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    120:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    121:      145, 145, 145, 145,  25, 145, 145, 145, 145, 145,
                    122:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    123:      145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
                    124:      145, 145, 145, 145, 145,   1, 145,  46,   8,  15,
                    125:       61,   6,  36,  48,   3,   5, 145,  18,  63,  25,
                    126:       29,  76,   1, 145,  13,   2,   1,  51,  37,   9,
                    127:        9,   1,   3, 145, 145, 145, 145, 145,
                    128:     };
                    129:   register int hval = len;
                    130: 
                    131:   switch (hval)
                    132:     {
                    133:       default:
                    134:       case 3:
                    135:         hval += asso_values[str[2]];
                    136:       case 2:
                    137:       case 1:
                    138:         hval += asso_values[str[0]];
                    139:     }
                    140:   return hval + asso_values[str[len - 1]];
                    141: }
                    142: 
                    143: static struct resword wordlist[] =
                    144: {
                    145:   {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, 
                    146:   {"",}, 
                    147:   {"int",  TYPESPEC, RID_INT},
                    148:   {"",}, {"",}, 
                    149:   {"__typeof__",  TYPEOF, NORID},
                    150:   {"__signed__",  TYPESPEC, RID_SIGNED},
                    151:   {"__imag__",  IMAGPART, NORID},
                    152:   {"switch",  SWITCH, NORID},
                    153:   {"__inline__",  SCSPEC, RID_INLINE},
                    154:   {"else",  ELSE, NORID},
                    155:   {"__iterator__",  SCSPEC, RID_ITERATOR},
                    156:   {"__inline",  SCSPEC, RID_INLINE},
                    157:   {"__extension__",  EXTENSION, NORID},
                    158:   {"struct",  STRUCT, NORID},
                    159:   {"__real__",  REALPART, NORID},
                    160:   {"__const",  TYPE_QUAL, RID_CONST},
                    161:   {"while",  WHILE, NORID},
                    162:   {"__const__",  TYPE_QUAL, RID_CONST},
                    163:   {"case",  CASE, NORID},
                    164:   {"__complex__",  TYPESPEC, RID_COMPLEX},
                    165:   {"__iterator",  SCSPEC, RID_ITERATOR},
                    166:   {"bycopy",  TYPE_QUAL, RID_BYCOPY},
                    167:   {"",}, {"",}, {"",}, 
                    168:   {"__complex",  TYPESPEC, RID_COMPLEX},
                    169:   {"",}, 
                    170:   {"in",  TYPE_QUAL, RID_IN},
                    171:   {"break",  BREAK, NORID},
                    172:   {"@defs",  DEFS, NORID},
                    173:   {"",}, {"",}, {"",}, 
                    174:   {"extern",  SCSPEC, RID_EXTERN},
                    175:   {"if",  IF, NORID},
                    176:   {"typeof",  TYPEOF, NORID},
                    177:   {"typedef",  SCSPEC, RID_TYPEDEF},
                    178:   {"__typeof",  TYPEOF, NORID},
                    179:   {"sizeof",  SIZEOF, NORID},
                    180:   {"",}, 
                    181:   {"return",  RETURN, NORID},
                    182:   {"const",  TYPE_QUAL, RID_CONST},
                    183:   {"__volatile__",  TYPE_QUAL, RID_VOLATILE},
                    184:   {"@private",  PRIVATE, NORID},
                    185:   {"@selector",  SELECTOR, NORID},
                    186:   {"__volatile",  TYPE_QUAL, RID_VOLATILE},
                    187:   {"__asm__",  ASM_KEYWORD, NORID},
                    188:   {"",}, {"",}, 
                    189:   {"continue",  CONTINUE, NORID},
                    190:   {"__alignof__",  ALIGNOF, NORID},
                    191:   {"__imag",  IMAGPART, NORID},
                    192:   {"__attribute__",  ATTRIBUTE, NORID},
                    193:   {"",}, {"",}, 
                    194:   {"__attribute",  ATTRIBUTE, NORID},
                    195:   {"for",  FOR, NORID},
                    196:   {"",}, 
                    197:   {"@encode",  ENCODE, NORID},
                    198:   {"id",  OBJECTNAME, RID_ID},
                    199:   {"static",  SCSPEC, RID_STATIC},
                    200:   {"@interface",  INTERFACE, NORID},
                    201:   {"",}, 
                    202:   {"__signed",  TYPESPEC, RID_SIGNED},
                    203:   {"",}, 
                    204:   {"__label__",  LABEL, NORID},
                    205:   {"",}, {"",}, 
                    206:   {"__asm",  ASM_KEYWORD, NORID},
                    207:   {"char",  TYPESPEC, RID_CHAR},
                    208:   {"",}, 
                    209:   {"inline",  SCSPEC, RID_INLINE},
                    210:   {"out",  TYPE_QUAL, RID_OUT},
                    211:   {"register",  SCSPEC, RID_REGISTER},
                    212:   {"__real",  REALPART, NORID},
                    213:   {"short",  TYPESPEC, RID_SHORT},
                    214:   {"",}, 
                    215:   {"enum",  ENUM, NORID},
                    216:   {"inout",  TYPE_QUAL, RID_INOUT},
                    217:   {"",}, 
                    218:   {"oneway",  TYPE_QUAL, RID_ONEWAY},
                    219:   {"union",  UNION, NORID},
                    220:   {"",}, 
                    221:   {"__alignof",  ALIGNOF, NORID},
                    222:   {"",}, 
                    223:   {"@implementation",  IMPLEMENTATION, NORID},
                    224:   {"",}, 
                    225:   {"@class",  CLASS, NORID},
                    226:   {"",}, 
                    227:   {"@public",  PUBLIC, NORID},
                    228:   {"asm",  ASM_KEYWORD, NORID},
                    229:   {"",}, {"",}, {"",}, {"",}, {"",}, 
                    230:   {"default",  DEFAULT, NORID},
                    231:   {"",}, 
                    232:   {"void",  TYPESPEC, RID_VOID},
                    233:   {"",}, 
                    234:   {"@protected",  PROTECTED, NORID},
                    235:   {"@protocol",  PROTOCOL, NORID},
                    236:   {"",}, {"",}, {"",}, 
                    237:   {"volatile",  TYPE_QUAL, RID_VOLATILE},
                    238:   {"",}, {"",}, 
                    239:   {"signed",  TYPESPEC, RID_SIGNED},
                    240:   {"float",  TYPESPEC, RID_FLOAT},
                    241:   {"@end",  END, NORID},
                    242:   {"",}, {"",}, 
                    243:   {"unsigned",  TYPESPEC, RID_UNSIGNED},
                    244:   {"@compatibility_alias",  ALIAS, NORID},
                    245:   {"double",  TYPESPEC, RID_DOUBLE},
                    246:   {"",}, {"",}, 
                    247:   {"auto",  SCSPEC, RID_AUTO},
                    248:   {"",}, 
                    249:   {"goto",  GOTO, NORID},
                    250:   {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, 
                    251:   {"do",  DO, NORID},
                    252:   {"",}, {"",}, {"",}, {"",}, 
                    253:   {"long",  TYPESPEC, RID_LONG},
                    254: };
                    255: 
                    256: #ifdef __GNUC__
                    257: __inline
                    258: #endif
                    259: struct resword *
                    260: is_reserved_word (str, len)
                    261:      register char *str;
                    262:      register unsigned int len;
                    263: {
                    264:   if (len <= MAX_WORD_LENGTH && len >= MIN_WORD_LENGTH)
                    265:     {
                    266:       register int key = hash (str, len);
                    267: 
                    268:       if (key <= MAX_HASH_VALUE && key >= 0)
                    269:         {
                    270:           register char *s = wordlist[key].name;
                    271: 
                    272:           if (*s == *str && !strcmp (str + 1, s + 1))
                    273:             return &wordlist[key];
                    274:         }
                    275:     }
                    276:   return 0;
                    277: }
                    278: 
                    279: /* Return something to represent absolute declarators containing a *.
                    280:    TARGET is the absolute declarator that the * contains.
                    281:    TYPE_QUALS is a list of modifiers such as const or volatile
                    282:    to apply to the pointer type, represented as identifiers.
                    283: 
                    284:    We return an INDIRECT_REF whose "contents" are TARGET
                    285:    and whose type is the modifier list.  */
                    286: 
                    287: tree
                    288: make_pointer_declarator (type_quals, target)
                    289:      tree type_quals, target;
                    290: {
                    291:   return build1 (INDIRECT_REF, type_quals, target);
                    292: }
                    293: 
                    294: void
                    295: forget_protocol_qualifiers ()
                    296: {
                    297:   int i, n = sizeof wordlist / sizeof (struct resword);
                    298: 
                    299:   for (i = 0; i < n; i++)
                    300:     if ((int) wordlist[i].rid >= (int) RID_IN
                    301:         && (int) wordlist[i].rid <= (int) RID_ONEWAY)
                    302:       wordlist[i].name = "";
                    303: }
                    304: 
                    305: void
                    306: remember_protocol_qualifiers ()
                    307: {
                    308:   int i, n = sizeof wordlist / sizeof (struct resword);
                    309: 
                    310:   for (i = 0; i < n; i++)
                    311:     if (wordlist[i].rid == RID_IN)
                    312:       wordlist[i].name = "in";
                    313:     else if (wordlist[i].rid == RID_OUT)
                    314:       wordlist[i].name = "out";
                    315:     else if (wordlist[i].rid == RID_INOUT)
                    316:       wordlist[i].name = "inout";
                    317:     else if (wordlist[i].rid == RID_BYCOPY)
                    318:       wordlist[i].name = "bycopy";
                    319:     else if (wordlist[i].rid == RID_ONEWAY)
                    320:       wordlist[i].name = "oneway";   
                    321: }
                    322: 
                    323: void
                    324: init_lex ()
                    325: {
                    326:   /* Make identifier nodes long enough for the language-specific slots.  */
                    327:   set_identifier_size (sizeof (struct lang_identifier));
                    328: 
                    329:   /* Start it at 0, because check_newline is called at the very beginning
                    330:      and will increment it to 1.  */
                    331:   lineno = 0;
                    332: 
                    333: #ifdef MULTIBYTE_CHARS
                    334:   /* Change to the native locale for multibyte conversions.  */
                    335:   setlocale (LC_CTYPE, "");
                    336: #endif
                    337: 
                    338:   maxtoken = 40;
                    339:   token_buffer = (char *) xmalloc (maxtoken + 2);
                    340: 
                    341:   ridpointers[(int) RID_INT] = get_identifier ("int");
                    342:   ridpointers[(int) RID_CHAR] = get_identifier ("char");
                    343:   ridpointers[(int) RID_VOID] = get_identifier ("void");
                    344:   ridpointers[(int) RID_FLOAT] = get_identifier ("float");
                    345:   ridpointers[(int) RID_DOUBLE] = get_identifier ("double");
                    346:   ridpointers[(int) RID_SHORT] = get_identifier ("short");
                    347:   ridpointers[(int) RID_LONG] = get_identifier ("long");
                    348:   ridpointers[(int) RID_UNSIGNED] = get_identifier ("unsigned");
                    349:   ridpointers[(int) RID_SIGNED] = get_identifier ("signed");
                    350:   ridpointers[(int) RID_INLINE] = get_identifier ("inline");
                    351:   ridpointers[(int) RID_CONST] = get_identifier ("const");
                    352:   ridpointers[(int) RID_VOLATILE] = get_identifier ("volatile");
                    353:   ridpointers[(int) RID_AUTO] = get_identifier ("auto");
                    354:   ridpointers[(int) RID_STATIC] = get_identifier ("static");
                    355:   ridpointers[(int) RID_EXTERN] = get_identifier ("extern");
                    356:   ridpointers[(int) RID_TYPEDEF] = get_identifier ("typedef");
                    357:   ridpointers[(int) RID_REGISTER] = get_identifier ("register");
                    358:   ridpointers[(int) RID_ITERATOR] = get_identifier ("iterator");
                    359:   ridpointers[(int) RID_COMPLEX] = get_identifier ("complex");
                    360:   ridpointers[(int) RID_ID] = get_identifier ("id");
                    361:   ridpointers[(int) RID_IN] = get_identifier ("in");
                    362:   ridpointers[(int) RID_OUT] = get_identifier ("out");
                    363:   ridpointers[(int) RID_INOUT] = get_identifier ("inout");
                    364:   ridpointers[(int) RID_BYCOPY] = get_identifier ("bycopy");
                    365:   ridpointers[(int) RID_ONEWAY] = get_identifier ("oneway");
                    366:   forget_protocol_qualifiers();
                    367: 
                    368:   /* Some options inhibit certain reserved words.
                    369:      Clear those words out of the hash table so they won't be recognized.  */
                    370: #define UNSET_RESERVED_WORD(STRING) \
                    371:   do { struct resword *s = is_reserved_word (STRING, sizeof (STRING) - 1); \
                    372:        if (s) s->name = ""; } while (0)
                    373: 
                    374:   if (! doing_objc_thang)
                    375:     UNSET_RESERVED_WORD ("id");
                    376: 
                    377:   if (flag_traditional)
                    378:     {
                    379:       UNSET_RESERVED_WORD ("const");
                    380:       UNSET_RESERVED_WORD ("volatile");
                    381:       UNSET_RESERVED_WORD ("typeof");
                    382:       UNSET_RESERVED_WORD ("signed");
                    383:       UNSET_RESERVED_WORD ("inline");
                    384:       UNSET_RESERVED_WORD ("iterator");
                    385:       UNSET_RESERVED_WORD ("complex");
                    386:     }
                    387:   if (flag_no_asm)
                    388:     {
                    389:       UNSET_RESERVED_WORD ("asm");
                    390:       UNSET_RESERVED_WORD ("typeof");
                    391:       UNSET_RESERVED_WORD ("inline");
                    392:       UNSET_RESERVED_WORD ("iterator");
                    393:       UNSET_RESERVED_WORD ("complex");
                    394:     }
                    395: }
                    396: 
                    397: void
                    398: reinit_parse_for_function ()
                    399: {
                    400: }
                    401: 
                    402: /* Function used when yydebug is set, to print a token in more detail.  */
                    403: 
                    404: void
                    405: yyprint (file, yychar, yylval)
                    406:      FILE *file;
                    407:      int yychar;
                    408:      YYSTYPE yylval;
                    409: {
                    410:   tree t;
                    411:   switch (yychar)
                    412:     {
                    413:     case IDENTIFIER:
                    414:     case TYPENAME:
                    415:     case OBJECTNAME:
                    416:       t = yylval.ttype;
                    417:       if (IDENTIFIER_POINTER (t))
                    418:        fprintf (file, " `%s'", IDENTIFIER_POINTER (t));
                    419:       break;
                    420: 
                    421:     case CONSTANT:
                    422:       t = yylval.ttype;
                    423:       if (TREE_CODE (t) == INTEGER_CST)
                    424:        fprintf (file,
                    425: #if HOST_BITS_PER_WIDE_INT == 64
                    426: #if HOST_BITS_PER_WIDE_INT != HOST_BITS_PER_INT
                    427:                 " 0x%lx%016lx",
                    428: #else
                    429:                 " 0x%x%016x",
                    430: #endif
                    431: #else
                    432: #if HOST_BITS_PER_WIDE_INT != HOST_BITS_PER_INT
                    433:                 " 0x%lx%08lx",
                    434: #else
                    435:                 " 0x%x%08x",
                    436: #endif
                    437: #endif
                    438:                 TREE_INT_CST_HIGH (t), TREE_INT_CST_LOW (t));
                    439:       break;
                    440:     }
                    441: }
                    442: 
                    443: 
                    444: /* If C is not whitespace, return C.
                    445:    Otherwise skip whitespace and return first nonwhite char read.  */
                    446: 
                    447: static int
                    448: skip_white_space (c)
                    449:      register int c;
                    450: {
                    451:   static int newline_warning = 0;
                    452: 
                    453:   for (;;)
                    454:     {
                    455:       switch (c)
                    456:        {
                    457:          /* We don't recognize comments here, because
                    458:             cpp output can include / and * consecutively as operators.
                    459:             Also, there's no need, since cpp removes all comments.  */
                    460: 
                    461:        case '\n':
                    462:          c = check_newline ();
                    463:          break;
                    464: 
                    465:        case ' ':
                    466:        case '\t':
                    467:        case '\f':
                    468:        case '\v':
                    469:        case '\b':
                    470:          c = getc (finput);
                    471:          break;
                    472: 
                    473:        case '\r':
                    474:          /* ANSI C says the effects of a carriage return in a source file
                    475:             are undefined.  */
                    476:          if (pedantic && !newline_warning)
                    477:            {
                    478:              warning ("carriage return in source file");
                    479:              warning ("(we only warn about the first carriage return)");
                    480:              newline_warning = 1;
                    481:            }
                    482:          c = getc (finput);
                    483:          break;
                    484: 
                    485:        case '\\':
                    486:          c = getc (finput);
                    487:          if (c == '\n')
                    488:            lineno++;
                    489:          else
                    490:            error ("stray '\\' in program");
                    491:          c = getc (finput);
                    492:          break;
                    493: 
                    494:        default:
                    495:          return (c);
                    496:        }
                    497:     }
                    498: }
                    499: 
                    500: /* Skips all of the white space at the current location in the input file.
                    501:    Must use and reset nextchar if it has the next character.  */
                    502: 
                    503: void
                    504: position_after_white_space ()
                    505: {
                    506:   register int c;
                    507: 
                    508:   if (nextchar != -1)
                    509:     c = nextchar, nextchar = -1;
                    510:   else
                    511:     c = getc (finput);
                    512: 
                    513:   ungetc (skip_white_space (c), finput);
                    514: }
                    515: 
                    516: /* Make the token buffer longer, preserving the data in it.
                    517:    P should point to just beyond the last valid character in the old buffer.
                    518:    The value we return is a pointer to the new buffer
                    519:    at a place corresponding to P.  */
                    520: 
                    521: static char *
                    522: extend_token_buffer (p)
                    523:      char *p;
                    524: {
                    525:   int offset = p - token_buffer;
                    526: 
                    527:   maxtoken = maxtoken * 2 + 10;
                    528:   token_buffer = (char *) xrealloc (token_buffer, maxtoken + 2);
                    529: 
                    530:   return token_buffer + offset;
                    531: }
                    532: 
                    533: /* At the beginning of a line, increment the line number
                    534:    and process any #-directive on this line.
                    535:    If the line is a #-directive, read the entire line and return a newline.
                    536:    Otherwise, return the line's first non-whitespace character.  */
                    537: 
                    538: int
                    539: check_newline ()
                    540: {
                    541:   register int c;
                    542:   register int token;
                    543: 
                    544:   lineno++;
                    545: 
                    546:   /* Read first nonwhite char on the line.  */
                    547: 
                    548:   c = getc (finput);
                    549:   while (c == ' ' || c == '\t')
                    550:     c = getc (finput);
                    551: 
                    552:   if (c != '#')
                    553:     {
                    554:       /* If not #, return it so caller will use it.  */
                    555:       return c;
                    556:     }
                    557: 
                    558:   /* Read first nonwhite char after the `#'.  */
                    559: 
                    560:   c = getc (finput);
                    561:   while (c == ' ' || c == '\t')
                    562:     c = getc (finput);
                    563: 
                    564:   /* If a letter follows, then if the word here is `line', skip
                    565:      it and ignore it; otherwise, ignore the line, with an error
                    566:      if the word isn't `pragma', `ident', `define', or `undef'.  */
                    567: 
                    568:   if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z'))
                    569:     {
                    570:       if (c == 'p')
                    571:        {
                    572:          if (getc (finput) == 'r'
                    573:              && getc (finput) == 'a'
                    574:              && getc (finput) == 'g'
                    575:              && getc (finput) == 'm'
                    576:              && getc (finput) == 'a'
                    577:              && ((c = getc (finput)) == ' ' || c == '\t' || c == '\n'))
                    578:            {
                    579: #ifdef HANDLE_SYSV_PRAGMA
                    580:              return handle_sysv_pragma (finput, c);
                    581: #endif /* HANDLE_SYSV_PRAGMA */
                    582: #ifdef HANDLE_PRAGMA
                    583:              HANDLE_PRAGMA (finput);
                    584: #endif /* HANDLE_PRAGMA */
                    585:              goto skipline;
                    586:            }
                    587:        }
                    588: 
                    589:       else if (c == 'd')
                    590:        {
                    591:          if (getc (finput) == 'e'
                    592:              && getc (finput) == 'f'
                    593:              && getc (finput) == 'i'
                    594:              && getc (finput) == 'n'
                    595:              && getc (finput) == 'e'
                    596:              && ((c = getc (finput)) == ' ' || c == '\t' || c == '\n'))
                    597:            {
                    598: #ifdef DWARF_DEBUGGING_INFO
                    599:              if ((debug_info_level == DINFO_LEVEL_VERBOSE)
                    600:                  && (write_symbols == DWARF_DEBUG))
                    601:                dwarfout_define (lineno, get_directive_line (finput));
                    602: #endif /* DWARF_DEBUGGING_INFO */
                    603:              goto skipline;
                    604:            }
                    605:        }
                    606:       else if (c == 'u')
                    607:        {
                    608:          if (getc (finput) == 'n'
                    609:              && getc (finput) == 'd'
                    610:              && getc (finput) == 'e'
                    611:              && getc (finput) == 'f'
                    612:              && ((c = getc (finput)) == ' ' || c == '\t' || c == '\n'))
                    613:            {
                    614: #ifdef DWARF_DEBUGGING_INFO
                    615:              if ((debug_info_level == DINFO_LEVEL_VERBOSE)
                    616:                  && (write_symbols == DWARF_DEBUG))
                    617:                dwarfout_undef (lineno, get_directive_line (finput));
                    618: #endif /* DWARF_DEBUGGING_INFO */
                    619:              goto skipline;
                    620:            }
                    621:        }
                    622:       else if (c == 'l')
                    623:        {
                    624:          if (getc (finput) == 'i'
                    625:              && getc (finput) == 'n'
                    626:              && getc (finput) == 'e'
                    627:              && ((c = getc (finput)) == ' ' || c == '\t'))
                    628:            goto linenum;
                    629:        }
                    630:       else if (c == 'i')
                    631:        {
                    632:          if (getc (finput) == 'd'
                    633:              && getc (finput) == 'e'
                    634:              && getc (finput) == 'n'
                    635:              && getc (finput) == 't'
                    636:              && ((c = getc (finput)) == ' ' || c == '\t'))
                    637:            {
                    638:              /* #ident.  The pedantic warning is now in cccp.c.  */
                    639: 
                    640:              /* Here we have just seen `#ident '.
                    641:                 A string constant should follow.  */
                    642: 
                    643:              while (c == ' ' || c == '\t')
                    644:                c = getc (finput);
                    645: 
                    646:              /* If no argument, ignore the line.  */
                    647:              if (c == '\n')
                    648:                return c;
                    649: 
                    650:              ungetc (c, finput);
                    651:              token = yylex ();
                    652:              if (token != STRING
                    653:                  || TREE_CODE (yylval.ttype) != STRING_CST)
                    654:                {
                    655:                  error ("invalid #ident");
                    656:                  goto skipline;
                    657:                }
                    658: 
                    659:              if (!flag_no_ident)
                    660:                {
                    661: #ifdef ASM_OUTPUT_IDENT
                    662:                  ASM_OUTPUT_IDENT (asm_out_file, TREE_STRING_POINTER (yylval.ttype));
                    663: #endif
                    664:                }
                    665: 
                    666:              /* Skip the rest of this line.  */
                    667:              goto skipline;
                    668:            }
                    669:        }
                    670: 
                    671:       error ("undefined or invalid # directive");
                    672:       goto skipline;
                    673:     }
                    674: 
                    675: linenum:
                    676:   /* Here we have either `#line' or `# <nonletter>'.
                    677:      In either case, it should be a line number; a digit should follow.  */
                    678: 
                    679:   while (c == ' ' || c == '\t')
                    680:     c = getc (finput);
                    681: 
                    682:   /* If the # is the only nonwhite char on the line,
                    683:      just ignore it.  Check the new newline.  */
                    684:   if (c == '\n')
                    685:     return c;
                    686: 
                    687:   /* Something follows the #; read a token.  */
                    688: 
                    689:   ungetc (c, finput);
                    690:   token = yylex ();
                    691: 
                    692:   if (token == CONSTANT
                    693:       && TREE_CODE (yylval.ttype) == INTEGER_CST)
                    694:     {
                    695:       int old_lineno = lineno;
                    696:       int used_up = 0;
                    697:       /* subtract one, because it is the following line that
                    698:         gets the specified number */
                    699: 
                    700:       int l = TREE_INT_CST_LOW (yylval.ttype) - 1;
                    701: 
                    702:       /* Is this the last nonwhite stuff on the line?  */
                    703:       c = getc (finput);
                    704:       while (c == ' ' || c == '\t')
                    705:        c = getc (finput);
                    706:       if (c == '\n')
                    707:        {
                    708:          /* No more: store the line number and check following line.  */
                    709:          lineno = l;
                    710:          return c;
                    711:        }
                    712:       ungetc (c, finput);
                    713: 
                    714:       /* More follows: it must be a string constant (filename).  */
                    715: 
                    716:       /* Read the string constant.  */
                    717:       token = yylex ();
                    718: 
                    719:       if (token != STRING || TREE_CODE (yylval.ttype) != STRING_CST)
                    720:        {
                    721:          error ("invalid #line");
                    722:          goto skipline;
                    723:        }
                    724: 
                    725:       input_filename
                    726:        = (char *) permalloc (TREE_STRING_LENGTH (yylval.ttype) + 1);
                    727:       strcpy (input_filename, TREE_STRING_POINTER (yylval.ttype));
                    728:       lineno = l;
                    729: 
                    730:       /* Each change of file name
                    731:         reinitializes whether we are now in a system header.  */
                    732:       in_system_header = 0;
                    733: 
                    734:       if (main_input_filename == 0)
                    735:        main_input_filename = input_filename;
                    736: 
                    737:       /* Is this the last nonwhite stuff on the line?  */
                    738:       c = getc (finput);
                    739:       while (c == ' ' || c == '\t')
                    740:        c = getc (finput);
                    741:       if (c == '\n')
                    742:        {
                    743:          /* Update the name in the top element of input_file_stack.  */
                    744:          if (input_file_stack)
                    745:            input_file_stack->name = input_filename;
                    746: 
                    747:          return c;
                    748:        }
                    749:       ungetc (c, finput);
                    750: 
                    751:       token = yylex ();
                    752:       used_up = 0;
                    753: 
                    754:       /* `1' after file name means entering new file.
                    755:         `2' after file name means just left a file.  */
                    756: 
                    757:       if (token == CONSTANT
                    758:          && TREE_CODE (yylval.ttype) == INTEGER_CST)
                    759:        {
                    760:          if (TREE_INT_CST_LOW (yylval.ttype) == 1)
                    761:            {
                    762:              /* Pushing to a new file.  */
                    763:              struct file_stack *p
                    764:                = (struct file_stack *) xmalloc (sizeof (struct file_stack));
                    765:              input_file_stack->line = old_lineno;
                    766:              p->next = input_file_stack;
                    767:              p->name = input_filename;
                    768:              input_file_stack = p;
                    769:              input_file_stack_tick++;
                    770: #ifdef DWARF_DEBUGGING_INFO
                    771:              if (debug_info_level == DINFO_LEVEL_VERBOSE
                    772:                  && write_symbols == DWARF_DEBUG)
                    773:                dwarfout_start_new_source_file (input_filename);
                    774: #endif /* DWARF_DEBUGGING_INFO */
                    775: 
                    776:              used_up = 1;
                    777:            }
                    778:          else if (TREE_INT_CST_LOW (yylval.ttype) == 2)
                    779:            {
                    780:              /* Popping out of a file.  */
                    781:              if (input_file_stack->next)
                    782:                {
                    783:                  struct file_stack *p = input_file_stack;
                    784:                  input_file_stack = p->next;
                    785:                  free (p);
                    786:                  input_file_stack_tick++;
                    787: #ifdef DWARF_DEBUGGING_INFO
                    788:                  if (debug_info_level == DINFO_LEVEL_VERBOSE
                    789:                      && write_symbols == DWARF_DEBUG)
                    790:                    dwarfout_resume_previous_source_file (input_file_stack->line);
                    791: #endif /* DWARF_DEBUGGING_INFO */
                    792:                }
                    793:              else
                    794:                error ("#-lines for entering and leaving files don't match");
                    795: 
                    796:              used_up = 1;
                    797:            }
                    798:        }
                    799: 
                    800:       /* Now that we've pushed or popped the input stack,
                    801:         update the name in the top element.  */
                    802:       if (input_file_stack)
                    803:        input_file_stack->name = input_filename;
                    804: 
                    805:       /* If we have handled a `1' or a `2',
                    806:         see if there is another number to read.  */
                    807:       if (used_up)
                    808:        {
                    809:          /* Is this the last nonwhite stuff on the line?  */
                    810:          c = getc (finput);
                    811:          while (c == ' ' || c == '\t')
                    812:            c = getc (finput);
                    813:          if (c == '\n')
                    814:            return c;
                    815:          ungetc (c, finput);
                    816: 
                    817:          token = yylex ();
                    818:          used_up = 0;
                    819:        }
                    820: 
                    821:       /* `3' after file name means this is a system header file.  */
                    822: 
                    823:       if (token == CONSTANT
                    824:          && TREE_CODE (yylval.ttype) == INTEGER_CST
                    825:          && TREE_INT_CST_LOW (yylval.ttype) == 3)
                    826:        in_system_header = 1;
                    827:     }
                    828:   else
                    829:     error ("invalid #-line");
                    830: 
                    831:   /* skip the rest of this line.  */
                    832:  skipline:
                    833:   if (c == '\n')
                    834:     return c;
                    835:   while ((c = getc (finput)) != EOF && c != '\n');
                    836:   return c;
                    837: }
                    838: 
                    839: #ifdef HANDLE_SYSV_PRAGMA
                    840: 
                    841: /* Handle a #pragma directive.  INPUT is the current input stream,
                    842:    and C is a character to reread.  Processes the entire input line
                    843:    and returns a character for the caller to reread: either \n or EOF.  */
                    844: 
                    845: /* This function has to be in this file, in order to get at
                    846:    the token types.  */
                    847: 
                    848: int
                    849: handle_sysv_pragma (input, c)
                    850:      FILE *input;
                    851:      int c;
                    852: {
                    853:   for (;;)
                    854:     {
                    855:       while (c == ' ' || c == '\t')
                    856:        c = getc (input);
                    857:       if (c == '\n' || c == EOF)
                    858:        {
                    859:          handle_pragma_token (0, 0);
                    860:          return c;
                    861:        }
                    862:       ungetc (c, input);
                    863:       switch (yylex ())
                    864:        {
                    865:        case IDENTIFIER:
                    866:        case TYPENAME:
                    867:        case STRING:
                    868:        case CONSTANT:
                    869:          handle_pragma_token (token_buffer, yylval.ttype);
                    870:          break;
                    871:        default:
                    872:          handle_pragma_token (token_buffer, 0);
                    873:        }
                    874:       if (nextchar >= 0)
                    875:        c = nextchar, nextchar = -1;
                    876:       else
                    877:        c = getc (input);
                    878:     }
                    879: }
                    880: 
                    881: #endif /* HANDLE_SYSV_PRAGMA */
                    882: 
                    883: #define ENDFILE -1  /* token that represents end-of-file */
                    884: 
                    885: /* Read an escape sequence, returning its equivalent as a character,
                    886:    or store 1 in *ignore_ptr if it is backslash-newline.  */
                    887: 
                    888: static int
                    889: readescape (ignore_ptr)
                    890:      int *ignore_ptr;
                    891: {
                    892:   register int c = getc (finput);
                    893:   register int code;
                    894:   register unsigned count;
                    895:   unsigned firstdig;
                    896:   int nonnull;
                    897: 
                    898:   switch (c)
                    899:     {
                    900:     case 'x':
                    901:       if (warn_traditional)
                    902:        warning ("the meaning of `\\x' varies with -traditional");
                    903: 
                    904:       if (flag_traditional)
                    905:        return c;
                    906: 
                    907:       code = 0;
                    908:       count = 0;
                    909:       nonnull = 0;
                    910:       while (1)
                    911:        {
                    912:          c = getc (finput);
                    913:          if (!(c >= 'a' && c <= 'f')
                    914:              && !(c >= 'A' && c <= 'F')
                    915:              && !(c >= '0' && c <= '9'))
                    916:            {
                    917:              ungetc (c, finput);
                    918:              break;
                    919:            }
                    920:          code *= 16;
                    921:          if (c >= 'a' && c <= 'f')
                    922:            code += c - 'a' + 10;
                    923:          if (c >= 'A' && c <= 'F')
                    924:            code += c - 'A' + 10;
                    925:          if (c >= '0' && c <= '9')
                    926:            code += c - '0';
                    927:          if (code != 0 || count != 0)
                    928:            {
                    929:              if (count == 0)
                    930:                firstdig = code;
                    931:              count++;
                    932:            }
                    933:          nonnull = 1;
                    934:        }
                    935:       if (! nonnull)
                    936:        error ("\\x used with no following hex digits");
                    937:       else if (count == 0)
                    938:        /* Digits are all 0's.  Ok.  */
                    939:        ;
                    940:       else if ((count - 1) * 4 >= TYPE_PRECISION (integer_type_node)
                    941:               || (count > 1
                    942:                   && ((1 << (TYPE_PRECISION (integer_type_node) - (count - 1) * 4))
                    943:                       <= firstdig)))
                    944:        pedwarn ("hex escape out of range");
                    945:       return code;
                    946: 
                    947:     case '0':  case '1':  case '2':  case '3':  case '4':
                    948:     case '5':  case '6':  case '7':
                    949:       code = 0;
                    950:       count = 0;
                    951:       while ((c <= '7') && (c >= '0') && (count++ < 3))
                    952:        {
                    953:          code = (code * 8) + (c - '0');
                    954:          c = getc (finput);
                    955:        }
                    956:       ungetc (c, finput);
                    957:       return code;
                    958: 
                    959:     case '\\': case '\'': case '"':
                    960:       return c;
                    961: 
                    962:     case '\n':
                    963:       lineno++;
                    964:       *ignore_ptr = 1;
                    965:       return 0;
                    966: 
                    967:     case 'n':
                    968:       return TARGET_NEWLINE;
                    969: 
                    970:     case 't':
                    971:       return TARGET_TAB;
                    972: 
                    973:     case 'r':
                    974:       return TARGET_CR;
                    975: 
                    976:     case 'f':
                    977:       return TARGET_FF;
                    978: 
                    979:     case 'b':
                    980:       return TARGET_BS;
                    981: 
                    982:     case 'a':
                    983:       if (warn_traditional)
                    984:        warning ("the meaning of `\\a' varies with -traditional");
                    985: 
                    986:       if (flag_traditional)
                    987:        return c;
                    988:       return TARGET_BELL;
                    989: 
                    990:     case 'v':
                    991: #if 0 /* Vertical tab is present in common usage compilers.  */
                    992:       if (flag_traditional)
                    993:        return c;
                    994: #endif
                    995:       return TARGET_VT;
                    996: 
                    997:     case 'e':
                    998:     case 'E':
                    999:       if (pedantic)
                   1000:        pedwarn ("non-ANSI-standard escape sequence, `\\%c'", c);
                   1001:       return 033;
                   1002: 
                   1003:     case '?':
                   1004:       return c;
                   1005: 
                   1006:       /* `\(', etc, are used at beginning of line to avoid confusing Emacs.  */
                   1007:     case '(':
                   1008:     case '{':
                   1009:     case '[':
                   1010:       /* `\%' is used to prevent SCCS from getting confused.  */
                   1011:     case '%':
                   1012:       if (pedantic)
                   1013:        pedwarn ("non-ANSI escape sequence `\\%c'", c);
                   1014:       return c;
                   1015:     }
                   1016:   if (c >= 040 && c < 0177)
                   1017:     pedwarn ("unknown escape sequence `\\%c'", c);
                   1018:   else
                   1019:     pedwarn ("unknown escape sequence: `\\' followed by char code 0x%x", c);
                   1020:   return c;
                   1021: }
                   1022: 
                   1023: void
                   1024: yyerror (string)
                   1025:      char *string;
                   1026: {
                   1027:   char buf[200];
                   1028: 
                   1029:   strcpy (buf, string);
                   1030: 
                   1031:   /* We can't print string and character constants well
                   1032:      because the token_buffer contains the result of processing escapes.  */
                   1033:   if (end_of_file)
                   1034:     strcat (buf, " at end of input");
                   1035:   else if (token_buffer[0] == 0)
                   1036:     strcat (buf, " at null character");
                   1037:   else if (token_buffer[0] == '"')
                   1038:     strcat (buf, " before string constant");
                   1039:   else if (token_buffer[0] == '\'')
                   1040:     strcat (buf, " before character constant");
                   1041:   else if (token_buffer[0] < 040 || (unsigned char) token_buffer[0] >= 0177)
                   1042:     sprintf (buf + strlen (buf), " before character 0%o",
                   1043:             (unsigned char) token_buffer[0]);
                   1044:   else
                   1045:     strcat (buf, " before `%s'");
                   1046: 
                   1047:   error (buf, token_buffer);
                   1048: }
                   1049: 
                   1050: #if 0
                   1051: 
                   1052: struct try_type
                   1053: {
                   1054:   tree *node_var;
                   1055:   char unsigned_flag;
                   1056:   char long_flag;
                   1057:   char long_long_flag;
                   1058: };
                   1059: 
                   1060: struct try_type type_sequence[] = 
                   1061: {
                   1062:   { &integer_type_node, 0, 0, 0},
                   1063:   { &unsigned_type_node, 1, 0, 0},
                   1064:   { &long_integer_type_node, 0, 1, 0},
                   1065:   { &long_unsigned_type_node, 1, 1, 0},
                   1066:   { &long_long_integer_type_node, 0, 1, 1},
                   1067:   { &long_long_unsigned_type_node, 1, 1, 1}
                   1068: };
                   1069: #endif /* 0 */
                   1070: 
                   1071: int
                   1072: yylex ()
                   1073: {
                   1074:   register int c;
                   1075:   register char *p;
                   1076:   register int value;
                   1077:   int wide_flag = 0;
                   1078:   int objc_flag = 0;
                   1079: 
                   1080:   if (nextchar >= 0)
                   1081:     c = nextchar, nextchar = -1;
                   1082:   else
                   1083:     c = getc (finput);
                   1084: 
                   1085:   /* Effectively do c = skip_white_space (c)
                   1086:      but do it faster in the usual cases.  */
                   1087:   while (1)
                   1088:     switch (c)
                   1089:       {
                   1090:       case ' ':
                   1091:       case '\t':
                   1092:       case '\f':
                   1093:       case '\v':
                   1094:       case '\b':
                   1095:        c = getc (finput);
                   1096:        break;
                   1097: 
                   1098:       case '\r':
                   1099:        /* Call skip_white_space so we can warn if appropriate.  */
                   1100: 
                   1101:       case '\n':
                   1102:       case '/':
                   1103:       case '\\':
                   1104:        c = skip_white_space (c);
                   1105:       default:
                   1106:        goto found_nonwhite;
                   1107:       }
                   1108:  found_nonwhite:
                   1109: 
                   1110:   token_buffer[0] = c;
                   1111:   token_buffer[1] = 0;
                   1112: 
                   1113: /*  yylloc.first_line = lineno; */
                   1114: 
                   1115:   switch (c)
                   1116:     {
                   1117:     case EOF:
                   1118:       end_of_file = 1;
                   1119:       token_buffer[0] = 0;
                   1120:       value = ENDFILE;
                   1121:       break;
                   1122: 
                   1123:     case '$':
                   1124:       if (dollars_in_ident)
                   1125:        goto letter;
                   1126:       return '$';
                   1127: 
                   1128:     case 'L':
                   1129:       /* Capital L may start a wide-string or wide-character constant.  */
                   1130:       {
                   1131:        register int c = getc (finput);
                   1132:        if (c == '\'')
                   1133:          {
                   1134:            wide_flag = 1;
                   1135:            goto char_constant;
                   1136:          }
                   1137:        if (c == '"')
                   1138:          {
                   1139:            wide_flag = 1;
                   1140:            goto string_constant;
                   1141:          }
                   1142:        ungetc (c, finput);
                   1143:       }
                   1144:       goto letter;
                   1145: 
                   1146:     case '@':
                   1147:       if (!doing_objc_thang)
                   1148:        {
                   1149:          warning ("possible Objective-C token in C input.  Use -ObjC");
                   1150:          value = c;
                   1151:          break;
                   1152:        }
                   1153:       else
                   1154:        {
                   1155:          /* '@' may start a constant string object.  */
                   1156:          register int c = getc(finput);
                   1157:          if (c == '"')
                   1158:            {
                   1159:              objc_flag = 1;
                   1160:              goto string_constant;
                   1161:            }
                   1162:          ungetc(c, finput);
                   1163:          /* Fall through to treat '@' as the start of an indentifier.  */
                   1164:        }
                   1165: 
                   1166:     case 'A':  case 'B':  case 'C':  case 'D':  case 'E':
                   1167:     case 'F':  case 'G':  case 'H':  case 'I':  case 'J':
                   1168:     case 'K':            case 'M':  case 'N':  case 'O':
                   1169:     case 'P':  case 'Q':  case 'R':  case 'S':  case 'T':
                   1170:     case 'U':  case 'V':  case 'W':  case 'X':  case 'Y':
                   1171:     case 'Z':
                   1172:     case 'a':  case 'b':  case 'c':  case 'd':  case 'e':
                   1173:     case 'f':  case 'g':  case 'h':  case 'i':  case 'j':
                   1174:     case 'k':  case 'l':  case 'm':  case 'n':  case 'o':
                   1175:     case 'p':  case 'q':  case 'r':  case 's':  case 't':
                   1176:     case 'u':  case 'v':  case 'w':  case 'x':  case 'y':
                   1177:     case 'z':
                   1178:     case '_':
                   1179:     letter:
                   1180:       p = token_buffer;
                   1181:       while (isalnum (c) || c == '_' || c == '$' || c == '@')
                   1182:        {
                   1183:          /* Make sure this char really belongs in an identifier.  */
                   1184:          if (c == '@' && ! doing_objc_thang)
                   1185:            break;
                   1186:          if (c == '$' && ! dollars_in_ident)
                   1187:            break;
                   1188: 
                   1189:          if (p >= token_buffer + maxtoken)
                   1190:            p = extend_token_buffer (p);
                   1191: 
                   1192:          *p++ = c;
                   1193:          c = getc (finput);
                   1194:        }
                   1195: 
                   1196:       *p = 0;
                   1197:       nextchar = c;
                   1198: 
                   1199:       value = IDENTIFIER;
                   1200:       yylval.itype = 0;
                   1201: 
                   1202:       /* Try to recognize a keyword.  Uses minimum-perfect hash function */
                   1203: 
                   1204:       {
                   1205:        register struct resword *ptr;
                   1206: 
                   1207:        if (ptr = is_reserved_word (token_buffer, p - token_buffer))
                   1208:          {
                   1209:            if (ptr->rid)
                   1210:              yylval.ttype = ridpointers[(int) ptr->rid];
                   1211:            value = (int) ptr->token;
                   1212: 
                   1213:            /* Only return OBJECTNAME if it is a typedef.  */
                   1214:            if (doing_objc_thang && value == OBJECTNAME)
                   1215:              {
                   1216:                lastiddecl = lookup_name(yylval.ttype);
                   1217: 
                   1218:                if (lastiddecl == NULL_TREE
                   1219:                    || TREE_CODE (lastiddecl) != TYPE_DECL)
                   1220:                  value = IDENTIFIER;
                   1221:              }
                   1222: 
                   1223:            /* Even if we decided to recognize asm, still perhaps warn.  */
                   1224:            if (pedantic
                   1225:                && (value == ASM_KEYWORD || value == TYPEOF
                   1226:                    || ptr->rid == RID_INLINE)
                   1227:                && token_buffer[0] != '_')
                   1228:              pedwarn ("ANSI does not permit the keyword `%s'",
                   1229:                       token_buffer);
                   1230:          }
                   1231:       }
                   1232: 
                   1233:       /* If we did not find a keyword, look for an identifier
                   1234:         (or a typename).  */
                   1235: 
                   1236:       if (value == IDENTIFIER)
                   1237:        {
                   1238:          if (token_buffer[0] == '@')
                   1239:            error("invalid identifier `%s'", token_buffer);
                   1240: 
                   1241:           yylval.ttype = get_identifier (token_buffer);
                   1242:          lastiddecl = lookup_name (yylval.ttype);
                   1243: 
                   1244:          if (lastiddecl != 0 && TREE_CODE (lastiddecl) == TYPE_DECL)
                   1245:            value = TYPENAME;
                   1246:          /* A user-invisible read-only initialized variable
                   1247:             should be replaced by its value.
                   1248:             We handle only strings since that's the only case used in C.  */
                   1249:          else if (lastiddecl != 0 && TREE_CODE (lastiddecl) == VAR_DECL
                   1250:                   && DECL_IGNORED_P (lastiddecl)
                   1251:                   && TREE_READONLY (lastiddecl)
                   1252:                   && DECL_INITIAL (lastiddecl) != 0
                   1253:                   && TREE_CODE (DECL_INITIAL (lastiddecl)) == STRING_CST)
                   1254:            {
                   1255:              tree stringval = DECL_INITIAL (lastiddecl);
                   1256:              
                   1257:              /* Copy the string value so that we won't clobber anything
                   1258:                 if we put something in the TREE_CHAIN of this one.  */
                   1259:              yylval.ttype = build_string (TREE_STRING_LENGTH (stringval),
                   1260:                                           TREE_STRING_POINTER (stringval));
                   1261:              value = STRING;
                   1262:            }
                   1263:           else if (doing_objc_thang)
                   1264:             {
                   1265:              tree objc_interface_decl = is_class_name (yylval.ttype);
                   1266: 
                   1267:              if (objc_interface_decl)
                   1268:                {
                   1269:                  value = CLASSNAME;
                   1270:                  yylval.ttype = objc_interface_decl;
                   1271:                }
                   1272:            }
                   1273:        }
                   1274: 
                   1275:       break;
                   1276: 
                   1277:     case '0':  case '1':  case '2':  case '3':  case '4':
                   1278:     case '5':  case '6':  case '7':  case '8':  case '9':
                   1279:     case '.':
                   1280:       {
                   1281:        int base = 10;
                   1282:        int count = 0;
                   1283:        int largest_digit = 0;
                   1284:        int numdigits = 0;
                   1285:        /* for multi-precision arithmetic,
                   1286:           we actually store only HOST_BITS_PER_CHAR bits in each part.
                   1287:           The number of parts is chosen so as to be sufficient to hold
                   1288:           the enough bits to fit into the two HOST_WIDE_INTs that contain
                   1289:           the integer value (this is always at least as many bits as are
                   1290:           in a target `long long' value, but may be wider).  */
                   1291: #define TOTAL_PARTS ((HOST_BITS_PER_WIDE_INT / HOST_BITS_PER_CHAR) * 2 + 2)
                   1292:        int parts[TOTAL_PARTS];
                   1293:        int overflow = 0;
                   1294: 
                   1295:        enum anon1 { NOT_FLOAT, AFTER_POINT, TOO_MANY_POINTS} floatflag
                   1296:          = NOT_FLOAT;
                   1297: 
                   1298:        for (count = 0; count < TOTAL_PARTS; count++)
                   1299:          parts[count] = 0;
                   1300: 
                   1301:        p = token_buffer;
                   1302:        *p++ = c;
                   1303: 
                   1304:        if (c == '0')
                   1305:          {
                   1306:            *p++ = (c = getc (finput));
                   1307:            if ((c == 'x') || (c == 'X'))
                   1308:              {
                   1309:                base = 16;
                   1310:                *p++ = (c = getc (finput));
                   1311:              }
                   1312:            /* Leading 0 forces octal unless the 0 is the only digit.  */
                   1313:            else if (c >= '0' && c <= '9')
                   1314:              {
                   1315:                base = 8;
                   1316:                numdigits++;
                   1317:              }
                   1318:            else
                   1319:              numdigits++;
                   1320:          }
                   1321: 
                   1322:        /* Read all the digits-and-decimal-points.  */
                   1323: 
                   1324:        while (c == '.'
                   1325:               || (isalnum (c) && c != 'l' && c != 'L'
                   1326:                   && c != 'u' && c != 'U'
                   1327:                   && c != 'i' && c != 'I' && c != 'j' && c != 'J'
                   1328:                   && (floatflag == NOT_FLOAT || ((c != 'f') && (c != 'F')))))
                   1329:          {
                   1330:            if (c == '.')
                   1331:              {
                   1332:                if (base == 16)
                   1333:                  error ("floating constant may not be in radix 16");
                   1334:                if (floatflag == TOO_MANY_POINTS)
                   1335:                  /* We have already emitted an error.  Don't need another.  */
                   1336:                  ;
                   1337:                else if (floatflag == AFTER_POINT)
                   1338:                  {
                   1339:                    error ("malformed floating constant");
                   1340:                    floatflag = TOO_MANY_POINTS;
                   1341:                    /* Avoid another error from atof by forcing all characters
                   1342:                       from here on to be ignored.  */
                   1343:                    p[-1] = '\0';
                   1344:                  }
                   1345:                else
                   1346:                  floatflag = AFTER_POINT;
                   1347: 
                   1348:                base = 10;
                   1349:                *p++ = c = getc (finput);
                   1350:                /* Accept '.' as the start of a floating-point number
                   1351:                   only when it is followed by a digit.
                   1352:                   Otherwise, unread the following non-digit
                   1353:                   and use the '.' as a structural token.  */
                   1354:                if (p == token_buffer + 2 && !isdigit (c))
                   1355:                  {
                   1356:                    if (c == '.')
                   1357:                      {
                   1358:                        c = getc (finput);
                   1359:                        if (c == '.')
                   1360:                          {
                   1361:                            *p++ = c;
                   1362:                            *p = 0;
                   1363:                            return ELLIPSIS;
                   1364:                          }
                   1365:                        error ("parse error at `..'");
                   1366:                      }
                   1367:                    ungetc (c, finput);
                   1368:                    token_buffer[1] = 0;
                   1369:                    value = '.';
                   1370:                    goto done;
                   1371:                  }
                   1372:              }
                   1373:            else
                   1374:              {
                   1375:                /* It is not a decimal point.
                   1376:                   It should be a digit (perhaps a hex digit).  */
                   1377: 
                   1378:                if (isdigit (c))
                   1379:                  {
                   1380:                    c = c - '0';
                   1381:                  }
                   1382:                else if (base <= 10)
                   1383:                  {
                   1384:                    if (c == 'e' || c == 'E')
                   1385:                      {
                   1386:                        base = 10;
                   1387:                        floatflag = AFTER_POINT;
                   1388:                        break;   /* start of exponent */
                   1389:                      }
                   1390:                    error ("nondigits in number and not hexadecimal");
                   1391:                    c = 0;
                   1392:                  }
                   1393:                else if (c >= 'a')
                   1394:                  {
                   1395:                    c = c - 'a' + 10;
                   1396:                  }
                   1397:                else
                   1398:                  {
                   1399:                    c = c - 'A' + 10;
                   1400:                  }
                   1401:                if (c >= largest_digit)
                   1402:                  largest_digit = c;
                   1403:                numdigits++;
                   1404: 
                   1405:                for (count = 0; count < TOTAL_PARTS; count++)
                   1406:                  {
                   1407:                    parts[count] *= base;
                   1408:                    if (count)
                   1409:                      {
                   1410:                        parts[count]
                   1411:                          += (parts[count-1] >> HOST_BITS_PER_CHAR);
                   1412:                        parts[count-1]
                   1413:                          &= (1 << HOST_BITS_PER_CHAR) - 1;
                   1414:                      }
                   1415:                    else
                   1416:                      parts[0] += c;
                   1417:                  }
                   1418: 
                   1419:                /* If the extra highest-order part ever gets anything in it,
                   1420:                   the number is certainly too big.  */
                   1421:                if (parts[TOTAL_PARTS - 1] != 0)
                   1422:                  overflow = 1;
                   1423: 
                   1424:                if (p >= token_buffer + maxtoken - 3)
                   1425:                  p = extend_token_buffer (p);
                   1426:                *p++ = (c = getc (finput));
                   1427:              }
                   1428:          }
                   1429: 
                   1430:        if (numdigits == 0)
                   1431:          error ("numeric constant with no digits");
                   1432: 
                   1433:        if (largest_digit >= base)
                   1434:          error ("numeric constant contains digits beyond the radix");
                   1435: 
                   1436:        /* Remove terminating char from the token buffer and delimit the string */
                   1437:        *--p = 0;
                   1438: 
                   1439:        if (floatflag != NOT_FLOAT)
                   1440:          {
                   1441:            tree type = double_type_node;
                   1442:            int garbage_chars = 0, exceeds_double = 0;
                   1443:            int imag = 0;
                   1444:            REAL_VALUE_TYPE value;
                   1445:            jmp_buf handler;
                   1446: 
                   1447:            /* Read explicit exponent if any, and put it in tokenbuf.  */
                   1448: 
                   1449:            if ((c == 'e') || (c == 'E'))
                   1450:              {
                   1451:                if (p >= token_buffer + maxtoken - 3)
                   1452:                  p = extend_token_buffer (p);
                   1453:                *p++ = c;
                   1454:                c = getc (finput);
                   1455:                if ((c == '+') || (c == '-'))
                   1456:                  {
                   1457:                    *p++ = c;
                   1458:                    c = getc (finput);
                   1459:                  }
                   1460:                if (! isdigit (c))
                   1461:                  error ("floating constant exponent has no digits");
                   1462:                while (isdigit (c))
                   1463:                  {
                   1464:                    if (p >= token_buffer + maxtoken - 3)
                   1465:                      p = extend_token_buffer (p);
                   1466:                    *p++ = c;
                   1467:                    c = getc (finput);
                   1468:                  }
                   1469:              }
                   1470: 
                   1471:            *p = 0;
                   1472:            errno = 0;
                   1473: 
                   1474:            /* Convert string to a double, checking for overflow.  */
                   1475:            if (setjmp (handler))
                   1476:              {
                   1477:                error ("floating constant out of range");
                   1478:                value = dconst0;
                   1479:              }
                   1480:            else
                   1481:              {
                   1482:                int fflag = 0, lflag = 0;
                   1483:                /* Copy token_buffer now, while it has just the number
                   1484:                   and not the suffixes; once we add `f' or `i',
                   1485:                   REAL_VALUE_ATOF may not work any more.  */
                   1486:                char *copy = (char *) alloca (p - token_buffer + 1);
                   1487:                bcopy (token_buffer, copy, p - token_buffer + 1);
                   1488: 
                   1489:                set_float_handler (handler);
                   1490: 
                   1491:                while (1)
                   1492:                  {
                   1493:                    int lose = 0;
                   1494: 
                   1495:                    /* Read the suffixes to choose a data type.  */
                   1496:                    switch (c)
                   1497:                      {
                   1498:                      case 'f': case 'F':
                   1499:                        if (fflag)
                   1500:                          error ("more than one `f' in numeric constant");
                   1501:                        fflag = 1;
                   1502:                        break;
                   1503: 
                   1504:                      case 'l': case 'L':
                   1505:                        if (lflag)
                   1506:                          error ("more than one `l' in numeric constant");
                   1507:                        lflag = 1;
                   1508:                        break;
                   1509: 
                   1510:                      case 'i': case 'I':
                   1511:                        if (imag)
                   1512:                          error ("more than one `i' or `j' in numeric constant");
                   1513:                        imag = 1;
                   1514:                        break;
                   1515: 
                   1516:                      default:
                   1517:                        lose = 1;
                   1518:                      }
                   1519: 
                   1520:                    if (lose)
                   1521:                      break;
                   1522: 
                   1523:                    if (p >= token_buffer + maxtoken - 3)
                   1524:                      p = extend_token_buffer (p);
                   1525:                    *p++ = c;
                   1526:                    *p = 0;
                   1527:                    c = getc (finput);
                   1528:                  }
                   1529: 
                   1530:                /* The second argument, machine_mode, of REAL_VALUE_ATOF
                   1531:                   tells the desired precision of the binary result
                   1532:                   of decimal-to-binary conversion.  */
                   1533: 
                   1534:                if (fflag)
                   1535:                  {
                   1536:                    if (lflag)
                   1537:                      error ("both `f' and `l' in floating constant");
                   1538: 
                   1539:                    type = float_type_node;
                   1540:                    value = REAL_VALUE_ATOF (copy, TYPE_MODE (type));
                   1541:                    if (TARGET_FLOAT_FORMAT != IEEE_FLOAT_FORMAT
                   1542:                        && REAL_VALUE_ISINF (value) && pedantic)
                   1543:                      pedwarn ("floating point number exceeds range of `float'");
                   1544:                  }
                   1545:                else if (lflag)
                   1546:                  {
                   1547:                    type = long_double_type_node;
                   1548:                    value = REAL_VALUE_ATOF (copy, TYPE_MODE (type));
                   1549:                    if (TARGET_FLOAT_FORMAT != IEEE_FLOAT_FORMAT
                   1550:                        && REAL_VALUE_ISINF (value) && pedantic)
                   1551:                      pedwarn ("floating point number exceeds range of `long double'");
                   1552:                  }
                   1553:                else
                   1554:                  {
                   1555:                    value = REAL_VALUE_ATOF (copy, TYPE_MODE (type));
                   1556:                    if (TARGET_FLOAT_FORMAT != IEEE_FLOAT_FORMAT
                   1557:                        && REAL_VALUE_ISINF (value) && pedantic)
                   1558:                      pedwarn ("floating point number exceeds range of `double'");
                   1559:                  }
                   1560: 
                   1561:                set_float_handler (NULL_PTR);
                   1562:            }
                   1563: #ifdef ERANGE
                   1564:            if (errno == ERANGE && !flag_traditional && pedantic)
                   1565:              {
                   1566:                /* ERANGE is also reported for underflow,
                   1567:                   so test the value to distinguish overflow from that.  */
                   1568:                if (TARGET_FLOAT_FORMAT != IEEE_FLOAT_FORMAT
                   1569:                    && (REAL_VALUES_LESS (dconst1, value)
                   1570:                        || REAL_VALUES_LESS (value, dconstm1)))
                   1571:                  {
                   1572:                    pedwarn ("floating point number exceeds range of `double'");
                   1573:                    exceeds_double = 1;
                   1574:                  }
                   1575:              }
                   1576: #endif
                   1577:            garbage_chars = 0;
                   1578:            while (isalnum (c) || c == '.' || c == '_'
                   1579:                   || (!flag_traditional && (c == '+' || c == '-')
                   1580:                       && (p[-1] == 'e' || p[-1] == 'E')))
                   1581:              {
                   1582:                if (p >= token_buffer + maxtoken - 3)
                   1583:                  p = extend_token_buffer (p);
                   1584:                *p++ = c;
                   1585:                c = getc (finput);
                   1586:                garbage_chars++;
                   1587:              }
                   1588:            if (garbage_chars > 0)
                   1589:              error ("garbage at end of number");
                   1590: 
                   1591:            /* Create a node with determined type and value.  */
                   1592:            if (imag)
                   1593:              yylval.ttype = build_complex (convert (type, integer_zero_node),
                   1594:                                            build_real (type, value));
                   1595:            else
                   1596:              yylval.ttype = build_real (type, value);
                   1597: 
                   1598:            ungetc (c, finput);
                   1599:            *p = 0;
                   1600:          }
                   1601:        else
                   1602:          {
                   1603:            tree traditional_type, ansi_type, type;
                   1604:            HOST_WIDE_INT high, low;
                   1605:            int spec_unsigned = 0;
                   1606:            int spec_long = 0;
                   1607:            int spec_long_long = 0;
                   1608:            int spec_imag = 0;
                   1609:            int bytes, warn, i;
                   1610: 
                   1611:            while (1)
                   1612:              {
                   1613:                if (c == 'u' || c == 'U')
                   1614:                  {
                   1615:                    if (spec_unsigned)
                   1616:                      error ("two `u's in integer constant");
                   1617:                    spec_unsigned = 1;
                   1618:                  }
                   1619:                else if (c == 'l' || c == 'L')
                   1620:                  {
                   1621:                    if (spec_long)
                   1622:                      {
                   1623:                        if (spec_long_long)
                   1624:                          error ("three `l's in integer constant");
                   1625:                        else if (pedantic)
                   1626:                          pedwarn ("ANSI C forbids long long integer constants");
                   1627:                        spec_long_long = 1;
                   1628:                      }
                   1629:                    spec_long = 1;
                   1630:                  }
                   1631:                else if (c == 'i' || c == 'j' || c == 'I' || c == 'J')
                   1632:                  {
                   1633:                    if (spec_imag)
                   1634:                      error ("more than one `i' or `j' in numeric constant");
                   1635:                    spec_imag = 1;
                   1636:                  }
                   1637:                else
                   1638:                  {
                   1639:                    if (isalnum (c) || c == '.' || c == '_'
                   1640:                        || (!flag_traditional && (c == '+' || c == '-')
                   1641:                            && (p[-1] == 'e' || p[-1] == 'E')))
                   1642:                      {
                   1643:                        error ("garbage at end of number");
                   1644:                        while (isalnum (c) || c == '.' || c == '_'
                   1645:                               || (!flag_traditional && (c == '+' || c == '-')
                   1646:                                   && (p[-1] == 'e' || p[-1] == 'E')))
                   1647:                          {
                   1648:                            if (p >= token_buffer + maxtoken - 3)
                   1649:                              p = extend_token_buffer (p);
                   1650:                            *p++ = c;
                   1651:                            c = getc (finput);
                   1652:                          }
                   1653:                      }
                   1654:                    break;
                   1655:                  }
                   1656:                if (p >= token_buffer + maxtoken - 3)
                   1657:                  p = extend_token_buffer (p);
                   1658:                *p++ = c;
                   1659:                c = getc (finput);
                   1660:              }
                   1661: 
                   1662:            ungetc (c, finput);
                   1663: 
                   1664:            /* If the constant is not long long and it won't fit in an
                   1665:               unsigned long, or if the constant is long long and won't fit
                   1666:               in an unsigned long long, then warn that the constant is out
                   1667:               of range.  */
                   1668: 
                   1669:            /* ??? This assumes that long long and long integer types are
                   1670:               a multiple of 8 bits.  This better than the original code
                   1671:               though which assumed that long was exactly 32 bits and long
                   1672:               long was exactly 64 bits.  */
                   1673: 
                   1674:            if (spec_long_long)
                   1675:              bytes = TYPE_PRECISION (long_long_integer_type_node) / 8;
                   1676:            else
                   1677:              bytes = TYPE_PRECISION (long_integer_type_node) / 8;
                   1678: 
                   1679:            warn = overflow;
                   1680:            for (i = bytes; i < TOTAL_PARTS; i++)
                   1681:              if (parts[i])
                   1682:                warn = 1;
                   1683:            if (warn)
                   1684:              pedwarn ("integer constant out of range");
                   1685: 
                   1686:            /* This is simplified by the fact that our constant
                   1687:               is always positive.  */
                   1688: 
                   1689:            high = low = 0;
                   1690: 
                   1691:            for (i = 0; i < HOST_BITS_PER_WIDE_INT / HOST_BITS_PER_CHAR; i++)
                   1692:              {
                   1693:                high |= ((HOST_WIDE_INT) parts[i + (HOST_BITS_PER_WIDE_INT
                   1694:                                                    / HOST_BITS_PER_CHAR)]
                   1695:                         << (i * HOST_BITS_PER_CHAR));
                   1696:                low |= (HOST_WIDE_INT) parts[i] << (i * HOST_BITS_PER_CHAR);
                   1697:              }
                   1698:            
                   1699:            yylval.ttype = build_int_2 (low, high);
                   1700:            TREE_TYPE (yylval.ttype) = long_long_unsigned_type_node;
                   1701: 
                   1702:            /* If warn_traditional, calculate both the ANSI type and the
                   1703:               traditional type, then see if they disagree.
                   1704:               Otherwise, calculate only the type for the dialect in use.  */
                   1705:            if (warn_traditional || flag_traditional)
                   1706:              {
                   1707:                /* Calculate the traditional type.  */
                   1708:                /* Traditionally, any constant is signed;
                   1709:                   but if unsigned is specified explicitly, obey that.
                   1710:                   Use the smallest size with the right number of bits,
                   1711:                   except for one special case with decimal constants.  */
                   1712:                if (! spec_long && base != 10
                   1713:                    && int_fits_type_p (yylval.ttype, unsigned_type_node))
                   1714:                  traditional_type = (spec_unsigned ? unsigned_type_node
                   1715:                                      : integer_type_node);
                   1716:                /* A decimal constant must be long
                   1717:                   if it does not fit in type int.
                   1718:                   I think this is independent of whether
                   1719:                   the constant is signed.  */
                   1720:                else if (! spec_long && base == 10
                   1721:                         && int_fits_type_p (yylval.ttype, integer_type_node))
                   1722:                  traditional_type = (spec_unsigned ? unsigned_type_node
                   1723:                                      : integer_type_node);
                   1724:                else if (! spec_long_long)
                   1725:                  traditional_type = (spec_unsigned ? long_unsigned_type_node
                   1726:                                      : long_integer_type_node);
                   1727:                else
                   1728:                  traditional_type = (spec_unsigned
                   1729:                                      ? long_long_unsigned_type_node
                   1730:                                      : long_long_integer_type_node);
                   1731:              }
                   1732:            if (warn_traditional || ! flag_traditional)
                   1733:              {
                   1734:                /* Calculate the ANSI type.  */
                   1735:                if (! spec_long && ! spec_unsigned
                   1736:                    && int_fits_type_p (yylval.ttype, integer_type_node))
                   1737:                  ansi_type = integer_type_node;
                   1738:                else if (! spec_long && (base != 10 || spec_unsigned)
                   1739:                         && int_fits_type_p (yylval.ttype, unsigned_type_node))
                   1740:                  ansi_type = unsigned_type_node;
                   1741:                else if (! spec_unsigned && !spec_long_long
                   1742:                         && int_fits_type_p (yylval.ttype, long_integer_type_node))
                   1743:                  ansi_type = long_integer_type_node;
                   1744:                else if (! spec_long_long)
                   1745:                  ansi_type = long_unsigned_type_node;
                   1746:                else if (! spec_unsigned
                   1747:                         /* Verify value does not overflow into sign bit.  */
                   1748:                         && TREE_INT_CST_HIGH (yylval.ttype) >= 0
                   1749:                         && int_fits_type_p (yylval.ttype,
                   1750:                                             long_long_integer_type_node))
                   1751:                  ansi_type = long_long_integer_type_node;
                   1752:                else
                   1753:                  ansi_type = long_long_unsigned_type_node;
                   1754:              }
                   1755: 
                   1756:            type = flag_traditional ? traditional_type : ansi_type;
                   1757: 
                   1758:            if (warn_traditional && traditional_type != ansi_type)
                   1759:              {
                   1760:                if (TYPE_PRECISION (traditional_type)
                   1761:                    != TYPE_PRECISION (ansi_type))
                   1762:                  warning ("width of integer constant changes with -traditional");
                   1763:                else if (TREE_UNSIGNED (traditional_type)
                   1764:                         != TREE_UNSIGNED (ansi_type))
                   1765:                  warning ("integer constant is unsigned in ANSI C, signed with -traditional");
                   1766:                else
                   1767:                  warning ("width of integer constant may change on other systems with -traditional");
                   1768:              }
                   1769: 
                   1770:            if (!flag_traditional && !int_fits_type_p (yylval.ttype, type)
                   1771:                && !warn)
                   1772:              pedwarn ("integer constant out of range");
                   1773: 
                   1774:            if (base == 10 && ! spec_unsigned && TREE_UNSIGNED (type))
                   1775:              warning ("decimal constant is so large that it is unsigned");
                   1776: 
                   1777:            if (spec_imag)
                   1778:              {
                   1779:                if (TYPE_PRECISION (type)
                   1780:                    <= TYPE_PRECISION (integer_type_node))
                   1781:                  yylval.ttype
                   1782:                    = build_complex (integer_zero_node,
                   1783:                                     convert (integer_type_node, yylval.ttype));
                   1784:                else
                   1785:                  error ("complex integer constant is too wide for `complex int'");
                   1786:              }
                   1787:            else if (flag_traditional && !int_fits_type_p (yylval.ttype, type))
                   1788:              /* The traditional constant 0x80000000 is signed
                   1789:                 but doesn't fit in the range of int.
                   1790:                 This will change it to -0x80000000, which does fit.  */
                   1791:              {
                   1792:                TREE_TYPE (yylval.ttype) = unsigned_type (type);
                   1793:                yylval.ttype = convert (type, yylval.ttype);
                   1794:              }
                   1795:            else
                   1796:              TREE_TYPE (yylval.ttype) = type;
                   1797: 
                   1798:            *p = 0;
                   1799:          }
                   1800: 
                   1801:        value = CONSTANT; break;
                   1802:       }
                   1803: 
                   1804:     case '\'':
                   1805:     char_constant:
                   1806:       {
                   1807:        register int result = 0;
                   1808:        register int num_chars = 0;
                   1809:        unsigned width = TYPE_PRECISION (char_type_node);
                   1810:        int max_chars;
                   1811: 
                   1812:        if (wide_flag)
                   1813:          {
                   1814:            width = WCHAR_TYPE_SIZE;
                   1815: #ifdef MULTIBYTE_CHARS
                   1816:            max_chars = MB_CUR_MAX;
                   1817: #else
                   1818:            max_chars = 1;
                   1819: #endif
                   1820:          }
                   1821:        else
                   1822:          max_chars = TYPE_PRECISION (integer_type_node) / width;
                   1823: 
                   1824:        while (1)
                   1825:          {
                   1826:          tryagain:
                   1827: 
                   1828:            c = getc (finput);
                   1829: 
                   1830:            if (c == '\'' || c == EOF)
                   1831:              break;
                   1832: 
                   1833:            if (c == '\\')
                   1834:              {
                   1835:                int ignore = 0;
                   1836:                c = readescape (&ignore);
                   1837:                if (ignore)
                   1838:                  goto tryagain;
                   1839:                if (width < HOST_BITS_PER_INT
                   1840:                    && (unsigned) c >= (1 << width))
                   1841:                  pedwarn ("escape sequence out of range for character");
                   1842: #ifdef MAP_CHARACTER
                   1843:                if (isprint (c))
                   1844:                  c = MAP_CHARACTER (c);
                   1845: #endif
                   1846:              }
                   1847:            else if (c == '\n')
                   1848:              {
                   1849:                if (pedantic)
                   1850:                  pedwarn ("ANSI C forbids newline in character constant");
                   1851:                lineno++;
                   1852:              }
                   1853: #ifdef MAP_CHARACTER
                   1854:            else
                   1855:              c = MAP_CHARACTER (c);
                   1856: #endif
                   1857: 
                   1858:            num_chars++;
                   1859:            if (num_chars > maxtoken - 4)
                   1860:              extend_token_buffer (token_buffer);
                   1861: 
                   1862:            token_buffer[num_chars] = c;
                   1863: 
                   1864:            /* Merge character into result; ignore excess chars.  */
                   1865:            if (num_chars < max_chars + 1)
                   1866:              {
                   1867:                if (width < HOST_BITS_PER_INT)
                   1868:                  result = (result << width) | (c & ((1 << width) - 1));
                   1869:                else
                   1870:                  result = c;
                   1871:              }
                   1872:          }
                   1873: 
                   1874:        token_buffer[num_chars + 1] = '\'';
                   1875:        token_buffer[num_chars + 2] = 0;
                   1876: 
                   1877:        if (c != '\'')
                   1878:          error ("malformatted character constant");
                   1879:        else if (num_chars == 0)
                   1880:          error ("empty character constant");
                   1881:        else if (num_chars > max_chars)
                   1882:          {
                   1883:            num_chars = max_chars;
                   1884:            error ("character constant too long");
                   1885:          }
                   1886:        else if (num_chars != 1 && ! flag_traditional)
                   1887:          warning ("multi-character character constant");
                   1888: 
                   1889:        /* If char type is signed, sign-extend the constant.  */
                   1890:        if (! wide_flag)
                   1891:          {
                   1892:            int num_bits = num_chars * width;
                   1893:            if (num_bits == 0)
                   1894:              /* We already got an error; avoid invalid shift.  */
                   1895:              yylval.ttype = build_int_2 (0, 0);
                   1896:            else if (TREE_UNSIGNED (char_type_node)
                   1897:                     || ((result >> (num_bits - 1)) & 1) == 0)
                   1898:              yylval.ttype
                   1899:                = build_int_2 (result & ((unsigned HOST_WIDE_INT) ~0
                   1900:                                         >> (HOST_BITS_PER_WIDE_INT - num_bits)),
                   1901:                               0);
                   1902:            else
                   1903:              yylval.ttype
                   1904:                = build_int_2 (result | ~((unsigned HOST_WIDE_INT) ~0
                   1905:                                          >> (HOST_BITS_PER_WIDE_INT - num_bits)),
                   1906:                               -1);
                   1907:            TREE_TYPE (yylval.ttype) = integer_type_node;
                   1908:          }
                   1909:        else
                   1910:          {
                   1911: #ifdef MULTIBYTE_CHARS
                   1912:            /* Set the initial shift state and convert the next sequence.  */
                   1913:            result = 0;
                   1914:            /* In all locales L'\0' is zero and mbtowc will return zero,
                   1915:               so don't use it.  */
                   1916:            if (num_chars > 1
                   1917:                || (num_chars == 1 && token_buffer[1] != '\0'))
                   1918:              {
                   1919:                wchar_t wc;
                   1920:                (void) mbtowc (NULL_PTR, NULL_PTR, 0);
                   1921:                if (mbtowc (& wc, token_buffer + 1, num_chars) == num_chars)
                   1922:                  result = wc;
                   1923:                else
                   1924:                  warning ("Ignoring invalid multibyte character");
                   1925:              }
                   1926: #endif
                   1927:            yylval.ttype = build_int_2 (result, 0);
                   1928:            TREE_TYPE (yylval.ttype) = wchar_type_node;
                   1929:          }
                   1930: 
                   1931:        value = CONSTANT;
                   1932:        break;
                   1933:       }
                   1934: 
                   1935:     case '"':
                   1936:     string_constant:
                   1937:       {
                   1938:        c = getc (finput);
                   1939:        p = token_buffer + 1;
                   1940: 
                   1941:        while (c != '"' && c >= 0)
                   1942:          {
                   1943:            if (c == '\\')
                   1944:              {
                   1945:                int ignore = 0;
                   1946:                c = readescape (&ignore);
                   1947:                if (ignore)
                   1948:                  goto skipnewline;
                   1949:                if (!wide_flag
                   1950:                    && TYPE_PRECISION (char_type_node) < HOST_BITS_PER_INT
                   1951:                    && c >= (1 << TYPE_PRECISION (char_type_node)))
                   1952:                  pedwarn ("escape sequence out of range for character");
                   1953:              }
                   1954:            else if (c == '\n')
                   1955:              {
                   1956:                if (pedantic)
                   1957:                  pedwarn ("ANSI C forbids newline in string constant");
                   1958:                lineno++;
                   1959:              }
                   1960: 
                   1961:            if (p == token_buffer + maxtoken)
                   1962:              p = extend_token_buffer (p);
                   1963:            *p++ = c;
                   1964: 
                   1965:          skipnewline:
                   1966:            c = getc (finput);
                   1967:          }
                   1968:        *p = 0;
                   1969: 
                   1970:        /* We have read the entire constant.
                   1971:           Construct a STRING_CST for the result.  */
                   1972: 
                   1973:        if (wide_flag)
                   1974:          {
                   1975:            /* If this is a L"..." wide-string, convert the multibyte string
                   1976:               to a wide character string.  */
                   1977:            char *widep = (char *) alloca ((p - token_buffer) * WCHAR_BYTES);
                   1978:            int len;
                   1979: 
                   1980: #ifdef MULTIBYTE_CHARS
                   1981:            len = mbstowcs ((wchar_t *) widep, token_buffer + 1, p - token_buffer);
                   1982:            if (len < 0 || len >= (p - token_buffer))
                   1983:              {
                   1984:                warning ("Ignoring invalid multibyte string");
                   1985:                len = 0;
                   1986:              }
                   1987:            bzero (widep + (len * WCHAR_BYTES), WCHAR_BYTES);
                   1988: #else
                   1989:            {
                   1990:              union { long l; char c[sizeof (long)]; } u;
                   1991:              int big_endian;
                   1992:              char *wp, *cp;
                   1993: 
                   1994:              /* Determine whether host is little or big endian.  */
                   1995:              u.l = 1;
                   1996:              big_endian = u.c[sizeof (long) - 1];
                   1997:              wp = widep + (big_endian ? WCHAR_BYTES - 1 : 0);
                   1998: 
                   1999:              bzero (widep, (p - token_buffer) * WCHAR_BYTES);
                   2000:              for (cp = token_buffer + 1; cp < p; cp++)
                   2001:                *wp = *cp, wp += WCHAR_BYTES;
                   2002:              len = p - token_buffer - 1;
                   2003:            }
                   2004: #endif
                   2005:            yylval.ttype = build_string ((len + 1) * WCHAR_BYTES, widep);
                   2006:            TREE_TYPE (yylval.ttype) = wchar_array_type_node;
                   2007:            value = STRING;
                   2008:          }
                   2009:        else if (objc_flag)
                   2010:          {
                   2011:            extern tree build_objc_string();
                   2012:            /* Return an Objective-C @"..." constant string object.  */
                   2013:            yylval.ttype = build_objc_string (p - token_buffer,
                   2014:                                              token_buffer + 1);
                   2015:            TREE_TYPE (yylval.ttype) = char_array_type_node;
                   2016:            value = OBJC_STRING;
                   2017:          }
                   2018:        else
                   2019:          {
                   2020:            yylval.ttype = build_string (p - token_buffer, token_buffer + 1);
                   2021:            TREE_TYPE (yylval.ttype) = char_array_type_node;
                   2022:            value = STRING;
                   2023:          }
                   2024: 
                   2025:        *p++ = '"';
                   2026:        *p = 0;
                   2027: 
                   2028:        break;
                   2029:       }
                   2030: 
                   2031:     case '+':
                   2032:     case '-':
                   2033:     case '&':
                   2034:     case '|':
                   2035:     case '<':
                   2036:     case '>':
                   2037:     case '*':
                   2038:     case '/':
                   2039:     case '%':
                   2040:     case '^':
                   2041:     case '!':
                   2042:     case '=':
                   2043:       {
                   2044:        register int c1;
                   2045: 
                   2046:       combine:
                   2047: 
                   2048:        switch (c)
                   2049:          {
                   2050:          case '+':
                   2051:            yylval.code = PLUS_EXPR; break;
                   2052:          case '-':
                   2053:            yylval.code = MINUS_EXPR; break;
                   2054:          case '&':
                   2055:            yylval.code = BIT_AND_EXPR; break;
                   2056:          case '|':
                   2057:            yylval.code = BIT_IOR_EXPR; break;
                   2058:          case '*':
                   2059:            yylval.code = MULT_EXPR; break;
                   2060:          case '/':
                   2061:            yylval.code = TRUNC_DIV_EXPR; break;
                   2062:          case '%':
                   2063:            yylval.code = TRUNC_MOD_EXPR; break;
                   2064:          case '^':
                   2065:            yylval.code = BIT_XOR_EXPR; break;
                   2066:          case LSHIFT:
                   2067:            yylval.code = LSHIFT_EXPR; break;
                   2068:          case RSHIFT:
                   2069:            yylval.code = RSHIFT_EXPR; break;
                   2070:          case '<':
                   2071:            yylval.code = LT_EXPR; break;
                   2072:          case '>':
                   2073:            yylval.code = GT_EXPR; break;
                   2074:          }
                   2075: 
                   2076:        token_buffer[1] = c1 = getc (finput);
                   2077:        token_buffer[2] = 0;
                   2078: 
                   2079:        if (c1 == '=')
                   2080:          {
                   2081:            switch (c)
                   2082:              {
                   2083:              case '<':
                   2084:                value = ARITHCOMPARE; yylval.code = LE_EXPR; goto done;
                   2085:              case '>':
                   2086:                value = ARITHCOMPARE; yylval.code = GE_EXPR; goto done;
                   2087:              case '!':
                   2088:                value = EQCOMPARE; yylval.code = NE_EXPR; goto done;
                   2089:              case '=':
                   2090:                value = EQCOMPARE; yylval.code = EQ_EXPR; goto done;
                   2091:              }
                   2092:            value = ASSIGN; goto done;
                   2093:          }
                   2094:        else if (c == c1)
                   2095:          switch (c)
                   2096:            {
                   2097:            case '+':
                   2098:              value = PLUSPLUS; goto done;
                   2099:            case '-':
                   2100:              value = MINUSMINUS; goto done;
                   2101:            case '&':
                   2102:              value = ANDAND; goto done;
                   2103:            case '|':
                   2104:              value = OROR; goto done;
                   2105:            case '<':
                   2106:              c = LSHIFT;
                   2107:              goto combine;
                   2108:            case '>':
                   2109:              c = RSHIFT;
                   2110:              goto combine;
                   2111:            }
                   2112:        else if ((c == '-') && (c1 == '>'))
                   2113:          { value = POINTSAT; goto done; }
                   2114:        ungetc (c1, finput);
                   2115:        token_buffer[1] = 0;
                   2116: 
                   2117:        if ((c == '<') || (c == '>'))
                   2118:          value = ARITHCOMPARE;
                   2119:        else value = c;
                   2120:        goto done;
                   2121:       }
                   2122: 
                   2123:     case 0:
                   2124:       /* Don't make yyparse think this is eof.  */
                   2125:       value = 1;
                   2126:       break;
                   2127: 
                   2128:     default:
                   2129:       value = c;
                   2130:     }
                   2131: 
                   2132: done:
                   2133: /*  yylloc.last_line = lineno; */
                   2134: 
                   2135:   return value;
                   2136: }
                   2137: 
                   2138: /* Sets the value of the 'yydebug' variable to VALUE.
                   2139:    This is a function so we don't have to have YYDEBUG defined
                   2140:    in order to build the compiler.  */
                   2141: 
                   2142: void
                   2143: set_yydebug (value)
                   2144:      int value;
                   2145: {
                   2146: #if YYDEBUG != 0
                   2147:   yydebug = value;
                   2148: #else
                   2149:   warning ("YYDEBUG not defined.");
                   2150: #endif
                   2151: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.