1 /**************************************************************** 2 Copyright (C) Lucent Technologies 1997 3 All Rights Reserved 4 5 Permission to use, copy, modify, and distribute this software and 6 its documentation for any purpose and without fee is hereby 7 granted, provided that the above copyright notice appear in all 8 copies and that both that the copyright notice and this 9 permission notice and warranty disclaimer appear in supporting 10 documentation, and that the name Lucent Technologies or any of 11 its entities not be used in advertising or publicity pertaining 12 to distribution of the software without specific, written prior 13 permission. 14 15 LUCENT DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, 16 INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. 17 IN NO EVENT SHALL LUCENT OR ANY OF ITS ENTITIES BE LIABLE FOR ANY 18 SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 19 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER 20 IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, 21 ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF 22 THIS SOFTWARE. 23 ****************************************************************/ 24 25 #define DEBUG 26 #include <stdio.h> 27 #include <string.h> 28 #include <strings.h> 29 #include <ctype.h> 30 #include <errno.h> 31 #include <stdlib.h> 32 #include <stdarg.h> 33 #include <limits.h> 34 #include <math.h> 35 #include "awk.h" 36 37 char EMPTY[] = { '\0' }; 38 FILE *infile = NULL; 39 bool innew; /* true = infile has not been read by readrec */ 40 char *file = EMPTY; 41 char *record; 42 int recsize = RECSIZE; 43 char *fields; 44 int fieldssize = RECSIZE; 45 46 Cell **fldtab; /* pointers to Cells */ 47 static size_t len_inputFS = 0; 48 static char *inputFS = NULL; /* FS at time of input, for field splitting */ 49 50 #define MAXFLD 2 51 int nfields = MAXFLD; /* last allocated slot for $i */ 52 53 bool donefld; /* true = implies rec broken into fields */ 54 bool donerec; /* true = record is valid (no flds have changed) */ 55 56 int lastfld = 0; /* last used field */ 57 int argno = 1; /* current input argument number */ 58 extern Awkfloat *ARGC; 59 60 static Cell dollar0 = { OCELL, CFLD, NULL, EMPTY, 0.0, REC|STR|DONTFREE, NULL, NULL }; 61 static Cell dollar1 = { OCELL, CFLD, NULL, EMPTY, 0.0, FLD|STR|DONTFREE, NULL, NULL }; 62 63 void recinit(unsigned int n) 64 { 65 if ( (record = (char *) malloc(n)) == NULL 66 || (fields = (char *) malloc(n+1)) == NULL 67 || (fldtab = (Cell **) calloc(nfields+2, sizeof(*fldtab))) == NULL 68 || (fldtab[0] = (Cell *) malloc(sizeof(**fldtab))) == NULL) 69 FATAL("out of space for $0 and fields"); 70 *record = '\0'; 71 *fldtab[0] = dollar0; 72 fldtab[0]->sval = record; 73 fldtab[0]->nval = tostring("0"); 74 makefields(1, nfields); 75 } 76 77 void makefields(int n1, int n2) /* create $n1..$n2 inclusive */ 78 { 79 char temp[50]; 80 int i; 81 82 for (i = n1; i <= n2; i++) { 83 fldtab[i] = (Cell *) malloc(sizeof(**fldtab)); 84 if (fldtab[i] == NULL) 85 FATAL("out of space in makefields %d", i); 86 *fldtab[i] = dollar1; 87 snprintf(temp, sizeof(temp), "%d", i); 88 fldtab[i]->nval = tostring(temp); 89 } 90 } 91 92 void initgetrec(void) 93 { 94 int i; 95 char *p; 96 97 for (i = 1; i < *ARGC; i++) { 98 p = getargv(i); /* find 1st real filename */ 99 if (p == NULL || *p == '\0') { /* deleted or zapped */ 100 argno++; 101 continue; 102 } 103 if (!isclvar(p)) { 104 setsval(lookup("FILENAME", symtab), p); 105 return; 106 } 107 setclvar(p); /* a commandline assignment before filename */ 108 argno++; 109 } 110 infile = stdin; /* no filenames, so use stdin */ 111 innew = true; 112 } 113 114 /* 115 * POSIX specifies that fields are supposed to be evaluated as if they were 116 * split using the value of FS at the time that the record's value ($0) was 117 * read. 118 * 119 * Since field-splitting is done lazily, we save the current value of FS 120 * whenever a new record is read in (implicitly or via getline), or when 121 * a new value is assigned to $0. 122 */ 123 void savefs(void) 124 { 125 size_t len; 126 if ((len = strlen(getsval(fsloc))) < len_inputFS) { 127 strcpy(inputFS, *FS); /* for subsequent field splitting */ 128 return; 129 } 130 131 len_inputFS = len + 1; 132 inputFS = (char *) realloc(inputFS, len_inputFS); 133 if (inputFS == NULL) 134 FATAL("field separator %.10s... is too long", *FS); 135 memcpy(inputFS, *FS, len_inputFS); 136 } 137 138 static bool firsttime = true; 139 140 int getrec(char **pbuf, int *pbufsize, bool isrecord) /* get next input record */ 141 { /* note: cares whether buf == record */ 142 int c; 143 char *buf = *pbuf; 144 uschar saveb0; 145 int bufsize = *pbufsize, savebufsize = bufsize; 146 147 if (firsttime) { 148 firsttime = false; 149 initgetrec(); 150 } 151 DPRINTF("RS=<%s>, FS=<%s>, ARGC=%g, FILENAME=%s\n", 152 *RS, *FS, *ARGC, *FILENAME); 153 if (isrecord) { 154 donefld = false; 155 donerec = true; 156 savefs(); 157 } 158 saveb0 = buf[0]; 159 buf[0] = 0; 160 while (argno < *ARGC || infile == stdin) { 161 DPRINTF("argno=%d, file=|%s|\n", argno, file); 162 if (infile == NULL) { /* have to open a new file */ 163 file = getargv(argno); 164 if (file == NULL || *file == '\0') { /* deleted or zapped */ 165 argno++; 166 continue; 167 } 168 if (isclvar(file)) { /* a var=value arg */ 169 setclvar(file); 170 argno++; 171 continue; 172 } 173 *FILENAME = file; 174 DPRINTF("opening file %s\n", file); 175 if (*file == '-' && *(file+1) == '\0') 176 infile = stdin; 177 else if ((infile = fopen(file, "r")) == NULL) 178 FATAL("can't open file %s", file); 179 setfval(fnrloc, 0.0); 180 } 181 c = readrec(&buf, &bufsize, infile, innew); 182 if (innew) 183 innew = false; 184 if (c != 0 || buf[0] != '\0') { /* normal record */ 185 if (isrecord) { 186 double result; 187 188 if (freeable(fldtab[0])) 189 xfree(fldtab[0]->sval); 190 fldtab[0]->sval = buf; /* buf == record */ 191 fldtab[0]->tval = REC | STR | DONTFREE; 192 if (is_number(fldtab[0]->sval, & result)) { 193 fldtab[0]->fval = result; 194 fldtab[0]->tval |= NUM; 195 } 196 } 197 setfval(nrloc, nrloc->fval+1); 198 setfval(fnrloc, fnrloc->fval+1); 199 *pbuf = buf; 200 *pbufsize = bufsize; 201 return 1; 202 } 203 /* EOF arrived on this file; set up next */ 204 if (infile != stdin) 205 fclose(infile); 206 infile = NULL; 207 argno++; 208 } 209 buf[0] = saveb0; 210 *pbuf = buf; 211 *pbufsize = savebufsize; 212 return 0; /* true end of file */ 213 } 214 215 void nextfile(void) 216 { 217 if (infile != NULL && infile != stdin) 218 fclose(infile); 219 infile = NULL; 220 argno++; 221 } 222 223 int readrec(char **pbuf, int *pbufsize, FILE *inf, bool newflag) /* read one record into buf */ 224 { 225 int sep, c, isrec; 226 char *rr, *buf = *pbuf; 227 int bufsize = *pbufsize; 228 char *rs = getsval(rsloc); 229 230 if (*rs && rs[1]) { 231 bool found; 232 233 fa *pfa = makedfa(rs, 1); 234 if (newflag) 235 found = fnematch(pfa, inf, &buf, &bufsize, recsize); 236 else { 237 int tempstat = pfa->initstat; 238 pfa->initstat = 2; 239 found = fnematch(pfa, inf, &buf, &bufsize, recsize); 240 pfa->initstat = tempstat; 241 } 242 if (found) 243 setptr(patbeg, '\0'); 244 } else { 245 if ((sep = *rs) == 0) { 246 sep = '\n'; 247 while ((c=getc(inf)) == '\n' && c != EOF) /* skip leading \n's */ 248 ; 249 if (c != EOF) 250 ungetc(c, inf); 251 } 252 for (rr = buf; ; ) { 253 for (; (c=getc(inf)) != sep && c != EOF; ) { 254 if (rr-buf+1 > bufsize) 255 if (!adjbuf(&buf, &bufsize, 1+rr-buf, 256 recsize, &rr, "readrec 1")) 257 FATAL("input record `%.30s...' too long", buf); 258 *rr++ = c; 259 } 260 if (*rs == sep || c == EOF) 261 break; 262 if ((c = getc(inf)) == '\n' || c == EOF) /* 2 in a row */ 263 break; 264 if (!adjbuf(&buf, &bufsize, 2+rr-buf, recsize, &rr, 265 "readrec 2")) 266 FATAL("input record `%.30s...' too long", buf); 267 *rr++ = '\n'; 268 *rr++ = c; 269 } 270 if (!adjbuf(&buf, &bufsize, 1+rr-buf, recsize, &rr, "readrec 3")) 271 FATAL("input record `%.30s...' too long", buf); 272 *rr = 0; 273 } 274 *pbuf = buf; 275 *pbufsize = bufsize; 276 isrec = *buf || !feof(inf); 277 DPRINTF("readrec saw <%s>, returns %d\n", buf, isrec); 278 return isrec; 279 } 280 281 char *getargv(int n) /* get ARGV[n] */ 282 { 283 Cell *x; 284 char *s, temp[50]; 285 extern Array *ARGVtab; 286 287 snprintf(temp, sizeof(temp), "%d", n); 288 if (lookup(temp, ARGVtab) == NULL) 289 return NULL; 290 x = setsymtab(temp, "", 0.0, STR, ARGVtab); 291 s = getsval(x); 292 DPRINTF("getargv(%d) returns |%s|\n", n, s); 293 return s; 294 } 295 296 void setclvar(char *s) /* set var=value from s */ 297 { 298 char *p; 299 Cell *q; 300 double result; 301 302 for (p=s; *p != '='; p++) 303 ; 304 *p++ = 0; 305 p = qstring(p, '\0'); 306 q = setsymtab(s, p, 0.0, STR, symtab); 307 setsval(q, p); 308 if (is_number(q->sval, & result)) { 309 q->fval = result; 310 q->tval |= NUM; 311 } 312 DPRINTF("command line set %s to |%s|\n", s, p); 313 } 314 315 316 void fldbld(void) /* create fields from current record */ 317 { 318 /* this relies on having fields[] the same length as $0 */ 319 /* the fields are all stored in this one array with \0's */ 320 /* possibly with a final trailing \0 not associated with any field */ 321 char *r, *fr, sep; 322 Cell *p; 323 int i, j, n; 324 325 if (donefld) 326 return; 327 if (!isstr(fldtab[0])) 328 getsval(fldtab[0]); 329 r = fldtab[0]->sval; 330 n = strlen(r); 331 if (n > fieldssize) { 332 xfree(fields); 333 if ((fields = (char *) malloc(n+2)) == NULL) /* possibly 2 final \0s */ 334 FATAL("out of space for fields in fldbld %d", n); 335 fieldssize = n; 336 } 337 fr = fields; 338 i = 0; /* number of fields accumulated here */ 339 if (inputFS == NULL) /* make sure we have a copy of FS */ 340 savefs(); 341 if (strlen(inputFS) > 1) { /* it's a regular expression */ 342 i = refldbld(r, inputFS); 343 } else if ((sep = *inputFS) == ' ') { /* default whitespace */ 344 for (i = 0; ; ) { 345 while (*r == ' ' || *r == '\t' || *r == '\n') 346 r++; 347 if (*r == 0) 348 break; 349 i++; 350 if (i > nfields) 351 growfldtab(i); 352 if (freeable(fldtab[i])) 353 xfree(fldtab[i]->sval); 354 fldtab[i]->sval = fr; 355 fldtab[i]->tval = FLD | STR | DONTFREE; 356 do 357 *fr++ = *r++; 358 while (*r != ' ' && *r != '\t' && *r != '\n' && *r != '\0'); 359 *fr++ = 0; 360 } 361 *fr = 0; 362 } else if ((sep = *inputFS) == 0) { /* new: FS="" => 1 char/field */ 363 for (i = 0; *r != '\0'; r += n) { 364 char buf[MB_LEN_MAX + 1]; 365 366 i++; 367 if (i > nfields) 368 growfldtab(i); 369 if (freeable(fldtab[i])) 370 xfree(fldtab[i]->sval); 371 n = mblen(r, MB_LEN_MAX); 372 if (n < 0) 373 n = 1; 374 memcpy(buf, r, n); 375 buf[n] = '\0'; 376 fldtab[i]->sval = tostring(buf); 377 fldtab[i]->tval = FLD | STR; 378 } 379 *fr = 0; 380 } else if (*r != 0) { /* if 0, it's a null field */ 381 /* subtlecase : if length(FS) == 1 && length(RS > 0) 382 * \n is NOT a field separator (cf awk book 61,84). 383 * this variable is tested in the inner while loop. 384 */ 385 int rtest = '\n'; /* normal case */ 386 if (strlen(*RS) > 0) 387 rtest = '\0'; 388 for (;;) { 389 i++; 390 if (i > nfields) 391 growfldtab(i); 392 if (freeable(fldtab[i])) 393 xfree(fldtab[i]->sval); 394 fldtab[i]->sval = fr; 395 fldtab[i]->tval = FLD | STR | DONTFREE; 396 while (*r != sep && *r != rtest && *r != '\0') /* \n is always a separator */ 397 *fr++ = *r++; 398 *fr++ = 0; 399 if (*r++ == 0) 400 break; 401 } 402 *fr = 0; 403 } 404 if (i > nfields) 405 FATAL("record `%.30s...' has too many fields; can't happen", r); 406 cleanfld(i+1, lastfld); /* clean out junk from previous record */ 407 lastfld = i; 408 donefld = true; 409 for (j = 1; j <= lastfld; j++) { 410 double result; 411 412 p = fldtab[j]; 413 if(is_number(p->sval, & result)) { 414 p->fval = result; 415 p->tval |= NUM; 416 } 417 } 418 setfval(nfloc, (Awkfloat) lastfld); 419 donerec = true; /* restore */ 420 if (dbg) { 421 for (j = 0; j <= lastfld; j++) { 422 p = fldtab[j]; 423 printf("field %d (%s): |%s|\n", j, p->nval, p->sval); 424 } 425 } 426 } 427 428 void cleanfld(int n1, int n2) /* clean out fields n1 .. n2 inclusive */ 429 { /* nvals remain intact */ 430 Cell *p; 431 int i; 432 433 for (i = n1; i <= n2; i++) { 434 p = fldtab[i]; 435 if (freeable(p)) 436 xfree(p->sval); 437 p->sval = EMPTY, 438 p->tval = FLD | STR | DONTFREE; 439 } 440 } 441 442 void newfld(int n) /* add field n after end of existing lastfld */ 443 { 444 if (n > nfields) 445 growfldtab(n); 446 cleanfld(lastfld+1, n); 447 lastfld = n; 448 setfval(nfloc, (Awkfloat) n); 449 } 450 451 void setlastfld(int n) /* set lastfld cleaning fldtab cells if necessary */ 452 { 453 if (n < 0) 454 FATAL("cannot set NF to a negative value"); 455 if (n > nfields) 456 growfldtab(n); 457 458 if (lastfld < n) 459 cleanfld(lastfld+1, n); 460 else 461 cleanfld(n+1, lastfld); 462 463 lastfld = n; 464 } 465 466 Cell *fieldadr(int n) /* get nth field */ 467 { 468 if (n < 0) 469 FATAL("trying to access out of range field %d", n); 470 if (n > nfields) /* fields after NF are empty */ 471 growfldtab(n); /* but does not increase NF */ 472 return(fldtab[n]); 473 } 474 475 void growfldtab(int n) /* make new fields up to at least $n */ 476 { 477 int nf = 2 * nfields; 478 size_t s; 479 480 if (n > nf) 481 nf = n; 482 s = (nf+1) * (sizeof (struct Cell *)); /* freebsd: how much do we need? */ 483 if (s / sizeof(struct Cell *) - 1 == (size_t)nf) /* didn't overflow */ 484 fldtab = (Cell **) realloc(fldtab, s); 485 else /* overflow sizeof int */ 486 xfree(fldtab); /* make it null */ 487 if (fldtab == NULL) 488 FATAL("out of space creating %d fields", nf); 489 makefields(nfields+1, nf); 490 nfields = nf; 491 } 492 493 int refldbld(const char *rec, const char *fs) /* build fields from reg expr in FS */ 494 { 495 /* this relies on having fields[] the same length as $0 */ 496 /* the fields are all stored in this one array with \0's */ 497 char *fr; 498 int i, tempstat, n; 499 fa *pfa; 500 501 n = strlen(rec); 502 if (n > fieldssize) { 503 xfree(fields); 504 if ((fields = (char *) malloc(n+1)) == NULL) 505 FATAL("out of space for fields in refldbld %d", n); 506 fieldssize = n; 507 } 508 fr = fields; 509 *fr = '\0'; 510 if (*rec == '\0') 511 return 0; 512 pfa = makedfa(fs, 1); 513 DPRINTF("into refldbld, rec = <%s>, pat = <%s>\n", rec, fs); 514 tempstat = pfa->initstat; 515 for (i = 1; ; i++) { 516 if (i > nfields) 517 growfldtab(i); 518 if (freeable(fldtab[i])) 519 xfree(fldtab[i]->sval); 520 fldtab[i]->tval = FLD | STR | DONTFREE; 521 fldtab[i]->sval = fr; 522 DPRINTF("refldbld: i=%d\n", i); 523 if (nematch(pfa, rec)) { 524 pfa->initstat = 2; /* horrible coupling to b.c */ 525 DPRINTF("match %s (%d chars)\n", patbeg, patlen); 526 strncpy(fr, rec, patbeg-rec); 527 fr += patbeg - rec + 1; 528 *(fr-1) = '\0'; 529 rec = patbeg + patlen; 530 } else { 531 DPRINTF("no match %s\n", rec); 532 strcpy(fr, rec); 533 pfa->initstat = tempstat; 534 break; 535 } 536 } 537 return i; 538 } 539 540 void recbld(void) /* create $0 from $1..$NF if necessary */ 541 { 542 int i; 543 char *r, *p; 544 char *sep = getsval(ofsloc); 545 546 if (donerec) 547 return; 548 r = record; 549 for (i = 1; i <= *NF; i++) { 550 p = getsval(fldtab[i]); 551 if (!adjbuf(&record, &recsize, 1+strlen(p)+r-record, recsize, &r, "recbld 1")) 552 FATAL("created $0 `%.30s...' too long", record); 553 while ((*r = *p++) != 0) 554 r++; 555 if (i < *NF) { 556 if (!adjbuf(&record, &recsize, 2+strlen(sep)+r-record, recsize, &r, "recbld 2")) 557 FATAL("created $0 `%.30s...' too long", record); 558 for (p = sep; (*r = *p++) != 0; ) 559 r++; 560 } 561 } 562 if (!adjbuf(&record, &recsize, 2+r-record, recsize, &r, "recbld 3")) 563 FATAL("built giant record `%.30s...'", record); 564 *r = '\0'; 565 DPRINTF("in recbld inputFS=%s, fldtab[0]=%p\n", inputFS, (void*)fldtab[0]); 566 567 if (freeable(fldtab[0])) 568 xfree(fldtab[0]->sval); 569 fldtab[0]->tval = REC | STR | DONTFREE; 570 fldtab[0]->sval = record; 571 572 DPRINTF("in recbld inputFS=%s, fldtab[0]=%p\n", inputFS, (void*)fldtab[0]); 573 DPRINTF("recbld = |%s|\n", record); 574 donerec = true; 575 } 576 577 int errorflag = 0; 578 579 void yyerror(const char *s) 580 { 581 SYNTAX("%s", s); 582 } 583 584 void SYNTAX(const char *fmt, ...) 585 { 586 extern char *cmdname, *curfname; 587 static int been_here = 0; 588 va_list varg; 589 590 if (been_here++ > 2) 591 return; 592 fprintf(stderr, "%s: ", cmdname); 593 va_start(varg, fmt); 594 vfprintf(stderr, fmt, varg); 595 va_end(varg); 596 fprintf(stderr, " at source line %d", lineno); 597 if (curfname != NULL) 598 fprintf(stderr, " in function %s", curfname); 599 if (compile_time == COMPILING && cursource() != NULL) 600 fprintf(stderr, " source file %s", cursource()); 601 fprintf(stderr, "\n"); 602 errorflag = 2; 603 eprint(); 604 } 605 606 extern int bracecnt, brackcnt, parencnt; 607 608 void bracecheck(void) 609 { 610 int c; 611 static int beenhere = 0; 612 613 if (beenhere++) 614 return; 615 while ((c = input()) != EOF && c != '\0') 616 bclass(c); 617 bcheck2(bracecnt, '{', '}'); 618 bcheck2(brackcnt, '[', ']'); 619 bcheck2(parencnt, '(', ')'); 620 } 621 622 void bcheck2(int n, int c1, int c2) 623 { 624 if (n == 1) 625 fprintf(stderr, "\tmissing %c\n", c2); 626 else if (n > 1) 627 fprintf(stderr, "\t%d missing %c's\n", n, c2); 628 else if (n == -1) 629 fprintf(stderr, "\textra %c\n", c2); 630 else if (n < -1) 631 fprintf(stderr, "\t%d extra %c's\n", -n, c2); 632 } 633 634 void FATAL(const char *fmt, ...) 635 { 636 extern char *cmdname; 637 va_list varg; 638 639 fflush(stdout); 640 fprintf(stderr, "%s: ", cmdname); 641 va_start(varg, fmt); 642 vfprintf(stderr, fmt, varg); 643 va_end(varg); 644 error(); 645 if (dbg > 1) /* core dump if serious debugging on */ 646 abort(); 647 exit(2); 648 } 649 650 void WARNING(const char *fmt, ...) 651 { 652 extern char *cmdname; 653 va_list varg; 654 655 fflush(stdout); 656 fprintf(stderr, "%s: ", cmdname); 657 va_start(varg, fmt); 658 vfprintf(stderr, fmt, varg); 659 va_end(varg); 660 error(); 661 } 662 663 void error() 664 { 665 extern Node *curnode; 666 667 fprintf(stderr, "\n"); 668 if (compile_time != ERROR_PRINTING) { 669 if (NR && *NR > 0) { 670 fprintf(stderr, " input record number %d", (int) (*FNR)); 671 if (strcmp(*FILENAME, "-") != 0) 672 fprintf(stderr, ", file %s", *FILENAME); 673 fprintf(stderr, "\n"); 674 } 675 if (curnode) 676 fprintf(stderr, " source line number %d", curnode->lineno); 677 else if (lineno) 678 fprintf(stderr, " source line number %d", lineno); 679 if (compile_time == COMPILING && cursource() != NULL) 680 fprintf(stderr, " source file %s", cursource()); 681 fprintf(stderr, "\n"); 682 eprint(); 683 } 684 } 685 686 void eprint(void) /* try to print context around error */ 687 { 688 char *p, *q; 689 int c; 690 static int been_here = 0; 691 extern char ebuf[], *ep; 692 693 if (compile_time != COMPILING || been_here++ > 0 || ebuf == ep) 694 return; 695 if (ebuf == ep) 696 return; 697 p = ep - 1; 698 if (p > ebuf && *p == '\n') 699 p--; 700 for ( ; p > ebuf && *p != '\n' && *p != '\0'; p--) 701 ; 702 while (*p == '\n') 703 p++; 704 fprintf(stderr, " context is\n\t"); 705 for (q=ep-1; q>=p && *q!=' ' && *q!='\t' && *q!='\n'; q--) 706 ; 707 for ( ; p < q; p++) 708 if (*p) 709 putc(*p, stderr); 710 fprintf(stderr, " >>> "); 711 for ( ; p < ep; p++) 712 if (*p) 713 putc(*p, stderr); 714 fprintf(stderr, " <<< "); 715 if (*ep) 716 while ((c = input()) != '\n' && c != '\0' && c != EOF) { 717 putc(c, stderr); 718 bclass(c); 719 } 720 putc('\n', stderr); 721 ep = ebuf; 722 } 723 724 void bclass(int c) 725 { 726 switch (c) { 727 case '{': bracecnt++; break; 728 case '}': bracecnt--; break; 729 case '[': brackcnt++; break; 730 case ']': brackcnt--; break; 731 case '(': parencnt++; break; 732 case ')': parencnt--; break; 733 } 734 } 735 736 double errcheck(double x, const char *s) 737 { 738 739 if (errno == EDOM) { 740 errno = 0; 741 WARNING("%s argument out of domain", s); 742 x = 1; 743 } else if (errno == ERANGE) { 744 errno = 0; 745 WARNING("%s result out of range", s); 746 x = 1; 747 } 748 return x; 749 } 750 751 int isclvar(const char *s) /* is s of form var=something ? */ 752 { 753 const char *os = s; 754 755 if (!isalpha((uschar) *s) && *s != '_') 756 return 0; 757 for ( ; *s; s++) 758 if (!(isalnum((uschar) *s) || *s == '_')) 759 break; 760 return *s == '=' && s > os; 761 } 762 763 /* strtod is supposed to be a proper test of what's a valid number */ 764 /* appears to be broken in gcc on linux: thinks 0x123 is a valid FP number */ 765 /* wrong: violates 4.10.1.4 of ansi C standard */ 766 767 /* well, not quite. As of C99, hex floating point is allowed. so this is 768 * a bit of a mess. We work around the mess by checking for a hexadecimal 769 * value and disallowing it. Similarly, we now follow gawk and allow only 770 * +nan, -nan, +inf, and -inf for NaN and infinity values. 771 */ 772 773 /* 774 * This routine now has a more complicated interface, the main point 775 * being to avoid the double conversion of a string to double, and 776 * also to convey out, if requested, the information that the numeric 777 * value was a leading string or is all of the string. The latter bit 778 * is used in getfval(). 779 */ 780 781 bool is_valid_number(const char *s, bool trailing_stuff_ok, 782 bool *no_trailing, double *result) 783 { 784 double r; 785 char *ep; 786 bool retval = false; 787 bool is_nan = false; 788 bool is_inf = false; 789 790 if (no_trailing) 791 *no_trailing = false; 792 793 while (isspace(*s)) 794 s++; 795 796 // no hex floating point, sorry 797 if (s[0] == '0' && tolower(s[1]) == 'x') 798 return false; 799 800 // allow +nan, -nan, +inf, -inf, any other letter, no 801 if (s[0] == '+' || s[0] == '-') { 802 is_nan = (strncasecmp(s+1, "nan", 3) == 0); 803 is_inf = (strncasecmp(s+1, "inf", 3) == 0); 804 if ((is_nan || is_inf) 805 && (isspace(s[4]) || s[4] == '\0')) 806 goto convert; 807 else if (! isdigit(s[1]) && s[1] != '.') 808 return false; 809 } 810 else if (! isdigit(s[0]) && s[0] != '.') 811 return false; 812 813 convert: 814 errno = 0; 815 r = strtod(s, &ep); 816 if (ep == s || errno == ERANGE) 817 return false; 818 819 if (isnan(r) && s[0] == '-' && signbit(r) == 0) 820 r = -r; 821 822 if (result != NULL) 823 *result = r; 824 825 /* 826 * check for trailing stuff 827 */ 828 while (isspace(*ep)) 829 ep++; 830 831 if (no_trailing != NULL) 832 *no_trailing = (*ep == '\0'); 833 834 // return true if found the end, or trailing stuff is allowed 835 retval = *ep == '\0' || trailing_stuff_ok; 836 837 return retval; 838 } 839