1 /* 2 * This file and its contents are supplied under the terms of the 3 * Common Development and Distribution License ("CDDL"), version 1.0. 4 * You may only use this file in accordance with the terms version 1.0 5 * of the CDDL. 6 * 7 * A full copy of the text of the CDDL should have accompanied this 8 * source. A copy of the CDDL is also available via the Internet at 9 * http://www.illumos.org/license/CDDL. 10 */ 11 12 /* 13 * Copyright 2010 Nexenta Systems, Inc. All rights reserved. 14 */ 15 16 /* 17 * This file contains the "scanner", which tokenizes the input files 18 * for localedef for processing by the higher level grammar processor. 19 */ 20 21 #include <stdio.h> 22 #include <stdlib.h> 23 #include <ctype.h> 24 #include <limits.h> 25 #include <string.h> 26 #include <widec.h> 27 #include <sys/types.h> 28 #include <assert.h> 29 #include "localedef.h" 30 #include "parser.tab.h" 31 32 int com_char = '#'; 33 int esc_char = '\\'; 34 int mb_cur_min = 1; 35 int mb_cur_max = 1; 36 int lineno = 1; 37 int warnings = 0; 38 static int nextline; 39 static FILE *input = stdin; 40 static const char *filename = "<stdin>"; 41 static int instring = 0; 42 static int escaped = 0; 43 44 /* 45 * Token space ... grows on demand. 46 */ 47 static char *token = NULL; 48 static int tokidx; 49 static int toksz = 0; 50 static int hadtok = 0; 51 52 /* 53 * Wide string space ... grows on demand. 54 */ 55 static wchar_t *widestr = NULL; 56 static int wideidx = 0; 57 static int widesz = 0; 58 59 /* 60 * The last keyword seen. This is useful to trigger the special lexer rules 61 * for "copy" and also collating symbols and elements. 62 */ 63 int last_kw = 0; 64 static int category = T_END; 65 66 static struct token { 67 int id; 68 const char *name; 69 } keywords[] = { 70 { T_COM_CHAR, "comment_char" }, 71 { T_ESC_CHAR, "escape_char" }, 72 { T_END, "END" }, 73 { T_COPY, "copy" }, 74 { T_MESSAGES, "LC_MESSAGES" }, 75 { T_YESSTR, "yesstr" }, 76 { T_YESEXPR, "yesexpr" }, 77 { T_NOSTR, "nostr" }, 78 { T_NOEXPR, "noexpr" }, 79 { T_MONETARY, "LC_MONETARY" }, 80 { T_INT_CURR_SYMBOL, "int_curr_symbol" }, 81 { T_CURRENCY_SYMBOL, "currency_symbol" }, 82 { T_MON_DECIMAL_POINT, "mon_decimal_point" }, 83 { T_MON_THOUSANDS_SEP, "mon_thousands_sep" }, 84 { T_POSITIVE_SIGN, "positive_sign" }, 85 { T_NEGATIVE_SIGN, "negative_sign" }, 86 { T_MON_GROUPING, "mon_grouping" }, 87 { T_INT_FRAC_DIGITS, "int_frac_digits" }, 88 { T_FRAC_DIGITS, "frac_digits" }, 89 { T_P_CS_PRECEDES, "p_cs_precedes" }, 90 { T_P_SEP_BY_SPACE, "p_sep_by_space" }, 91 { T_N_CS_PRECEDES, "n_cs_precedes" }, 92 { T_N_SEP_BY_SPACE, "n_sep_by_space" }, 93 { T_P_SIGN_POSN, "p_sign_posn" }, 94 { T_N_SIGN_POSN, "n_sign_posn" }, 95 { T_INT_P_CS_PRECEDES, "int_p_cs_precedes" }, 96 { T_INT_N_CS_PRECEDES, "int_n_cs_precedes" }, 97 { T_INT_P_SEP_BY_SPACE, "int_p_sep_by_space" }, 98 { T_INT_N_SEP_BY_SPACE, "int_n_sep_by_space" }, 99 { T_INT_P_SIGN_POSN, "int_p_sign_posn" }, 100 { T_INT_N_SIGN_POSN, "int_n_sign_posn" }, 101 { T_COLLATE, "LC_COLLATE" }, 102 { T_COLLATING_SYMBOL, "collating-symbol" }, 103 { T_COLLATING_ELEMENT, "collating-element" }, 104 { T_FROM, "from" }, 105 { T_ORDER_START, "order_start" }, 106 { T_ORDER_END, "order_end" }, 107 { T_FORWARD, "forward" }, 108 { T_BACKWARD, "backward" }, 109 { T_POSITION, "position" }, 110 { T_IGNORE, "IGNORE" }, 111 { T_UNDEFINED, "UNDEFINED" }, 112 { T_NUMERIC, "LC_NUMERIC" }, 113 { T_DECIMAL_POINT, "decimal_point" }, 114 { T_THOUSANDS_SEP, "thousands_sep" }, 115 { T_GROUPING, "grouping" }, 116 { T_TIME, "LC_TIME" }, 117 { T_ABDAY, "abday" }, 118 { T_DAY, "day" }, 119 { T_ABMON, "abmon" }, 120 { T_MON, "mon" }, 121 { T_D_T_FMT, "d_t_fmt" }, 122 { T_D_FMT, "d_fmt" }, 123 { T_T_FMT, "t_fmt" }, 124 { T_AM_PM, "am_pm" }, 125 { T_T_FMT_AMPM, "t_fmt_ampm" }, 126 { T_ERA, "era" }, 127 { T_ERA_D_FMT, "era_d_fmt" }, 128 { T_ERA_T_FMT, "era_t_fmt" }, 129 { T_ERA_D_T_FMT, "era_d_t_fmt" }, 130 { T_ALT_DIGITS, "alt_digits" }, 131 { T_CTYPE, "LC_CTYPE" }, 132 { T_ISUPPER, "upper" }, 133 { T_ISLOWER, "lower" }, 134 { T_ISALPHA, "alpha" }, 135 { T_ISDIGIT, "digit" }, 136 { T_ISPUNCT, "punct" }, 137 { T_ISXDIGIT, "xdigit" }, 138 { T_ISSPACE, "space" }, 139 { T_ISPRINT, "print" }, 140 { T_ISGRAPH, "graph" }, 141 { T_ISBLANK, "blank" }, 142 { T_ISCNTRL, "cntrl" }, 143 /* 144 * These entries are local additions, and not specified by 145 * TOG. Note that they are not guaranteed to be accurate for 146 * all locales, and so applications should not depend on them. 147 */ 148 { T_ISSPECIAL, "special" }, 149 { T_ISENGLISH, "english" }, 150 { T_ISPHONOGRAM, "phonogram" }, 151 { T_ISIDEOGRAM, "ideogram" }, 152 { T_ISNUMBER, "number" }, 153 /* 154 * We have to support this in the grammar, but it would be a 155 * syntax error to define a character as one of these without 156 * also defining it as an alpha or digit. We ignore it in our 157 * parsing. 158 */ 159 { T_ISALNUM, "alnum" }, 160 { T_TOUPPER, "toupper" }, 161 { T_TOLOWER, "tolower" }, 162 163 /* 164 * These are keywords used in the charmap file. Note that 165 * Solaris orginally used angle brackets to wrap some of them, 166 * but we removed that to simplify our parser. The first of these 167 * items are "global items." 168 */ 169 { T_CHARMAP, "CHARMAP" }, 170 { T_WIDTH, "WIDTH" }, 171 { T_WIDTH_DEFAULT, "WIDTH_DEFAULT" }, 172 173 { -1, NULL }, 174 }; 175 176 /* 177 * These special words are only used in a charmap file, enclosed in <>. 178 */ 179 static struct token symwords[] = { 180 { T_COM_CHAR, "comment_char" }, 181 { T_ESC_CHAR, "escape_char" }, 182 { T_CODE_SET, "code_set_name" }, 183 { T_MB_CUR_MAX, "mb_cur_max" }, 184 { T_MB_CUR_MIN, "mb_cur_min" }, 185 { -1, NULL }, 186 }; 187 188 static int categories[] = { 189 T_CHARMAP, 190 T_CTYPE, 191 T_COLLATE, 192 T_MESSAGES, 193 T_MONETARY, 194 T_NUMERIC, 195 T_TIME, 196 0 197 }; 198 199 void 200 reset_scanner(const char *fname) 201 { 202 if (fname == NULL) { 203 filename = "<stdin>"; 204 input = stdin; 205 } else { 206 if (input != stdin) 207 (void) fclose(input); 208 if ((input = fopen(fname, "r")) == NULL) { 209 perror("fopen"); 210 exit(4); 211 } 212 filename = fname; 213 } 214 com_char = '#'; 215 esc_char = '\\'; 216 instring = 0; 217 escaped = 0; 218 lineno = 1; 219 nextline = 1; 220 tokidx = 0; 221 wideidx = 0; 222 } 223 224 #define hex(x) \ 225 (isdigit(x) ? (x - '0') : ((islower(x) ? (x - 'a') : (x - 'A')) + 10)) 226 #define isodigit(x) ((x >= '0') && (x <= '7')) 227 228 static int 229 scanc(void) 230 { 231 int c; 232 233 c = getc(input); 234 lineno = nextline; 235 if (c == '\n') { 236 nextline++; 237 } 238 return (c); 239 } 240 241 static void 242 unscanc(int c) 243 { 244 if (c == '\n') { 245 nextline--; 246 } 247 if (ungetc(c, input) < 0) { 248 yyerror(_("ungetc failed")); 249 } 250 } 251 252 static int 253 scan_hex_byte(void) 254 { 255 int c1, c2; 256 int v; 257 258 c1 = scanc(); 259 if (!isxdigit(c1)) { 260 yyerror(_("malformed hex digit")); 261 return (0); 262 } 263 c2 = scanc(); 264 if (!isxdigit(c2)) { 265 yyerror(_("malformed hex digit")); 266 return (0); 267 } 268 v = ((hex(c1) << 4) | hex(c2)); 269 return (v); 270 } 271 272 static int 273 scan_dec_byte(void) 274 { 275 int c1, c2, c3; 276 int b; 277 278 c1 = scanc(); 279 if (!isdigit(c1)) { 280 yyerror(_("malformed decimal digit")); 281 return (0); 282 } 283 b = c1 - '0'; 284 c2 = scanc(); 285 if (!isdigit(c2)) { 286 yyerror(_("malformed decimal digit")); 287 return (0); 288 } 289 b *= 10; 290 b += (c2 - '0'); 291 c3 = scanc(); 292 if (!isdigit(c3)) { 293 unscanc(c3); 294 } else { 295 b *= 10; 296 b += (c3 - '0'); 297 } 298 return (b); 299 } 300 301 static int 302 scan_oct_byte(void) 303 { 304 int c1, c2, c3; 305 int b; 306 307 b = 0; 308 309 c1 = scanc(); 310 if (!isodigit(c1)) { 311 yyerror(_("malformed octal digit")); 312 return (0); 313 } 314 b = c1 - '0'; 315 c2 = scanc(); 316 if (!isodigit(c2)) { 317 yyerror(_("malformed octal digit")); 318 return (0); 319 } 320 b *= 8; 321 b += (c2 - '0'); 322 c3 = scanc(); 323 if (!isodigit(c3)) { 324 unscanc(c3); 325 } else { 326 b *= 8; 327 b += (c3 - '0'); 328 } 329 return (b); 330 } 331 332 void 333 add_tok(int c) 334 { 335 if ((tokidx + 1) >= toksz) { 336 toksz += 64; 337 if ((token = realloc(token, toksz)) == NULL) { 338 yyerror(_("out of memory")); 339 tokidx = 0; 340 toksz = 0; 341 return; 342 } 343 } 344 345 token[tokidx++] = (char)c; 346 token[tokidx] = 0; 347 } 348 void 349 add_wcs(wchar_t c) 350 { 351 if ((wideidx + 1) >= widesz) { 352 widesz += 64; 353 widestr = realloc(widestr, (widesz * sizeof (wchar_t))); 354 if (widestr == NULL) { 355 yyerror(_("out of memory")); 356 wideidx = 0; 357 widesz = 0; 358 return; 359 } 360 } 361 362 widestr[wideidx++] = c; 363 widestr[wideidx] = 0; 364 } 365 366 wchar_t * 367 get_wcs(void) 368 { 369 wchar_t *ws = widestr; 370 wideidx = 0; 371 widestr = NULL; 372 widesz = 0; 373 if (ws == NULL) { 374 if ((ws = wsdup(L"")) == NULL) { 375 yyerror(_("out of memory")); 376 } 377 } 378 return (ws); 379 } 380 381 static int 382 get_byte(void) 383 { 384 int c; 385 386 if ((c = scanc()) != esc_char) { 387 unscanc(c); 388 return (EOF); 389 } 390 c = scanc(); 391 392 switch (c) { 393 case 'd': 394 case 'D': 395 return (scan_dec_byte()); 396 case 'x': 397 case 'X': 398 return (scan_hex_byte()); 399 case '0': 400 case '1': 401 case '2': 402 case '3': 403 case '4': 404 case '5': 405 case '6': 406 case '7': 407 /* put the character back so we can get it */ 408 unscanc(c); 409 return (scan_oct_byte()); 410 default: 411 unscanc(c); 412 unscanc(esc_char); 413 return (EOF); 414 } 415 } 416 417 int 418 get_escaped(int c) 419 { 420 switch (c) { 421 case 'n': 422 return ('\n'); 423 case 'r': 424 return ('\r'); 425 case 't': 426 return ('\t'); 427 case 'f': 428 return ('\f'); 429 case 'v': 430 return ('\v'); 431 case 'b': 432 return ('\b'); 433 case 'a': 434 return ('\a'); 435 default: 436 return (c); 437 } 438 } 439 440 int 441 get_wide(void) 442 { 443 static char mbs[MB_LEN_MAX + 1] = ""; 444 static int mbi = 0; 445 int c; 446 wchar_t wc; 447 448 if (mb_cur_max >= sizeof (mbs)) { 449 yyerror(_("max multibyte character size too big")); 450 mbi = 0; 451 return (T_NULL); 452 } 453 for (;;) { 454 if ((mbi == mb_cur_max) || ((c = get_byte()) == EOF)) { 455 /* 456 * end of the byte sequence reached, but no 457 * valid wide decoding. fatal error. 458 */ 459 mbi = 0; 460 yyerror(_("not a valid character encoding")); 461 return (T_NULL); 462 } 463 mbs[mbi++] = c; 464 mbs[mbi] = 0; 465 466 /* does it decode? */ 467 if (to_wide(&wc, mbs) >= 0) { 468 break; 469 } 470 } 471 472 mbi = 0; 473 if (category != T_CHARMAP) { 474 if (check_charmap(wc) < 0) { 475 yyerror(_("no symbolic name for character")); 476 return (T_NULL); 477 } 478 } 479 480 yylval.wc = wc; 481 return (T_CHAR); 482 } 483 484 int 485 get_symbol(void) 486 { 487 int c; 488 489 while ((c = scanc()) != EOF) { 490 if (escaped) { 491 escaped = 0; 492 if (c == '\n') 493 continue; 494 add_tok(get_escaped(c)); 495 continue; 496 } 497 if (c == esc_char) { 498 escaped = 1; 499 continue; 500 } 501 if (c == '\n') { /* well that's strange! */ 502 yyerror(_("unterminated symbolic name")); 503 continue; 504 } 505 if (c == '>') { /* end of symbol */ 506 507 /* 508 * This restarts the token from the beginning 509 * the next time we scan a character. (This 510 * token is complete.) 511 */ 512 513 if (token == NULL) { 514 yyerror(_("missing symbolic name")); 515 return (T_NULL); 516 } 517 tokidx = 0; 518 519 /* 520 * A few symbols are handled as keywords outside 521 * of the normal categories. 522 */ 523 if (category == T_END) { 524 int i; 525 for (i = 0; symwords[i].name != 0; i++) { 526 if (strcmp(token, symwords[i].name) == 527 0) { 528 last_kw = symwords[i].id; 529 return (last_kw); 530 } 531 } 532 } 533 /* 534 * Contextual rule: Only literal characters are 535 * permitted in CHARMAP. Anywhere else the symbolic 536 * forms are fine. 537 */ 538 if ((category != T_CHARMAP) && 539 (lookup_charmap(token, &yylval.wc)) != -1) { 540 return (T_CHAR); 541 } 542 if ((yylval.collsym = lookup_collsym(token)) != NULL) { 543 return (T_COLLSYM); 544 } 545 if ((yylval.collelem = lookup_collelem(token)) != 546 NULL) { 547 return (T_COLLELEM); 548 } 549 /* its an undefined symbol */ 550 yylval.token = strdup(token); 551 token = NULL; 552 toksz = 0; 553 tokidx = 0; 554 return (T_SYMBOL); 555 } 556 add_tok(c); 557 } 558 559 yyerror(_("unterminated symbolic name")); 560 return (EOF); 561 } 562 563 int 564 get_category(void) 565 { 566 return (category); 567 } 568 569 static int 570 consume_token(void) 571 { 572 int len = tokidx; 573 int i; 574 575 tokidx = 0; 576 if (token == NULL) 577 return (T_NULL); 578 579 /* 580 * this one is special, because we don't want it to alter the 581 * last_kw field. 582 */ 583 if (strcmp(token, "...") == 0) { 584 return (T_ELLIPSIS); 585 } 586 587 /* search for reserved words first */ 588 for (i = 0; keywords[i].name; i++) { 589 int j; 590 if (strcmp(keywords[i].name, token) != 0) { 591 continue; 592 } 593 594 last_kw = keywords[i].id; 595 596 /* clear the top level category if we're done with it */ 597 if (last_kw == T_END) { 598 category = T_END; 599 } 600 601 /* set the top level category if we're changing */ 602 for (j = 0; categories[j]; j++) { 603 if (categories[j] != last_kw) 604 continue; 605 category = last_kw; 606 } 607 608 return (keywords[i].id); 609 } 610 611 /* maybe its a numeric constant? */ 612 if (isdigit(*token) || (*token == '-' && isdigit(token[1]))) { 613 char *eptr; 614 yylval.num = strtol(token, &eptr, 10); 615 if (*eptr != 0) 616 yyerror(_("malformed number")); 617 return (T_NUMBER); 618 } 619 620 /* 621 * A single lone character is treated as a character literal. 622 * To avoid duplication of effort, we stick in the charmap. 623 */ 624 if (len == 1) { 625 yylval.wc = token[0]; 626 return (T_CHAR); 627 } 628 629 /* anything else is treated as a symbolic name */ 630 yylval.token = strdup(token); 631 token = NULL; 632 toksz = 0; 633 tokidx = 0; 634 return (T_NAME); 635 } 636 637 void 638 scan_to_eol(void) 639 { 640 int c; 641 while ((c = scanc()) != '\n') { 642 if (c == EOF) { 643 /* end of file without newline! */ 644 errf(_("missing newline")); 645 return; 646 } 647 } 648 assert(c == '\n'); 649 } 650 651 int 652 yylex(void) 653 { 654 int c; 655 656 while ((c = scanc()) != EOF) { 657 658 /* special handling for quoted string */ 659 if (instring) { 660 if (escaped) { 661 escaped = 0; 662 663 /* if newline, just eat and forget it */ 664 if (c == '\n') 665 continue; 666 667 if (strchr("xXd01234567", c)) { 668 unscanc(c); 669 unscanc(esc_char); 670 return (get_wide()); 671 } 672 yylval.wc = get_escaped(c); 673 return (T_CHAR); 674 } 675 if (c == esc_char) { 676 escaped = 1; 677 continue; 678 } 679 switch (c) { 680 case '<': 681 return (get_symbol()); 682 case '>': 683 /* oops! should generate syntax error */ 684 return (T_GT); 685 case '"': 686 instring = 0; 687 return (T_QUOTE); 688 default: 689 yylval.wc = c; 690 return (T_CHAR); 691 } 692 } 693 694 /* escaped characters first */ 695 if (escaped) { 696 escaped = 0; 697 if (c == '\n') { 698 /* eat the newline */ 699 continue; 700 } 701 hadtok = 1; 702 if (tokidx) { 703 /* an escape mid-token is nonsense */ 704 return (T_NULL); 705 } 706 707 /* numeric escapes are treated as wide characters */ 708 if (strchr("xXd01234567", c)) { 709 unscanc(c); 710 unscanc(esc_char); 711 return (get_wide()); 712 } 713 714 add_tok(get_escaped(c)); 715 continue; 716 } 717 718 /* if it is the escape charter itself note it */ 719 if (c == esc_char) { 720 escaped = 1; 721 continue; 722 } 723 724 /* remove from the comment char to end of line */ 725 if (c == com_char) { 726 while (c != '\n') { 727 if ((c = scanc()) == EOF) { 728 /* end of file without newline! */ 729 return (EOF); 730 } 731 } 732 assert(c == '\n'); 733 if (!hadtok) { 734 /* 735 * If there were no tokens on this line, 736 * then just pretend it didn't exist at all. 737 */ 738 continue; 739 } 740 hadtok = 0; 741 return (T_NL); 742 } 743 744 if (strchr(" \t\n;()<>,\"", c) && (tokidx != 0)) { 745 /* 746 * These are all token delimiters. If there 747 * is a token already in progress, we need to 748 * process it. 749 */ 750 unscanc(c); 751 return (consume_token()); 752 } 753 754 switch (c) { 755 case '\n': 756 if (!hadtok) { 757 /* 758 * If the line was completely devoid of tokens, 759 * then just ignore it. 760 */ 761 continue; 762 } 763 /* we're starting a new line, reset the token state */ 764 hadtok = 0; 765 return (T_NL); 766 case ',': 767 hadtok = 1; 768 return (T_COMMA); 769 case ';': 770 hadtok = 1; 771 return (T_SEMI); 772 case '(': 773 hadtok = 1; 774 return (T_LPAREN); 775 case ')': 776 hadtok = 1; 777 return (T_RPAREN); 778 case '>': 779 hadtok = 1; 780 return (T_GT); 781 case '<': 782 /* symbol start! */ 783 hadtok = 1; 784 return (get_symbol()); 785 case ' ': 786 case '\t': 787 /* whitespace, just ignore it */ 788 continue; 789 case '"': 790 hadtok = 1; 791 instring = 1; 792 return (T_QUOTE); 793 default: 794 hadtok = 1; 795 add_tok(c); 796 continue; 797 } 798 } 799 return (EOF); 800 } 801 802 void 803 yyerror(const char *msg) 804 { 805 (void) fprintf(stderr, _("%s: %d: error: %s\n"), 806 filename, lineno, msg); 807 exit(4); 808 } 809 810 void 811 errf(const char *fmt, ...) 812 { 813 char *msg; 814 815 va_list va; 816 va_start(va, fmt); 817 (void) vasprintf(&msg, fmt, va); 818 va_end(va); 819 820 (void) fprintf(stderr, _("%s: %d: error: %s\n"), 821 filename, lineno, msg); 822 free(msg); 823 exit(4); 824 } 825 826 void 827 warn(const char *fmt, ...) 828 { 829 char *msg; 830 831 va_list va; 832 va_start(va, fmt); 833 (void) vasprintf(&msg, fmt, va); 834 va_end(va); 835 836 (void) fprintf(stderr, _("%s: %d: warning: %s\n"), 837 filename, lineno, msg); 838 free(msg); 839 warnings++; 840 if (!warnok) 841 exit(4); 842 } 843