1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright (c) 1989, 2010, Oracle and/or its affiliates. All rights reserved. 24 */ 25 26 /* 27 * Copyright (c) 2011 Gary Mills 28 */ 29 30 /* Copyright (c) 1988 AT&T */ 31 /* All Rights Reserved */ 32 33 /* 34 * Copyright (c) 2018, Joyent, Inc. 35 */ 36 37 #include <signal.h> 38 #include <unistd.h> 39 #include <fcntl.h> 40 #include "m4.h" 41 42 #if defined(__lint) 43 extern int yydebug; 44 #endif 45 46 #define match(c, s) (c == *s && (!s[1] || inpmatch(s+1))) 47 48 static char tmp_name[] = "/tmp/m4aXXXXX"; 49 static wchar_t prev_char; 50 static int mb_cur_max; 51 52 static void getflags(int *, char ***, int *); 53 static void initalloc(void); 54 static void expand(wchar_t **, int); 55 static void lnsync(FILE *); 56 static void fpath(FILE *); 57 static void puttok(wchar_t *); 58 static void error3(void); 59 static wchar_t itochr(int); 60 /*LINTED: E_STATIC_UNUSED*/ 61 static wchar_t *chkbltin(wchar_t *); 62 static wchar_t *inpmatch(wchar_t *); 63 static void chkspace(char **, int *, char ***); 64 static void catchsig(int); 65 static FILE *m4open(char ***, char *, int *); 66 static void showwrap(void); 67 static void sputchr(wchar_t, FILE *); 68 static void putchr(wchar_t); 69 static void *xcalloc(size_t, size_t); 70 static wint_t myfgetwc(FILE *, int); 71 static wint_t myfputwc(wchar_t, FILE *); 72 static int myfeof(int); 73 74 int 75 main(int argc, char **argv) 76 { 77 wchar_t t; 78 int i, opt_end = 0; 79 int sigs[] = {SIGHUP, SIGINT, SIGPIPE, 0}; 80 81 #if defined(__lint) 82 yydebug = 0; 83 #endif 84 85 for (i = 0; sigs[i]; ++i) { 86 if (signal(sigs[i], SIG_IGN) != SIG_IGN) 87 (void) signal(sigs[i], catchsig); 88 } 89 tempfile = mktemp(tmp_name); 90 (void) close(creat(tempfile, 0)); 91 92 (void) setlocale(LC_ALL, ""); 93 94 #if !defined(TEXT_DOMAIN) /* Should be defined by cc -D */ 95 #define TEXT_DOMAIN "SYS_TEST" 96 #endif 97 (void) textdomain(TEXT_DOMAIN); 98 99 if ((mb_cur_max = MB_CUR_MAX) > 1) 100 wide = 1; 101 102 procnam = argv[0]; 103 getflags(&argc, &argv, &opt_end); 104 initalloc(); 105 106 setfname("-"); 107 if (argc > 1) { 108 --argc; 109 ++argv; 110 if (strcmp(argv[0], "-")) { 111 ifile[ifx] = m4open(&argv, "r", &argc); 112 setfname(argv[0]); 113 } 114 } 115 116 for (;;) { 117 token[0] = t = getchr(); 118 token[1] = EOS; 119 120 if (t == WEOF) { 121 if (ifx > 0) { 122 (void) fclose(ifile[ifx]); 123 ipflr = ipstk[--ifx]; 124 continue; 125 } 126 127 getflags(&argc, &argv, &opt_end); 128 129 if (argc <= 1) 130 /* 131 * If dowrap() has been called, the m4wrap 132 * macro has been processed, and a linked 133 * list of m4wrap strings has been created. 134 * The list starts at wrapstart. 135 */ 136 if (wrapstart) { 137 /* 138 * Now that EOF has been processed, 139 * display the m4wrap strings. 140 */ 141 showwrap(); 142 continue; 143 } else 144 break; 145 --argc; 146 ++argv; 147 148 if (ifile[ifx] != stdin) 149 (void) fclose(ifile[ifx]); 150 151 if (strcmp(argv[0], "-")) 152 ifile[ifx] = m4open(&argv, "r", &argc); 153 else 154 ifile[ifx] = stdin; 155 156 setfname(argv[0]); 157 continue; 158 } 159 160 if (is_alpha(t) || t == '_') { 161 wchar_t *tp = token+1; 162 int tlim = toksize; 163 struct nlist *macadd; /* temp variable */ 164 165 while ((*tp = getchr()) != WEOF && 166 (is_alnum(*tp) || *tp == '_')) { 167 tp++; 168 if (--tlim <= 0) 169 error2(gettext( 170 "more than %d chars in word"), 171 toksize); 172 } 173 putbak(*tp); 174 *tp = EOS; 175 176 macadd = lookup(token); 177 *Ap = (wchar_t *)macadd; 178 if (macadd->def) { 179 if ((wchar_t *)(++Ap) >= astklm) { 180 --Ap; 181 error2(gettext( 182 "more than %d items on " 183 "argument stack"), 184 stksize); 185 } 186 187 if (Cp++ == NULL) 188 Cp = callst; 189 190 Cp->argp = Ap; 191 *Ap++ = op; 192 puttok(token); 193 stkchr(EOS); 194 t = getchr(); 195 putbak(t); 196 197 if (t != '(') 198 pbstr(L"()"); 199 else /* try to fix arg count */ 200 *Ap++ = op; 201 202 Cp->plev = 0; 203 } else { 204 puttok(token); 205 } 206 } else if (match(t, lquote)) { 207 int qlev = 1; 208 209 for (;;) { 210 token[0] = t = getchr(); 211 token[1] = EOS; 212 213 if (match(t, rquote)) { 214 if (--qlev > 0) 215 puttok(token); 216 else 217 break; 218 } else if (match(t, lquote)) { 219 ++qlev; 220 puttok(token); 221 } else { 222 if (t == WEOF) 223 error(gettext( 224 "EOF in quote")); 225 putchr(t); 226 } 227 } 228 } else if (match(t, lcom) && 229 ((lcom[0] != L'#' || lcom[1] != L'\0') || 230 prev_char != '$')) { 231 232 /* 233 * Don't expand commented macro (between lcom and 234 * rcom). 235 * What we know so far is that we have found the 236 * left comment char (lcom). 237 * Make sure we haven't found '#' (lcom) immediately 238 * preceded by '$' because we want to expand "$#". 239 */ 240 241 puttok(token); 242 for (;;) { 243 token[0] = t = getchr(); 244 token[1] = EOS; 245 if (match(t, rcom)) { 246 puttok(token); 247 break; 248 } else { 249 if (t == WEOF) 250 error(gettext( 251 "EOF in comment")); 252 putchr(t); 253 } 254 } 255 } else if (Cp == NULL) { 256 putchr(t); 257 } else if (t == '(') { 258 if (Cp->plev) 259 stkchr(t); 260 else { 261 /* skip white before arg */ 262 while ((t = getchr()) != WEOF && is_space(t)) 263 ; 264 265 putbak(t); 266 } 267 268 ++Cp->plev; 269 } else if (t == ')') { 270 --Cp->plev; 271 272 if (Cp->plev == 0) { 273 stkchr(EOS); 274 expand(Cp->argp, Ap-Cp->argp-1); 275 op = *Cp->argp; 276 Ap = Cp->argp-1; 277 278 if (--Cp < callst) 279 Cp = NULL; 280 } else 281 stkchr(t); 282 } else if (t == ',' && Cp->plev <= 1) { 283 stkchr(EOS); 284 *Ap = op; 285 286 if ((wchar_t *)(++Ap) >= astklm) { 287 --Ap; 288 error2(gettext( 289 "more than %d items on argument stack"), 290 stksize); 291 } 292 293 while ((t = getchr()) != WEOF && is_space(t)) 294 ; 295 296 putbak(t); 297 } else { 298 stkchr(t); 299 } 300 } 301 302 if (Cp != NULL) 303 error(gettext( 304 "EOF in argument list")); 305 306 delexit(exitstat, 1); 307 return (0); 308 } 309 310 static wchar_t * 311 inpmatch(wchar_t *s) 312 { 313 wchar_t *tp = token+1; 314 315 while (*s) { 316 *tp = getchr(); 317 318 if (*tp++ != *s++) { 319 *tp = EOS; 320 pbstr(token+1); 321 return (0); 322 } 323 } 324 325 *tp = EOS; 326 return (token); 327 } 328 329 static void 330 getflags(int *xargc, char ***xargv, int *option_end) 331 { 332 char *arg; 333 char *t; 334 wchar_t *s[3]; 335 336 while (*xargc > 1) { 337 arg = (*xargv)[1]; /* point arg to current argument */ 338 339 /* 340 * This argument is not an option if it equals "-" or if 341 * "--" has already been parsed. 342 */ 343 if (arg[0] != '-' || arg[1] == EOS || *option_end) 344 break; 345 if (arg[0] == '-' && arg[1] == '-' && arg[2] == '\0') { 346 *option_end = 1; 347 } else { 348 switch (arg[1]) { 349 case 'B': 350 chkspace(&arg, xargc, xargv); 351 bufsize = atoi(&arg[2]); 352 if (bufsize <= 0) { 353 bufsize = DEF_BUFSIZE; 354 } 355 break; 356 case 'D': 357 initalloc(); 358 chkspace(&arg, xargc, xargv); 359 for (t = &arg[2]; *t; t++) { 360 if (*t == '=') { 361 *t++ = EOS; 362 break; 363 } 364 } 365 s[1] = str2wstr(&arg[2], 1); 366 s[2] = str2wstr(t, 1); 367 dodef(&s[0], 2); 368 free(s[1]); 369 free(s[2]); 370 break; 371 case 'H': 372 chkspace(&arg, xargc, xargv); 373 hshsize = atoi(&arg[2]); 374 if (hshsize <= 0) { 375 hshsize = DEF_HSHSIZE; 376 } 377 break; 378 case 'S': 379 chkspace(&arg, xargc, xargv); 380 stksize = atoi(&arg[2]); 381 if (stksize <= 0) { 382 stksize = DEF_STKSIZE; 383 } 384 break; 385 case 'T': 386 chkspace(&arg, xargc, xargv); 387 toksize = atoi(&arg[2]); 388 if (toksize <= 0) { 389 toksize = DEF_TOKSIZE; 390 } 391 break; 392 case 'U': 393 initalloc(); 394 chkspace(&arg, xargc, xargv); 395 s[1] = str2wstr(&arg[2], 1); 396 doundef(&s[0], 1); 397 free(s[1]); 398 break; 399 case 'e': 400 setbuf(stdout, NULL); 401 (void) signal(SIGINT, SIG_IGN); 402 break; 403 case 's': 404 /* turn on line sync */ 405 sflag = 1; 406 break; 407 default: 408 (void) fprintf(stderr, 409 gettext("%s: bad option: %s\n"), 410 procnam, arg); 411 delexit(NOT_OK, 0); 412 } 413 } /* end else not "--" */ 414 415 (*xargv)++; 416 --(*xargc); 417 } /* end while options to process */ 418 } 419 420 /* 421 * Function: chkspace 422 * 423 * If there is a space between the option and its argument, 424 * adjust argptr so that &arg[2] will point to beginning of the option argument. 425 * This will ensure that processing in getflags() will work, because &arg[2] 426 * will point to the beginning of the option argument whether or not we have 427 * a space between the option and its argument. If there is a space between 428 * the option and its argument, also adjust xargv and xargc because we are 429 * processing the next argument. 430 */ 431 static void 432 chkspace(char **argptr, int *xargc, char ***xargv) 433 { 434 if ((*argptr)[2] == EOS) { 435 /* there is a space between the option and its argument */ 436 (*xargv)++; /* look at the next argument */ 437 --(*xargc); 438 /* 439 * Adjust argptr if the option is followed by an 440 * option argument. 441 */ 442 if (*xargc > 1) { 443 *argptr = (*xargv)[1]; 444 /* point &arg[2] to beginning of option argument */ 445 *argptr -= 2; 446 } 447 } 448 } 449 450 static void 451 initalloc(void) 452 { 453 static int done = 0; 454 int t; 455 456 if (done++) 457 return; 458 459 hshtab = xcalloc(hshsize, sizeof (struct nlist *)); 460 callst = xcalloc(stksize/3+1, sizeof (struct call)); 461 Ap = argstk = xcalloc(stksize+3, sizeof (wchar_t *)); 462 ipstk[0] = ipflr = ip = ibuf = xcalloc(bufsize+1, sizeof (wchar_t)); 463 op = obuf = xcalloc(bufsize+1, sizeof (wchar_t)); 464 token = xcalloc(toksize+1, sizeof (wchar_t)); 465 466 astklm = (wchar_t *)(&argstk[stksize]); 467 ibuflm = &ibuf[bufsize]; 468 obuflm = &obuf[bufsize]; 469 toklm = &token[toksize]; 470 471 for (t = 0; barray[t].bname; ++t) { 472 wchar_t p[2] = {0, EOS}; 473 474 p[0] = builtin(t); 475 install(barray[t].bname, p, NOPUSH); 476 } 477 install(L"unix", nullstr, NOPUSH); 478 } 479 480 void 481 install(wchar_t *nam, wchar_t *val, int mode) 482 { 483 struct nlist *np; 484 wchar_t *cp; 485 int l; 486 487 if (mode == PUSH) 488 (void) lookup(nam); /* lookup sets hshval */ 489 else 490 while (undef(nam)) /* undef calls lookup */ 491 ; 492 493 np = xcalloc(1, sizeof (*np)); 494 np->name = wstrdup(nam); 495 np->next = hshtab[hshval]; 496 hshtab[hshval] = np; 497 498 cp = xcalloc((l = wcslen(val))+1, sizeof (*val)); 499 np->def = cp; 500 cp = &cp[l]; 501 502 while (*val) 503 *--cp = *val++; 504 } 505 506 struct nlist * 507 lookup(wchar_t *str) 508 { 509 wchar_t *s1; 510 struct nlist *np; 511 static struct nlist nodef; 512 513 s1 = str; 514 515 for (hshval = 0; *s1; ) 516 hshval += *s1++; 517 518 hshval %= hshsize; 519 520 for (np = hshtab[hshval]; np != NULL; np = np->next) { 521 if (*str == *np->name && wcscmp(str, np->name) == 0) 522 return (np); 523 } 524 return (&nodef); 525 } 526 527 static void 528 expand(wchar_t **a1, int c) 529 { 530 wchar_t *dp; 531 struct nlist *sp; 532 533 sp = (struct nlist *)a1[-1]; 534 535 if (sp->tflag || trace) { 536 #if !defined(__lint) /* lint doesn't grok "%ws" */ 537 int i; 538 539 (void) fprintf(stderr, 540 "Trace(%d): %ws", Cp-callst, a1[0]); 541 #endif 542 543 if (c > 0) { 544 #if !defined(__lint) /* lint doesn't grok "%ws" */ 545 (void) fprintf(stderr, "(%ws", chkbltin(a1[1])); 546 for (i = 2; i <= c; ++i) 547 (void) fprintf(stderr, ",%ws", chkbltin(a1[i])); 548 #endif 549 (void) fprintf(stderr, ")"); 550 } 551 (void) fprintf(stderr, "\n"); 552 } 553 554 dp = sp->def; 555 556 for (; *dp; ++dp) { 557 if (is_builtin(*dp)) { 558 (*barray[builtin_idx(*dp)].bfunc)(a1, c); 559 } else if (dp[1] == '$') { 560 if (is_digit(*dp)) { 561 int n; 562 if ((n = *dp-'0') <= c) 563 pbstr(a1[n]); 564 ++dp; 565 } else if (*dp == '#') { 566 pbnum((long)c); 567 ++dp; 568 } else if (*dp == '*' || *dp == '@') { 569 int i = c; 570 wchar_t **a = a1; 571 572 if (i > 0) 573 for (;;) { 574 if (*dp == '@') 575 pbstr(rquote); 576 577 pbstr(a[i--]); 578 579 if (*dp == '@') 580 pbstr(lquote); 581 582 if (i <= 0) 583 break; 584 585 pbstr(L","); 586 } 587 ++dp; 588 } else 589 putbak(*dp); 590 } else 591 putbak(*dp); 592 } 593 } 594 595 void 596 setfname(char *s) 597 { 598 if (fname[ifx]) 599 free(fname[ifx]); 600 if ((fname[ifx] = strdup(s)) == NULL) 601 error(gettext("out of storage")); 602 fline[ifx] = 1; 603 nflag = 1; 604 lnsync(stdout); 605 } 606 607 static void 608 lnsync(FILE *iop) 609 { 610 static int cline = 0; 611 static int cfile = 0; 612 613 if (!sflag || iop != stdout) 614 return; 615 616 if (nflag || ifx != cfile) { 617 nflag = 0; 618 cfile = ifx; 619 (void) fprintf(iop, "#line %d \"", cline = fline[ifx]); 620 fpath(iop); 621 (void) fprintf(iop, "\"\n"); 622 } else if (++cline != fline[ifx]) 623 (void) fprintf(iop, "#line %d\n", cline = fline[ifx]); 624 } 625 626 static void 627 fpath(FILE *iop) 628 { 629 int i; 630 631 if (fname[0] == NULL) 632 return; 633 634 (void) fprintf(iop, "%s", fname[0]); 635 636 for (i = 1; i <= ifx; ++i) 637 (void) fprintf(iop, ":%s", fname[i]); 638 } 639 640 /* ARGSUSED */ 641 static void 642 catchsig(int i) 643 { 644 (void) signal(SIGHUP, SIG_IGN); 645 (void) signal(SIGINT, SIG_IGN); 646 delexit(NOT_OK, 0); 647 } 648 649 void 650 delexit(int code, int flushio) 651 { 652 int i; 653 654 cf = stdout; 655 656 /* 657 * if (ofx != 0) { 658 * ofx = 0; 659 * code = NOT_OK; 660 * } 661 */ 662 ofx = 0; /* ensure that everything comes out */ 663 for (i = 1; i < 10; i++) 664 undiv(i, code); 665 666 tempfile[7] = 'a'; 667 (void) unlink(tempfile); 668 669 /* flush standard I/O buffers, ie: call exit() not _exit() */ 670 if (flushio) 671 exit(code); 672 673 _exit(code); 674 } 675 676 static void 677 puttok(wchar_t *tp) 678 { 679 if (Cp) { 680 while (*tp) 681 stkchr(*tp++); 682 } else if (cf) { 683 while (*tp) { 684 sputchr(*tp++, cf); 685 } 686 } 687 } 688 689 void 690 pbstr(wchar_t *str) 691 { 692 wchar_t *p; 693 694 for (p = str + wcslen(str); --p >= str; ) 695 putbak(*p); 696 } 697 698 void 699 undiv(int i, int code) 700 { 701 FILE *fp; 702 wint_t c; 703 704 if (i < 1 || i > 9 || i == ofx || !ofile[i]) 705 return; 706 707 (void) fclose(ofile[i]); 708 tempfile[7] = 'a'+i; 709 710 if (code == OK && cf) { 711 fp = xfopen(tempfile, "r"); 712 713 if (wide) { 714 while ((c = myfgetwc(fp, -1)) != WEOF) 715 sputchr((wchar_t)c, cf); 716 } else { 717 while ((c = (wint_t)getc(fp)) != WEOF) 718 sputchr((wchar_t)c, cf); 719 } 720 721 (void) fclose(fp); 722 } 723 724 (void) unlink(tempfile); 725 ofile[i] = NULL; 726 } 727 728 void 729 pbnum(long num) 730 { 731 pbnbr(num, 10, 1); 732 } 733 734 void 735 pbnbr(long nbr, int base, int len) 736 { 737 int neg = 0; 738 739 if (base <= 0) 740 return; 741 742 if (nbr < 0) 743 neg = 1; 744 else 745 nbr = -nbr; 746 747 while (nbr < 0) { 748 int i; 749 if (base > 1) { 750 i = nbr%base; 751 nbr /= base; 752 #if (-3 % 2) != -1 753 while (i > 0) { 754 i -= base; 755 ++nbr; 756 } 757 #endif 758 i = -i; 759 } else { 760 i = 1; 761 ++nbr; 762 } 763 putbak(itochr(i)); 764 --len; 765 } 766 767 while (--len >= 0) 768 putbak('0'); 769 770 if (neg) 771 putbak('-'); 772 } 773 774 static wchar_t 775 itochr(int i) 776 { 777 if (i > 9) 778 return ((wchar_t)(i-10+'A')); 779 else 780 return ((wchar_t)(i+'0')); 781 } 782 783 long 784 ctol(wchar_t *str) 785 { 786 int sign; 787 long num; 788 789 while (is_space(*str)) 790 ++str; 791 num = 0; 792 if (*str == '-') { 793 sign = -1; 794 ++str; 795 } else 796 sign = 1; 797 while (is_digit(*str)) 798 num = num*10 + *str++ - '0'; 799 return (sign * num); 800 } 801 802 int 803 min(int a, int b) 804 { 805 if (a > b) 806 return (b); 807 return (a); 808 } 809 810 FILE * 811 xfopen(char *name, char *mode) 812 { 813 FILE *fp; 814 815 if ((fp = fopen(name, mode)) == NULL) 816 errorf(gettext("cannot open file: %s"), 817 strerror(errno)); 818 819 return (fp); 820 } 821 822 /* 823 * m4open 824 * 825 * Continue processing files when unable to open the given file argument. 826 */ 827 FILE * 828 m4open(char ***argvec, char *mode, int *argcnt) 829 { 830 FILE *fp; 831 char *arg; 832 833 while (*argcnt > 0) { 834 arg = (*argvec)[0]; /* point arg to current file name */ 835 if (arg[0] == '-' && arg[1] == EOS) 836 return (stdin); 837 else { 838 if ((fp = fopen(arg, mode)) == NULL) { 839 (void) fprintf(stderr, gettext( 840 "m4: cannot open %s: "), arg); 841 perror(""); 842 if (*argcnt == 1) { 843 /* last arg therefore exit */ 844 error3(); 845 } else { 846 exitstat = 1; 847 (*argvec)++; /* try next arg */ 848 (*argcnt)--; 849 } 850 } else 851 break; 852 } 853 } 854 return (fp); 855 } 856 857 void * 858 xmalloc(size_t size) 859 { 860 void *ptr; 861 862 if ((ptr = malloc(size)) == NULL) 863 error(gettext("out of storage")); 864 return (ptr); 865 } 866 867 static void * 868 xcalloc(size_t nbr, size_t size) 869 { 870 void *ptr; 871 872 ptr = xmalloc(nbr * size); 873 (void) memset(ptr, '\0', nbr * size); 874 return (ptr); 875 } 876 877 /* Typical format: "cannot open file: %s" */ 878 /* PRINTFLIKE1 */ 879 void 880 errorf(char *str, char *serr) 881 { 882 char buf[500]; 883 884 (void) snprintf(buf, sizeof (buf), str, serr); 885 error(buf); 886 } 887 888 /* PRINTFLIKE1 */ 889 void 890 error2(char *str, int num) 891 { 892 char buf[500]; 893 894 (void) snprintf(buf, sizeof (buf), str, num); 895 error(buf); 896 } 897 898 void 899 error(char *str) 900 { 901 (void) fprintf(stderr, "\n%s:", procnam); 902 fpath(stderr); 903 (void) fprintf(stderr, ":%d %s\n", fline[ifx], str); 904 error3(); 905 } 906 907 static void 908 error3() 909 { 910 if (Cp) { 911 struct call *mptr; 912 913 /* fix limit */ 914 *op = EOS; 915 (Cp+1)->argp = Ap+1; 916 917 for (mptr = callst; mptr <= Cp; ++mptr) { 918 wchar_t **aptr, **lim; 919 920 aptr = mptr->argp; 921 lim = (mptr+1)->argp-1; 922 if (mptr == callst) 923 (void) fputws(*aptr, stderr); 924 ++aptr; 925 (void) fputs("(", stderr); 926 if (aptr < lim) 927 for (;;) { 928 (void) fputws(*aptr++, stderr); 929 if (aptr >= lim) 930 break; 931 (void) fputs(",", stderr); 932 } 933 } 934 while (--mptr >= callst) 935 (void) fputs(")", stderr); 936 937 (void) fputs("\n", stderr); 938 } 939 delexit(NOT_OK, 1); 940 } 941 942 static wchar_t * 943 chkbltin(wchar_t *s) 944 { 945 static wchar_t buf[24]; 946 947 if (is_builtin(*s)) { 948 (void) swprintf(buf, sizeof (buf)/sizeof (wchar_t), L"<%ls>", 949 barray[builtin_idx(*s)].bname); 950 return (buf); 951 } 952 return (s); 953 } 954 955 wchar_t 956 getchr() 957 { 958 static wchar_t C; 959 960 prev_char = C; 961 if (ip > ipflr) 962 return (*--ip); 963 if (wide) { 964 C = (wchar_t)(myfeof(ifx) ? WEOF : myfgetwc(NULL, ifx)); 965 } else { 966 C = (wchar_t)(feof(ifile[ifx]) ? 967 WEOF : (wint_t)getc(ifile[ifx])); 968 } 969 if (C == '\n') 970 fline[ifx]++; 971 return (C); 972 } 973 974 /* 975 * showwrap 976 * 977 * Loop through the list of m4wrap strings. Call pbstr() so that the 978 * string will be displayed, then delete the list entry and free the memory 979 * allocated for it. 980 */ 981 static void 982 showwrap() 983 { 984 struct Wrap *prev; 985 986 while (wrapstart) { 987 pbstr(wrapstart->wrapstr); 988 free(wrapstart->wrapstr); 989 prev = wrapstart; 990 wrapstart = wrapstart->nxt; 991 free(prev); 992 } 993 } 994 995 static void 996 sputchr(wchar_t c, FILE *f) 997 { 998 wint_t ret; 999 1000 if (is_builtin(c)) 1001 return; 1002 if (wide) 1003 ret = myfputwc(c, f); 1004 else 1005 ret = (wint_t)putc((int)c, f); 1006 if (ret == WEOF) 1007 error(gettext("output error")); 1008 if (ret == '\n') 1009 lnsync(f); 1010 } 1011 1012 static void 1013 putchr(wchar_t c) 1014 { 1015 wint_t ret; 1016 1017 if (Cp) 1018 stkchr(c); 1019 else if (cf) { 1020 if (sflag) 1021 sputchr(c, cf); 1022 else { 1023 if (is_builtin(c)) 1024 return; 1025 if (wide) 1026 ret = myfputwc(c, cf); 1027 else 1028 ret = (wint_t)putc((int)c, cf); 1029 if (ret == WEOF) { 1030 error(gettext("output error")); 1031 } 1032 } 1033 } 1034 } 1035 1036 wchar_t * 1037 wstrdup(wchar_t *p) 1038 { 1039 size_t len = wcslen(p); 1040 wchar_t *ret; 1041 1042 ret = xmalloc((len + 1) * sizeof (wchar_t)); 1043 (void) wcscpy(ret, p); 1044 return (ret); 1045 } 1046 1047 int 1048 wstoi(wchar_t *p) 1049 { 1050 return ((int)wcstol(p, NULL, 10)); 1051 } 1052 1053 char * 1054 wstr2str(wchar_t *from, int alloc) 1055 { 1056 static char *retbuf; 1057 static size_t bsiz; 1058 char *p, *ret; 1059 1060 if (alloc) { 1061 ret = p = xmalloc(wcslen(from) * mb_cur_max + 1); 1062 } else { 1063 while (bsiz < (wcslen(from) * mb_cur_max + 1)) { 1064 if ((p = realloc(retbuf, bsiz + 256)) == NULL) 1065 error(gettext("out of storage")); 1066 bsiz += 256; 1067 retbuf = p; 1068 } 1069 ret = p = retbuf; 1070 } 1071 1072 if (wide) { 1073 while (*from) { 1074 int len; 1075 1076 if (*from & INVALID_CHAR) { 1077 *p = (char)(*from & ~INVALID_CHAR); 1078 len = 1; 1079 } else { 1080 if ((len = wctomb(p, *from)) == -1) { 1081 *p = (char)*from; 1082 len = 1; 1083 } 1084 } 1085 p += len; 1086 from++; 1087 } 1088 } else { 1089 while (*from) 1090 *p++ = (char)*from++; 1091 } 1092 *p = '\0'; 1093 1094 return (ret); 1095 } 1096 1097 wchar_t * 1098 str2wstr(char *from, int alloc) 1099 { 1100 static wchar_t *retbuf; 1101 static size_t bsiz; 1102 wchar_t *p, *ret; 1103 1104 if (alloc) { 1105 ret = p = xmalloc((strlen(from) + 1) * sizeof (wchar_t)); 1106 } else { 1107 while (bsiz < (strlen(from) + 1)) { 1108 if ((p = realloc(retbuf, 1109 (bsiz + 256) * sizeof (wchar_t))) == NULL) { 1110 error(gettext("out of storage")); 1111 } 1112 bsiz += 256; 1113 retbuf = p; 1114 } 1115 ret = p = retbuf; 1116 } 1117 1118 if (wide) { 1119 while (*from) { 1120 int len; 1121 wchar_t wc; 1122 1123 if ((len = mbtowc(&wc, from, mb_cur_max)) <= 0) { 1124 wc = *from | INVALID_CHAR; 1125 len = 1; 1126 } 1127 *p++ = wc; 1128 from += len; 1129 } 1130 } else { 1131 while (*from) 1132 *p++ = (unsigned char) *from++; 1133 } 1134 *p = 0; 1135 1136 return (ret); 1137 } 1138 1139 static wint_t 1140 myfgetwc(FILE *fp, int idx) 1141 { 1142 int i, c, len, nb; 1143 wchar_t wc; 1144 unsigned char *buf; 1145 1146 if (fp == NULL) 1147 fp = ifile[idx]; 1148 else 1149 idx = 10; /* extra slot */ 1150 buf = ibuffer[idx].buffer; 1151 nb = ibuffer[idx].nbytes; 1152 len = 0; 1153 for (i = 1; i <= mb_cur_max; i++) { 1154 if (nb < i) { 1155 c = getc(fp); 1156 if (c == EOF) { 1157 if (nb == 0) 1158 return (WEOF); 1159 else 1160 break; 1161 } 1162 buf[nb++] = (unsigned char)c; 1163 } 1164 if ((len = mbtowc(&wc, (char *)buf, i)) >= 0) 1165 break; 1166 } 1167 if (len <= 0) { 1168 wc = buf[0] | INVALID_CHAR; 1169 len = 1; 1170 } 1171 nb -= len; 1172 if (nb > 0) { 1173 for (i = 0; i < nb; i++) 1174 buf[i] = buf[i + len]; 1175 } 1176 ibuffer[idx].nbytes = nb; 1177 return (wc); 1178 } 1179 1180 static wint_t 1181 myfputwc(wchar_t wc, FILE *fp) 1182 { 1183 if (wc & INVALID_CHAR) { 1184 wc &= ~INVALID_CHAR; 1185 return (fputc((int)wc, fp)); 1186 } 1187 return (fputwc(wc, fp)); 1188 } 1189 1190 static int 1191 myfeof(int idx) 1192 { 1193 return (ibuffer[idx].nbytes == 0 && feof(ifile[idx])); 1194 }