]> git.saurik.com Git - bison.git/blame - src/scan-gram.l
build: lower gettext requirements.
[bison.git] / src / scan-gram.l
CommitLineData
e9955c83 1/* Bison Grammar Scanner -*- C -*-
3b1e470c 2
6789b8bd
JD
3 Copyright (C) 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009 Free
4 Software Foundation, Inc.
e9955c83
AD
5
6 This file is part of Bison, the GNU Compiler Compiler.
7
f16b0819 8 This program is free software: you can redistribute it and/or modify
e9955c83 9 it under the terms of the GNU General Public License as published by
f16b0819 10 the Free Software Foundation, either version 3 of the License, or
e9955c83
AD
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
f16b0819 19 along with this program. If not, see <http://www.gnu.org/licenses/>. */
e9955c83 20
4521fcdf 21%option debug nodefault noinput nounput noyywrap never-interactive
e9955c83
AD
22%option prefix="gram_" outfile="lex.yy.c"
23
24%{
4f6e011e
PE
25/* Work around a bug in flex 2.5.31. See Debian bug 333231
26 <http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=333231>. */
27#undef gram_wrap
28#define gram_wrap() 1
29
e9071366
AD
30#define FLEX_PREFIX(Id) gram_ ## Id
31#include "flex-scanner.h"
223ff46e 32
e9955c83 33#include "complain.h"
3f2d73f1 34#include "files.h"
e9955c83 35#include "gram.h"
ca407bdf 36#include "quotearg.h"
e9955c83 37#include "reader.h"
223ff46e 38#include "uniqstr.h"
e9955c83 39
e9071366
AD
40#include <mbswidth.h>
41#include <quote.h>
42
43#include "scan-gram.h"
44
45#define YY_DECL GRAM_LEX_DECL
2346344a 46
3f2d73f1 47#define YY_USER_INIT \
e9071366 48 code_start = scanner_cursor = loc->start; \
dc9701e8 49
3f2d73f1 50/* Location of scanner cursor. */
4a678af8 51static boundary scanner_cursor;
41141c56 52
e9071366 53#define YY_USER_ACTION location_compute (loc, &scanner_cursor, yytext, yyleng);
d8d3f94a 54
6c30d641 55static size_t no_cr_read (FILE *, char *, size_t);
d8d3f94a
PE
56#define YY_INPUT(buf, result, size) ((result) = no_cr_read (yyin, buf, size))
57
7685e2f7
AR
58#define ROLLBACK_CURRENT_TOKEN \
59 do { \
60 scanner_cursor.column -= mbsnwidth (yytext, yyleng, 0); \
61 yyless (0); \
62 } while (0)
63
7ec2d4cd 64/* A string representing the most recently saved token. */
7c0c6181 65static char *last_string;
7ec2d4cd 66
d5e8574b 67/* Bracketed identifier. */
7685e2f7
AR
68static uniqstr bracketed_id_str = 0;
69static location bracketed_id_loc;
70static boundary bracketed_id_start;
71static int bracketed_id_context_state = 0;
72
7ec2d4cd 73void
e9071366 74gram_scanner_last_string_free (void)
7ec2d4cd 75{
41141c56 76 STRING_FREE;
7ec2d4cd 77}
e9955c83 78
4517da37 79static void handle_syncline (char *, location);
1452af69 80static unsigned long int scan_integer (char const *p, int base, location loc);
d8d3f94a 81static int convert_ucn_to_byte (char const *hex_text);
aa418041 82static void unexpected_eof (boundary, char const *);
4febdd96 83static void unexpected_newline (boundary, char const *);
e9955c83
AD
84
85%}
e9071366
AD
86 /* A C-like comment in directives/rules. */
87%x SC_YACC_COMMENT
88 /* Strings and characters in directives/rules. */
e9955c83 89%x SC_ESCAPED_STRING SC_ESCAPED_CHARACTER
e9071366
AD
90 /* A identifier was just read in directives/rules. Special state
91 to capture the sequence `identifier :'. */
92%x SC_AFTER_IDENTIFIER
e9071366
AD
93
94 /* Three types of user code:
95 - prologue (code between `%{' `%}' in the first section, before %%);
96 - actions, printers, union, etc, (between braced in the middle section);
97 - epilogue (everything after the second %%). */
98%x SC_PROLOGUE SC_BRACED_CODE SC_EPILOGUE
99 /* C and C++ comments in code. */
100%x SC_COMMENT SC_LINE_COMMENT
101 /* Strings and characters in code. */
102%x SC_STRING SC_CHARACTER
d5e8574b 103 /* Bracketed identifiers support. */
7685e2f7 104%x SC_BRACKETED_ID SC_RETURN_BRACKETED_ID
e9955c83 105
c046698e
AD
106letter [-.abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_]
107id {letter}({letter}|[0-9])*
663ce7bb 108directive %{id}
624a35e2 109int [0-9]+
d8d3f94a
PE
110
111/* POSIX says that a tag must be both an id and a C union member, but
112 historically almost any character is allowed in a tag. We disallow
113 NUL and newline, as this simplifies our implementation. */
114tag [^\0\n>]+
115
116/* Zero or more instances of backslash-newline. Following GCC, allow
117 white space between the backslash and the newline. */
118splice (\\[ \f\t\v]*\n)*
e9955c83
AD
119
120%%
121%{
a706a1cc 122 /* Nesting level of the current code in braces. */
5362ed19 123 int braces_level IF_LINT (= 0);
1a9e39f1 124
3f2d73f1 125 /* Parent context state, when applicable. */
5362ed19 126 int context_state IF_LINT (= 0);
a706a1cc 127
3f2d73f1 128 /* Location of most recent identifier, when applicable. */
a2bc9dbc 129 location id_loc IF_LINT (= empty_location);
3f2d73f1 130
a2bc9dbc
PE
131 /* Where containing code started, when applicable. Its initial
132 value is relevant only when yylex is invoked in the SC_EPILOGUE
133 start condition. */
134 boundary code_start = scanner_cursor;
3f2d73f1 135
223ff46e
PE
136 /* Where containing comment or string or character literal started,
137 when applicable. */
a2bc9dbc 138 boundary token_start IF_LINT (= scanner_cursor);
e9955c83
AD
139%}
140
141
3f2d73f1
PE
142 /*-----------------------.
143 | Scanning white space. |
144 `-----------------------*/
145
7685e2f7 146<INITIAL,SC_AFTER_IDENTIFIER,SC_BRACKETED_ID,SC_RETURN_BRACKETED_ID>
3f2d73f1 147{
4febdd96 148 /* Comments and white space. */
83adb046 149 "," warn_at (*loc, _("stray `,' treated as white space"));
4febdd96 150 [ \f\n\t\v] |
3f2d73f1 151 "//".* ;
83adb046
PE
152 "/*" {
153 token_start = loc->start;
154 context_state = YY_START;
155 BEGIN SC_YACC_COMMENT;
156 }
3f2d73f1
PE
157
158 /* #line directives are not documented, and may be withdrawn or
159 modified in future versions of Bison. */
160 ^"#line "{int}" \"".*"\"\n" {
4517da37 161 handle_syncline (yytext + sizeof "#line " - 1, *loc);
3f2d73f1
PE
162 }
163}
164
165
e9955c83
AD
166 /*----------------------------.
167 | Scanning Bison directives. |
168 `----------------------------*/
72183df4
DJ
169
170 /* For directives that are also command line options, the regex must be
171 "%..."
172 after "[-_]"s are removed, and the directive must match the --long
173 option name, with a single string argument. Otherwise, add exceptions
174 to ../build-aux/cross-options.pl. */
175
e9955c83
AD
176<INITIAL>
177{
43e6aea5 178 "%binary" return PERCENT_NONASSOC;
136a0f76 179 "%code" return PERCENT_CODE;
43e6aea5
AD
180 "%debug" return PERCENT_DEBUG;
181 "%default"[-_]"prec" return PERCENT_DEFAULT_PREC;
182 "%define" return PERCENT_DEFINE;
183 "%defines" return PERCENT_DEFINES;
184 "%destructor" return PERCENT_DESTRUCTOR;
185 "%dprec" return PERCENT_DPREC;
186 "%error"[-_]"verbose" return PERCENT_ERROR_VERBOSE;
187 "%expect" return PERCENT_EXPECT;
188 "%expect"[-_]"rr" return PERCENT_EXPECT_RR;
189 "%file-prefix" return PERCENT_FILE_PREFIX;
e9955c83 190 "%fixed"[-_]"output"[-_]"files" return PERCENT_YACC;
43e6aea5
AD
191 "%initial-action" return PERCENT_INITIAL_ACTION;
192 "%glr-parser" return PERCENT_GLR_PARSER;
193 "%language" return PERCENT_LANGUAGE;
194 "%left" return PERCENT_LEFT;
195 "%lex-param" return PERCENT_LEX_PARAM;
196 "%locations" return PERCENT_LOCATIONS;
197 "%merge" return PERCENT_MERGE;
198 "%name"[-_]"prefix" return PERCENT_NAME_PREFIX;
199 "%no"[-_]"default"[-_]"prec" return PERCENT_NO_DEFAULT_PREC;
200 "%no"[-_]"lines" return PERCENT_NO_LINES;
201 "%nonassoc" return PERCENT_NONASSOC;
202 "%nondeterministic-parser" return PERCENT_NONDETERMINISTIC_PARSER;
203 "%nterm" return PERCENT_NTERM;
204 "%output" return PERCENT_OUTPUT;
205 "%parse-param" return PERCENT_PARSE_PARAM;
206 "%prec" return PERCENT_PREC;
207 "%printer" return PERCENT_PRINTER;
208 "%pure"[-_]"parser" return PERCENT_PURE_PARSER;
209 "%require" return PERCENT_REQUIRE;
210 "%right" return PERCENT_RIGHT;
211 "%skeleton" return PERCENT_SKELETON;
212 "%start" return PERCENT_START;
213 "%term" return PERCENT_TOKEN;
214 "%token" return PERCENT_TOKEN;
215 "%token"[-_]"table" return PERCENT_TOKEN_TABLE;
216 "%type" return PERCENT_TYPE;
217 "%union" return PERCENT_UNION;
218 "%verbose" return PERCENT_VERBOSE;
219 "%yacc" return PERCENT_YACC;
e9955c83 220
3f2d73f1 221 {directive} {
41141c56 222 complain_at (*loc, _("invalid directive: %s"), quote (yytext));
412f8a59 223 }
900c5db5 224
e9955c83 225 "=" return EQUAL;
e9071366 226 "|" return PIPE;
e9955c83 227 ";" return SEMICOLON;
12e35840 228 "<*>" return TYPE_TAG_ANY;
3ebecc24 229 "<>" return TYPE_TAG_NONE;
e9955c83 230
3f2d73f1 231 {id} {
58d7a1a1 232 val->uniqstr = uniqstr_new (yytext);
3f2d73f1 233 id_loc = *loc;
7685e2f7 234 bracketed_id_str = NULL;
3f2d73f1 235 BEGIN SC_AFTER_IDENTIFIER;
e9955c83
AD
236 }
237
d8d3f94a 238 {int} {
1452af69
PE
239 val->integer = scan_integer (yytext, 10, *loc);
240 return INT;
241 }
242 0[xX][0-9abcdefABCDEF]+ {
243 val->integer = scan_integer (yytext, 16, *loc);
d8d3f94a
PE
244 return INT;
245 }
e9955c83 246
601bdfab
AD
247 /* Identifiers may not start with a digit. Yet, don't silently
248 accept "1FOO" as "1 FOO". */
249 {int}{id} {
250 complain_at (*loc, _("invalid identifier: %s"), quote (yytext));
251 }
252
ac9b0e95 253 /* Characters. */
07c0db18 254 "'" token_start = loc->start; BEGIN SC_ESCAPED_CHARACTER;
e9955c83
AD
255
256 /* Strings. */
ca407bdf 257 "\"" token_start = loc->start; BEGIN SC_ESCAPED_STRING;
e9955c83
AD
258
259 /* Prologue. */
3f2d73f1 260 "%{" code_start = loc->start; BEGIN SC_PROLOGUE;
e9955c83
AD
261
262 /* Code in between braces. */
3f2d73f1
PE
263 "{" {
264 STRING_GROW;
265 braces_level = 0;
266 code_start = loc->start;
267 BEGIN SC_BRACED_CODE;
268 }
e9955c83
AD
269
270 /* A type. */
d8d3f94a 271 "<"{tag}">" {
223ff46e 272 obstack_grow (&obstack_for_string, yytext + 1, yyleng - 2);
41141c56 273 STRING_FINISH;
223ff46e 274 val->uniqstr = uniqstr_new (last_string);
41141c56 275 STRING_FREE;
4cdb01db
AD
276 return TYPE;
277 }
278
a706a1cc
PE
279 "%%" {
280 static int percent_percent_count;
e9955c83 281 if (++percent_percent_count == 2)
a2bc9dbc 282 BEGIN SC_EPILOGUE;
e9955c83
AD
283 return PERCENT_PERCENT;
284 }
285
7685e2f7
AR
286 "[" {
287 bracketed_id_str = NULL;
288 bracketed_id_start = loc->start;
289 bracketed_id_context_state = YY_START;
290 BEGIN SC_BRACKETED_ID;
291 }
292
a706a1cc 293 . {
41141c56 294 complain_at (*loc, _("invalid character: %s"), quote (yytext));
3f2d73f1 295 }
379f0ac8
PE
296
297 <<EOF>> {
298 loc->start = loc->end = scanner_cursor;
299 yyterminate ();
300 }
3f2d73f1
PE
301}
302
303
304 /*-----------------------------------------------------------------.
305 | Scanning after an identifier, checking whether a colon is next. |
306 `-----------------------------------------------------------------*/
307
308<SC_AFTER_IDENTIFIER>
309{
7685e2f7 310 "[" {
d5e8574b 311 if (bracketed_id_str)
7685e2f7
AR
312 {
313 ROLLBACK_CURRENT_TOKEN;
314 BEGIN SC_RETURN_BRACKETED_ID;
315 *loc = id_loc;
316 return ID;
317 }
d5e8574b
AR
318 else
319 {
320 bracketed_id_start = loc->start;
321 bracketed_id_context_state = YY_START;
322 BEGIN SC_BRACKETED_ID;
323 }
7685e2f7 324 }
3f2d73f1 325 ":" {
7685e2f7 326 BEGIN (bracketed_id_str ? SC_RETURN_BRACKETED_ID : INITIAL);
3f2d73f1 327 *loc = id_loc;
3f2d73f1
PE
328 return ID_COLON;
329 }
330 . {
7685e2f7
AR
331 ROLLBACK_CURRENT_TOKEN;
332 BEGIN (bracketed_id_str ? SC_RETURN_BRACKETED_ID : INITIAL);
3f2d73f1 333 *loc = id_loc;
3f2d73f1
PE
334 return ID;
335 }
336 <<EOF>> {
7685e2f7 337 BEGIN (bracketed_id_str ? SC_RETURN_BRACKETED_ID : INITIAL);
3f2d73f1 338 *loc = id_loc;
3f2d73f1 339 return ID;
e9955c83
AD
340 }
341}
342
7685e2f7
AR
343 /*--------------------------------.
344 | Scanning bracketed identifiers. |
345 `--------------------------------*/
346
347<SC_BRACKETED_ID>
348{
349 {id} {
d5e8574b 350 if (bracketed_id_str)
7685e2f7 351 {
d5e8574b
AR
352 complain_at (*loc, _("unexpected identifier in bracketed name: %s"),
353 quote (yytext));
7685e2f7
AR
354 }
355 else
356 {
d5e8574b
AR
357 bracketed_id_str = uniqstr_new (yytext);
358 bracketed_id_loc = *loc;
7685e2f7
AR
359 }
360 }
361 "]" {
362 BEGIN bracketed_id_context_state;
363 if (bracketed_id_str)
364 {
365 if (INITIAL == bracketed_id_context_state)
366 {
367 val->uniqstr = bracketed_id_str;
368 bracketed_id_str = 0;
369 *loc = bracketed_id_loc;
370 return BRACKETED_ID;
371 }
372 }
373 else
d5e8574b 374 complain_at (*loc, _("an identifier expected"));
7685e2f7
AR
375 }
376 . {
377 complain_at (*loc, _("invalid character in bracketed name: %s"),
378 quote (yytext));
379 }
380 <<EOF>> {
381 BEGIN bracketed_id_context_state;
382 unexpected_eof (bracketed_id_start, "]");
383 }
384}
385
386<SC_RETURN_BRACKETED_ID>
387{
388 . {
389 ROLLBACK_CURRENT_TOKEN;
390 val->uniqstr = bracketed_id_str;
391 bracketed_id_str = 0;
392 *loc = bracketed_id_loc;
393 BEGIN INITIAL;
394 return BRACKETED_ID;
395 }
396}
397
e9955c83 398
d8d3f94a
PE
399 /*---------------------------------------------------------------.
400 | Scanning a Yacc comment. The initial `/ *' is already eaten. |
401 `---------------------------------------------------------------*/
e9955c83 402
d8d3f94a 403<SC_YACC_COMMENT>
e9955c83 404{
3f2d73f1 405 "*/" BEGIN context_state;
a706a1cc 406 .|\n ;
aa418041 407 <<EOF>> unexpected_eof (token_start, "*/"); BEGIN context_state;
d8d3f94a
PE
408}
409
410
411 /*------------------------------------------------------------.
412 | Scanning a C comment. The initial `/ *' is already eaten. |
413 `------------------------------------------------------------*/
414
415<SC_COMMENT>
416{
3f2d73f1 417 "*"{splice}"/" STRING_GROW; BEGIN context_state;
aa418041 418 <<EOF>> unexpected_eof (token_start, "*/"); BEGIN context_state;
e9955c83
AD
419}
420
421
d8d3f94a
PE
422 /*--------------------------------------------------------------.
423 | Scanning a line comment. The initial `//' is already eaten. |
424 `--------------------------------------------------------------*/
425
426<SC_LINE_COMMENT>
427{
3f2d73f1 428 "\n" STRING_GROW; BEGIN context_state;
41141c56 429 {splice} STRING_GROW;
3f2d73f1 430 <<EOF>> BEGIN context_state;
d8d3f94a
PE
431}
432
433
4febdd96
PE
434 /*------------------------------------------------.
435 | Scanning a Bison string, including its escapes. |
436 | The initial quote is already eaten. |
437 `------------------------------------------------*/
e9955c83
AD
438
439<SC_ESCAPED_STRING>
440{
47aee066
JD
441 "\""|"\n" {
442 if (yytext[0] == '\n')
443 unexpected_newline (token_start, "\"");
444 STRING_FINISH;
445 loc->start = token_start;
446 val->chars = last_string;
447 BEGIN INITIAL;
448 return STRING;
449 }
450 <<EOF>> {
451 unexpected_eof (token_start, "\"");
41141c56 452 STRING_FINISH;
3f2d73f1 453 loc->start = token_start;
223ff46e 454 val->chars = last_string;
a706a1cc 455 BEGIN INITIAL;
e9955c83
AD
456 return STRING;
457 }
e9955c83
AD
458}
459
4febdd96
PE
460 /*----------------------------------------------------------.
461 | Scanning a Bison character literal, decoding its escapes. |
462 | The initial quote is already eaten. |
463 `----------------------------------------------------------*/
e9955c83
AD
464
465<SC_ESCAPED_CHARACTER>
466{
47aee066 467 "'"|"\n" {
41141c56 468 STRING_FINISH;
3f2d73f1 469 loc->start = token_start;
07c0db18 470 val->character = last_string[0];
ac9b0e95
JD
471 {
472 /* FIXME: Eventually, make these errors. */
07c0db18
JD
473 if (last_string[0] == '\0')
474 {
475 warn_at (*loc, _("empty character literal"));
476 /* '\0' seems dangerous even if we are about to complain. */
477 val->character = '\'';
478 }
479 else if (last_string[1] != '\0')
ac9b0e95
JD
480 warn_at (*loc, _("extra characters in character literal"));
481 }
482 if (yytext[0] == '\n')
483 unexpected_newline (token_start, "'");
41141c56 484 STRING_FREE;
a706a1cc 485 BEGIN INITIAL;
58d7a1a1 486 return CHAR;
e9955c83 487 }
47aee066 488 <<EOF>> {
47aee066
JD
489 STRING_FINISH;
490 loc->start = token_start;
07c0db18 491 val->character = last_string[0];
ac9b0e95 492 {
ac9b0e95 493 /* FIXME: Eventually, make these errors. */
07c0db18
JD
494 if (last_string[0] == '\0')
495 {
496 warn_at (*loc, _("empty character literal"));
497 /* '\0' seems dangerous even if we are about to complain. */
498 val->character = '\'';
499 }
500 else if (last_string[1] != '\0')
ac9b0e95 501 warn_at (*loc, _("extra characters in character literal"));
ac9b0e95
JD
502 }
503 unexpected_eof (token_start, "'");
47aee066
JD
504 STRING_FREE;
505 BEGIN INITIAL;
506 return CHAR;
507 }
4febdd96 508}
a706a1cc 509
4febdd96
PE
510<SC_ESCAPED_CHARACTER,SC_ESCAPED_STRING>
511{
92ac3705 512 \0 complain_at (*loc, _("invalid null character"));
e9955c83
AD
513}
514
515
516 /*----------------------------.
517 | Decode escaped characters. |
518 `----------------------------*/
519
520<SC_ESCAPED_STRING,SC_ESCAPED_CHARACTER>
521{
d8d3f94a 522 \\[0-7]{1,3} {
4517da37 523 unsigned long int c = strtoul (yytext + 1, NULL, 8);
d8d3f94a 524 if (UCHAR_MAX < c)
3f2d73f1 525 complain_at (*loc, _("invalid escape sequence: %s"), quote (yytext));
05ac60f3 526 else if (! c)
92ac3705 527 complain_at (*loc, _("invalid null character: %s"), quote (yytext));
e9955c83 528 else
223ff46e 529 obstack_1grow (&obstack_for_string, c);
e9955c83
AD
530 }
531
6b0d38ab 532 \\x[0-9abcdefABCDEF]+ {
4517da37
PE
533 verify (UCHAR_MAX < ULONG_MAX);
534 unsigned long int c = strtoul (yytext + 2, NULL, 16);
535 if (UCHAR_MAX < c)
3f2d73f1 536 complain_at (*loc, _("invalid escape sequence: %s"), quote (yytext));
92ac3705
PE
537 else if (! c)
538 complain_at (*loc, _("invalid null character: %s"), quote (yytext));
d8d3f94a 539 else
223ff46e 540 obstack_1grow (&obstack_for_string, c);
e9955c83
AD
541 }
542
223ff46e
PE
543 \\a obstack_1grow (&obstack_for_string, '\a');
544 \\b obstack_1grow (&obstack_for_string, '\b');
545 \\f obstack_1grow (&obstack_for_string, '\f');
546 \\n obstack_1grow (&obstack_for_string, '\n');
547 \\r obstack_1grow (&obstack_for_string, '\r');
548 \\t obstack_1grow (&obstack_for_string, '\t');
549 \\v obstack_1grow (&obstack_for_string, '\v');
412f8a59
PE
550
551 /* \\[\"\'?\\] would be shorter, but it confuses xgettext. */
223ff46e 552 \\("\""|"'"|"?"|"\\") obstack_1grow (&obstack_for_string, yytext[1]);
412f8a59 553
6b0d38ab 554 \\(u|U[0-9abcdefABCDEF]{4})[0-9abcdefABCDEF]{4} {
d8d3f94a
PE
555 int c = convert_ucn_to_byte (yytext);
556 if (c < 0)
3f2d73f1 557 complain_at (*loc, _("invalid escape sequence: %s"), quote (yytext));
92ac3705
PE
558 else if (! c)
559 complain_at (*loc, _("invalid null character: %s"), quote (yytext));
d8d3f94a 560 else
223ff46e 561 obstack_1grow (&obstack_for_string, c);
d8d3f94a 562 }
4f25ebb0 563 \\(.|\n) {
3f2d73f1 564 complain_at (*loc, _("unrecognized escape sequence: %s"), quote (yytext));
41141c56 565 STRING_GROW;
e9955c83
AD
566 }
567}
568
4febdd96
PE
569 /*--------------------------------------------.
570 | Scanning user-code characters and strings. |
571 `--------------------------------------------*/
e9955c83 572
4febdd96
PE
573<SC_CHARACTER,SC_STRING>
574{
e9071366 575 {splice}|\\{splice}[^\n\[\]] STRING_GROW;
4febdd96 576}
e9955c83
AD
577
578<SC_CHARACTER>
579{
4febdd96
PE
580 "'" STRING_GROW; BEGIN context_state;
581 \n unexpected_newline (token_start, "'"); BEGIN context_state;
582 <<EOF>> unexpected_eof (token_start, "'"); BEGIN context_state;
e9955c83
AD
583}
584
e9955c83
AD
585<SC_STRING>
586{
4febdd96
PE
587 "\"" STRING_GROW; BEGIN context_state;
588 \n unexpected_newline (token_start, "\""); BEGIN context_state;
589 <<EOF>> unexpected_eof (token_start, "\""); BEGIN context_state;
e9955c83
AD
590}
591
592
593 /*---------------------------------------------------.
594 | Strings, comments etc. can be found in user code. |
595 `---------------------------------------------------*/
596
597<SC_BRACED_CODE,SC_PROLOGUE,SC_EPILOGUE>
598{
3f2d73f1
PE
599 "'" {
600 STRING_GROW;
601 context_state = YY_START;
602 token_start = loc->start;
603 BEGIN SC_CHARACTER;
604 }
605 "\"" {
606 STRING_GROW;
607 context_state = YY_START;
608 token_start = loc->start;
609 BEGIN SC_STRING;
610 }
611 "/"{splice}"*" {
612 STRING_GROW;
613 context_state = YY_START;
614 token_start = loc->start;
615 BEGIN SC_COMMENT;
616 }
617 "/"{splice}"/" {
618 STRING_GROW;
619 context_state = YY_START;
620 BEGIN SC_LINE_COMMENT;
621 }
e9955c83
AD
622}
623
624
624a35e2 625
58d7a1a1
AD
626 /*-----------------------------------------------------------.
627 | Scanning some code in braces (actions). The initial "{" is |
628 | already eaten. |
629 `-----------------------------------------------------------*/
e9955c83
AD
630
631<SC_BRACED_CODE>
632{
41141c56
PE
633 "{"|"<"{splice}"%" STRING_GROW; braces_level++;
634 "%"{splice}">" STRING_GROW; braces_level--;
e9955c83 635 "}" {
25522739
PE
636 obstack_1grow (&obstack_for_string, '}');
637
2346344a
AD
638 --braces_level;
639 if (braces_level < 0)
e9955c83 640 {
41141c56 641 STRING_FINISH;
3f2d73f1 642 loc->start = code_start;
eb095650 643 val->code = last_string;
a706a1cc 644 BEGIN INITIAL;
58d7a1a1 645 return BRACED_CODE;
e9955c83
AD
646 }
647 }
648
a706a1cc
PE
649 /* Tokenize `<<%' correctly (as `<<' `%') rather than incorrrectly
650 (as `<' `<%'). */
41141c56 651 "<"{splice}"<" STRING_GROW;
a706a1cc 652
47aee066
JD
653 <<EOF>> {
654 unexpected_eof (code_start, "}");
655 STRING_FINISH;
656 loc->start = code_start;
eb095650 657 val->code = last_string;
47aee066
JD
658 BEGIN INITIAL;
659 return BRACED_CODE;
660 }
e9955c83
AD
661}
662
663
664 /*--------------------------------------------------------------.
665 | Scanning some prologue: from "%{" (already scanned) to "%}". |
666 `--------------------------------------------------------------*/
667
668<SC_PROLOGUE>
669{
670 "%}" {
41141c56 671 STRING_FINISH;
3f2d73f1 672 loc->start = code_start;
223ff46e 673 val->chars = last_string;
a706a1cc 674 BEGIN INITIAL;
e9955c83
AD
675 return PROLOGUE;
676 }
677
47aee066
JD
678 <<EOF>> {
679 unexpected_eof (code_start, "%}");
680 STRING_FINISH;
681 loc->start = code_start;
682 val->chars = last_string;
683 BEGIN INITIAL;
684 return PROLOGUE;
685 }
e9955c83
AD
686}
687
688
689 /*---------------------------------------------------------------.
690 | Scanning the epilogue (everything after the second "%%", which |
d8d3f94a 691 | has already been eaten). |
e9955c83
AD
692 `---------------------------------------------------------------*/
693
694<SC_EPILOGUE>
695{
e9955c83 696 <<EOF>> {
41141c56 697 STRING_FINISH;
3f2d73f1 698 loc->start = code_start;
223ff46e 699 val->chars = last_string;
a706a1cc 700 BEGIN INITIAL;
e9955c83
AD
701 return EPILOGUE;
702 }
703}
704
705
4febdd96
PE
706 /*-----------------------------------------------------.
707 | By default, grow the string obstack with the input. |
708 `-----------------------------------------------------*/
709
710<SC_COMMENT,SC_LINE_COMMENT,SC_BRACED_CODE,SC_PROLOGUE,SC_EPILOGUE,SC_STRING,SC_CHARACTER,SC_ESCAPED_STRING,SC_ESCAPED_CHARACTER>. |
711<SC_COMMENT,SC_LINE_COMMENT,SC_BRACED_CODE,SC_PROLOGUE,SC_EPILOGUE>\n STRING_GROW;
712
e9955c83
AD
713%%
714
6c30d641
PE
715/* Read bytes from FP into buffer BUF of size SIZE. Return the
716 number of bytes read. Remove '\r' from input, treating \r\n
717 and isolated \r as \n. */
718
719static size_t
720no_cr_read (FILE *fp, char *buf, size_t size)
721{
a737b216
PE
722 size_t bytes_read = fread (buf, 1, size, fp);
723 if (bytes_read)
6c30d641 724 {
a737b216 725 char *w = memchr (buf, '\r', bytes_read);
6c30d641
PE
726 if (w)
727 {
728 char const *r = ++w;
a737b216 729 char const *lim = buf + bytes_read;
6c30d641
PE
730
731 for (;;)
732 {
733 /* Found an '\r'. Treat it like '\n', but ignore any
734 '\n' that immediately follows. */
735 w[-1] = '\n';
736 if (r == lim)
737 {
738 int ch = getc (fp);
739 if (ch != '\n' && ungetc (ch, fp) != ch)
740 break;
741 }
742 else if (*r == '\n')
743 r++;
744
745 /* Copy until the next '\r'. */
746 do
747 {
748 if (r == lim)
749 return w - buf;
750 }
751 while ((*w++ = *r++) != '\r');
752 }
753
754 return w - buf;
755 }
756 }
757
a737b216 758 return bytes_read;
6c30d641
PE
759}
760
761
f25bfb75 762
1452af69
PE
763/*------------------------------------------------------.
764| Scan NUMBER for a base-BASE integer at location LOC. |
765`------------------------------------------------------*/
766
767static unsigned long int
768scan_integer (char const *number, int base, location loc)
769{
4517da37
PE
770 verify (INT_MAX < ULONG_MAX);
771 unsigned long int num = strtoul (number, NULL, base);
772
773 if (INT_MAX < num)
1452af69
PE
774 {
775 complain_at (loc, _("integer out of range: %s"), quote (number));
776 num = INT_MAX;
777 }
4517da37 778
1452af69
PE
779 return num;
780}
781
782
d8d3f94a
PE
783/*------------------------------------------------------------------.
784| Convert universal character name UCN to a single-byte character, |
785| and return that character. Return -1 if UCN does not correspond |
786| to a single-byte character. |
787`------------------------------------------------------------------*/
788
789static int
790convert_ucn_to_byte (char const *ucn)
791{
4517da37
PE
792 verify (UCHAR_MAX <= INT_MAX);
793 unsigned long int code = strtoul (ucn + 2, NULL, 16);
d8d3f94a
PE
794
795 /* FIXME: Currently we assume Unicode-compatible unibyte characters
796 on ASCII hosts (i.e., Latin-1 on hosts with 8-bit bytes). On
797 non-ASCII hosts we support only the portable C character set.
798 These limitations should be removed once we add support for
799 multibyte characters. */
800
801 if (UCHAR_MAX < code)
802 return -1;
803
804#if ! ('$' == 0x24 && '@' == 0x40 && '`' == 0x60 && '~' == 0x7e)
805 {
806 /* A non-ASCII host. Use CODE to index into a table of the C
807 basic execution character set, which is guaranteed to exist on
808 all Standard C platforms. This table also includes '$', '@',
8e6ef483 809 and '`', which are not in the basic execution character set but
d8d3f94a
PE
810 which are unibyte characters on all the platforms that we know
811 about. */
812 static signed char const table[] =
813 {
814 '\0', -1, -1, -1, -1, -1, -1, '\a',
815 '\b', '\t', '\n', '\v', '\f', '\r', -1, -1,
816 -1, -1, -1, -1, -1, -1, -1, -1,
817 -1, -1, -1, -1, -1, -1, -1, -1,
818 ' ', '!', '"', '#', '$', '%', '&', '\'',
819 '(', ')', '*', '+', ',', '-', '.', '/',
820 '0', '1', '2', '3', '4', '5', '6', '7',
821 '8', '9', ':', ';', '<', '=', '>', '?',
822 '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G',
823 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
824 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',
825 'X', 'Y', 'Z', '[', '\\', ']', '^', '_',
826 '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g',
827 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',
828 'p', 'q', 'r', 's', 't', 'u', 'v', 'w',
829 'x', 'y', 'z', '{', '|', '}', '~'
830 };
831
832 code = code < sizeof table ? table[code] : -1;
833 }
834#endif
c4d720cd 835
d8d3f94a
PE
836 return code;
837}
838
839
900c5db5
AD
840/*----------------------------------------------------------------.
841| Handle `#line INT "FILE"'. ARGS has already skipped `#line '. |
842`----------------------------------------------------------------*/
843
844static void
4517da37 845handle_syncline (char *args, location loc)
900c5db5 846{
4517da37
PE
847 char *after_num;
848 unsigned long int lineno = strtoul (args, &after_num, 10);
849 char *file = strchr (after_num, '"') + 1;
850 *strchr (file, '"') = '\0';
851 if (INT_MAX <= lineno)
852 {
853 warn_at (loc, _("line number overflow"));
854 lineno = INT_MAX;
855 }
e9071366 856 current_file = uniqstr_new (file);
0c8e079f 857 boundary_set (&scanner_cursor, current_file, lineno, 1);
4517da37
PE
858}
859
860
4febdd96
PE
861/*----------------------------------------------------------------.
862| For a token or comment starting at START, report message MSGID, |
863| which should say that an end marker was found before |
864| the expected TOKEN_END. |
865`----------------------------------------------------------------*/
866
867static void
868unexpected_end (boundary start, char const *msgid, char const *token_end)
869{
870 location loc;
871 loc.start = start;
872 loc.end = scanner_cursor;
873 complain_at (loc, _(msgid), token_end);
874}
875
876
3f2d73f1
PE
877/*------------------------------------------------------------------------.
878| Report an unexpected EOF in a token or comment starting at START. |
879| An end of file was encountered and the expected TOKEN_END was missing. |
3f2d73f1 880`------------------------------------------------------------------------*/
a706a1cc
PE
881
882static void
aa418041 883unexpected_eof (boundary start, char const *token_end)
a706a1cc 884{
4febdd96
PE
885 unexpected_end (start, N_("missing `%s' at end of file"), token_end);
886}
887
888
889/*----------------------------------------.
890| Likewise, but for unexpected newlines. |
891`----------------------------------------*/
892
893static void
894unexpected_newline (boundary start, char const *token_end)
895{
896 unexpected_end (start, N_("missing `%s' at end of line"), token_end);
a706a1cc
PE
897}
898
899
f25bfb75
AD
900/*-------------------------.
901| Initialize the scanner. |
902`-------------------------*/
903
1d6412ad 904void
e9071366 905gram_scanner_initialize (void)
1d6412ad 906{
223ff46e 907 obstack_init (&obstack_for_string);
1d6412ad
AD
908}
909
910
f25bfb75
AD
911/*-----------------------------------------------.
912| Free all the memory allocated to the scanner. |
913`-----------------------------------------------*/
914
4cdb01db 915void
e9071366 916gram_scanner_free (void)
4cdb01db 917{
223ff46e 918 obstack_free (&obstack_for_string, 0);
536545f3 919 /* Reclaim Flex's buffers. */
580b8926 920 yylex_destroy ();
4cdb01db 921}