Annotation of 43BSDReno/bin/expr/expr.y, revision 1.1.1.1

1.1       root        1: /* Yacc productions for "expr" command: */
                      2: %{
                      3: typedef char *yystype;
                      4: #define        YYSTYPE yystype
                      5: %}
                      6: 
                      7: %token OR AND ADD SUBT MULT DIV REM EQ GT GEQ LT LEQ NEQ
                      8: %token A_STRING SUBSTR LENGTH INDEX NOARG MATCH
                      9: 
                     10: /* operators listed below in increasing precedence: */
                     11: %left OR
                     12: %left AND
                     13: %left EQ LT GT GEQ LEQ NEQ
                     14: %left ADD SUBT
                     15: %left MULT DIV REM
                     16: %left MCH
                     17: %left MATCH
                     18: %left SUBSTR
                     19: %left LENGTH INDEX
                     20: %%
                     21: 
                     22: /* a single `expression' is evaluated and printed: */
                     23: 
                     24: expression:    expr NOARG = {
                     25:                        printf("%s\n", $1);
                     26:                        exit((!strcmp($1,"0")||!strcmp($1,"\0"))? 1: 0);
                     27:                        }
                     28:        ;
                     29: 
                     30: 
                     31: expr:  '(' expr ')' = { $$ = $2; }
                     32:        | expr OR expr   = { $$ = conj(OR, $1, $3); }
                     33:        | expr AND expr   = { $$ = conj(AND, $1, $3); }
                     34:        | expr EQ expr   = { $$ = rel(EQ, $1, $3); }
                     35:        | expr GT expr   = { $$ = rel(GT, $1, $3); }
                     36:        | expr GEQ expr   = { $$ = rel(GEQ, $1, $3); }
                     37:        | expr LT expr   = { $$ = rel(LT, $1, $3); }
                     38:        | expr LEQ expr   = { $$ = rel(LEQ, $1, $3); }
                     39:        | expr NEQ expr   = { $$ = rel(NEQ, $1, $3); }
                     40:        | expr ADD expr   = { $$ = arith(ADD, $1, $3); }
                     41:        | expr SUBT expr   = { $$ = arith(SUBT, $1, $3); }
                     42:        | expr MULT expr   = { $$ = arith(MULT, $1, $3); }
                     43:        | expr DIV expr   = { $$ = arith(DIV, $1, $3); }
                     44:        | expr REM expr   = { $$ = arith(REM, $1, $3); }
                     45:        | expr MCH expr  = { $$ = match($1, $3); }
                     46:        | MATCH expr expr = { $$ = match($2, $3); }
                     47:        | SUBSTR expr expr expr = { $$ = substr($2, $3, $4); }
                     48:        | LENGTH expr       = { $$ = length($2); }
                     49:        | INDEX expr expr = { $$ = index($2, $3); }
                     50:        | A_STRING
                     51:        ;
                     52: %%
                     53: /*     expression command */
                     54: #include <stdio.h>
                     55: #define ESIZE  256
                     56: #define error(c)       errxx(c)
                     57: #define EQL(x,y) !strcmp(x,y)
                     58: long atol();
                     59: char   **Av;
                     60: int    Ac;
                     61: int    Argi;
                     62: 
                     63: char Mstring[1][128];
                     64: char *malloc();
                     65: extern int nbra;
                     66: 
                     67: main(argc, argv) char **argv; {
                     68:        Ac = argc;
                     69:        Argi = 1;
                     70:        Av = argv;
                     71:        yyparse();
                     72: }
                     73: 
                     74: char *operators[] = { "|", "&", "+", "-", "*", "/", "%", ":",
                     75:        "=", "==", "<", "<=", ">", ">=", "!=",
                     76:        "match", "substr", "length", "index", "\0" };
                     77: int op[] = { OR, AND, ADD,  SUBT, MULT, DIV, REM, MCH,
                     78:        EQ, EQ, LT, LEQ, GT, GEQ, NEQ,
                     79:        MATCH, SUBSTR, LENGTH, INDEX };
                     80: yylex() {
                     81:        register char *p;
                     82:        register i;
                     83: 
                     84:        if(Argi >= Ac) return NOARG;
                     85: 
                     86:        p = Av[Argi++];
                     87: 
                     88:        if(*p == '(' || *p == ')')
                     89:                return (int)*p;
                     90:        for(i = 0; *operators[i]; ++i)
                     91:                if(EQL(operators[i], p))
                     92:                        return op[i];
                     93: 
                     94:        yylval = p;
                     95:        return A_STRING;
                     96: }
                     97: 
                     98: char *rel(op, r1, r2) register char *r1, *r2; {
                     99:        register long i;
                    100: 
                    101:        if(ematch(r1, "-*[0-9]*$") && ematch(r2, "[0-9]*$"))
                    102:                i = atol(r1) - atol(r2);
                    103:        else
                    104:                i = strcmp(r1, r2);
                    105:        switch(op) {
                    106:        case EQ: i = i==0; break;
                    107:        case GT: i = i>0; break;
                    108:        case GEQ: i = i>=0; break;
                    109:        case LT: i = i<0; break;
                    110:        case LEQ: i = i<=0; break;
                    111:        case NEQ: i = i!=0; break;
                    112:        }
                    113:        return i? "1": "0";
                    114: }
                    115: 
                    116: char *arith(op, r1, r2) char *r1, *r2; {
                    117:        long i1, i2;
                    118:        register char *rv;
                    119: 
                    120:        if(!((ematch(r1, "[0-9]*$") || ematch(r1, "-[0-9]*$")) &&
                    121:             (ematch(r2, "[0-9]*$") || ematch(r2, "-[0-9]*$"))))
                    122:                yyerror("non-numeric argument");
                    123:        i1 = atol(r1);
                    124:        i2 = atol(r2);
                    125: 
                    126:        switch(op) {
                    127:        case ADD: i1 = i1 + i2; break;
                    128:        case SUBT: i1 = i1 - i2; break;
                    129:        case MULT: i1 = i1 * i2; break;
                    130:        case DIV: i1 = i1 / i2; break;
                    131:        case REM: i1 = i1 % i2; break;
                    132:        }
                    133:        rv = malloc(16);
                    134:        (void)sprintf(rv, "%ld", i1);
                    135:        return rv;
                    136: }
                    137: char *conj(op, r1, r2) char *r1, *r2; {
                    138:        register char *rv;
                    139: 
                    140:        switch(op) {
                    141: 
                    142:        case OR:
                    143:                if(EQL(r1, "0")
                    144:                || EQL(r1, ""))
                    145:                        if(EQL(r2, "0")
                    146:                        || EQL(r2, ""))
                    147:                                rv = "0";
                    148:                        else
                    149:                                rv = r2;
                    150:                else
                    151:                        rv = r1;
                    152:                break;
                    153:        case AND:
                    154:                if(EQL(r1, "0")
                    155:                || EQL(r1, ""))
                    156:                        rv = "0";
                    157:                else if(EQL(r2, "0")
                    158:                || EQL(r2, ""))
                    159:                        rv = "0";
                    160:                else
                    161:                        rv = r1;
                    162:                break;
                    163:        }
                    164:        return rv;
                    165: }
                    166: 
                    167: char *substr(v, s, w) char *v, *s, *w; {
                    168: register si, wi;
                    169: register char *res;
                    170: 
                    171:        si = atol(s);
                    172:        wi = atol(w);
                    173:        while(--si) if(*v) ++v;
                    174: 
                    175:        res = v;
                    176: 
                    177:        while(wi--) if(*v) ++v;
                    178: 
                    179:        *v = '\0';
                    180:        return res;
                    181: }
                    182: 
                    183: char *length(s) register char *s; {
                    184:        register i = 0;
                    185:        register char *rv;
                    186: 
                    187:        while(*s++) ++i;
                    188: 
                    189:        rv = malloc(8);
                    190:        (void)sprintf(rv, "%d", i);
                    191:        return rv;
                    192: }
                    193: 
                    194: char *index(s, t) char *s, *t; {
                    195:        register i, j;
                    196:        register char *rv;
                    197: 
                    198:        for(i = 0; s[i] ; ++i)
                    199:                for(j = 0; t[j] ; ++j)
                    200:                        if(s[i]==t[j]) {
                    201:                                (void)sprintf(rv = malloc(8), "%d", ++i);
                    202:                                return rv;
                    203:                        }
                    204:        return "0";
                    205: }
                    206: 
                    207: char *match(s, p)
                    208: {
                    209:        register char *rv;
                    210: 
                    211:        (void)sprintf(rv = malloc(8), "%d", ematch(s, p));
                    212:        if(nbra) {
                    213:                rv = malloc(strlen(Mstring[0])+1);
                    214:                strcpy(rv, Mstring[0]);
                    215:        }
                    216:        return rv;
                    217: }
                    218: 
                    219: #define INIT   register char *sp = instring;
                    220: #define GETC()         (*sp++)
                    221: #define PEEKC()                (*sp)
                    222: #define UNGETC(c)      (--sp)
                    223: #define RETURN(c)      return
                    224: #define ERROR(c)       errxx(c)
                    225: 
                    226: 
                    227: ematch(s, p)
                    228: char *s;
                    229: register char *p;
                    230: {
                    231:        static char expbuf[ESIZE];
                    232:        char *compile();
                    233:        register num;
                    234:        extern char *braslist[], *braelist[], *loc2;
                    235: 
                    236:        compile(p, expbuf, &expbuf[ESIZE], 0);
                    237:        if(nbra > 1)
                    238:                yyerror("Too many '\\('s");
                    239:        if(advance(s, expbuf)) {
                    240:                if(nbra == 1) {
                    241:                        p = braslist[0];
                    242:                        num = braelist[0] - p;
                    243:                        strncpy(Mstring[0], p, num);
                    244:                        Mstring[0][num] = '\0';
                    245:                }
                    246:                return(loc2-s);
                    247:        }
                    248:        return(0);
                    249: }
                    250: 
                    251: errxx(c)
                    252: {
                    253:        yyerror("RE error");
                    254: }
                    255: 
                    256: #define        CBRA    2
                    257: #define        CCHR    4
                    258: #define        CDOT    8
                    259: #define        CCL     12
                    260: #define        CDOL    20
                    261: #define        CEOF    22
                    262: #define        CKET    24
                    263: #define        CBACK   36
                    264: 
                    265: #define        STAR    01
                    266: #define RNGE   03
                    267: 
                    268: #define        NBRA    9
                    269: 
                    270: #define PLACE(c)       ep[c >> 3] |= bittab[c & 07]
                    271: #define ISTHERE(c)     (ep[c >> 3] & bittab[c & 07])
                    272: 
                    273: char   *braslist[NBRA];
                    274: char   *braelist[NBRA];
                    275: int    nbra;
                    276: char *loc1, *loc2, *locs;
                    277: int    sed;
                    278: 
                    279: int    circf;
                    280: int    low;
                    281: int    size;
                    282: 
                    283: char   bittab[] = {
                    284:        1,
                    285:        2,
                    286:        4,
                    287:        8,
                    288:        16,
                    289:        32,
                    290:        64,
                    291:        128
                    292: };
                    293: 
                    294: char *
                    295: compile(instring, ep, endbuf, seof)
                    296: register char *ep;
                    297: char *instring, *endbuf;
                    298: {
                    299:        INIT    /* Dependent declarations and initializations */
                    300:        register c;
                    301:        register eof = seof;
                    302:        char *lastep = instring;
                    303:        int cclcnt;
                    304:        char bracket[NBRA], *bracketp;
                    305:        int closed;
                    306:        char neg;
                    307:        int lc;
                    308:        int i, cflg;
                    309: 
                    310:        lastep = 0;
                    311:        if((c = GETC()) == eof) {
                    312:                if(*ep == 0 && !sed)
                    313:                        ERROR(41);
                    314:                RETURN(ep);
                    315:        }
                    316:        bracketp = bracket;
                    317:        circf = closed = nbra = 0;
                    318:        if (c == '^')
                    319:                circf++;
                    320:        else
                    321:                UNGETC(c);
                    322:        for (;;) {
                    323:                if (ep >= endbuf)
                    324:                        ERROR(50);
                    325:                if((c = GETC()) != '*' && ((c != '\\') || (PEEKC() != '{')))
                    326:                        lastep = ep;
                    327:                if (c == eof) {
                    328:                        *ep++ = CEOF;
                    329:                        RETURN(ep);
                    330:                }
                    331:                switch (c) {
                    332: 
                    333:                case '.':
                    334:                        *ep++ = CDOT;
                    335:                        continue;
                    336: 
                    337:                case '\n':
                    338:                        ERROR(36);
                    339:                case '*':
                    340:                        if (lastep==0 || *lastep==CBRA || *lastep==CKET)
                    341:                                goto defchar;
                    342:                        *lastep |= STAR;
                    343:                        continue;
                    344: 
                    345:                case '$':
                    346:                        if(PEEKC() != eof)
                    347:                                goto defchar;
                    348:                        *ep++ = CDOL;
                    349:                        continue;
                    350: 
                    351:                case '[':
                    352:                        if(&ep[17] >= endbuf)
                    353:                                ERROR(50);
                    354: 
                    355:                        *ep++ = CCL;
                    356:                        lc = 0;
                    357:                        for(i = 0; i < 16; i++)
                    358:                                ep[i] = 0;
                    359: 
                    360:                        neg = 0;
                    361:                        if((c = GETC()) == '^') {
                    362:                                neg = 1;
                    363:                                c = GETC();
                    364:                        }
                    365: 
                    366:                        do {
                    367:                                if(c == '\0' || c == '\n')
                    368:                                        ERROR(49);
                    369:                                if(c == '-' && lc != 0) {
                    370:                                        if ((c = GETC()) == ']') {
                    371:                                                PLACE('-');
                    372:                                                break;
                    373:                                        }
                    374:                                        while(lc < c) {
                    375:                                                PLACE(lc);
                    376:                                                lc++;
                    377:                                        }
                    378:                                }
                    379:                                lc = c;
                    380:                                PLACE(c);
                    381:                        } while((c = GETC()) != ']');
                    382:                        if(neg) {
                    383:                                for(cclcnt = 0; cclcnt < 16; cclcnt++)
                    384:                                        ep[cclcnt] ^= -1;
                    385:                                ep[0] &= 0376;
                    386:                        }
                    387: 
                    388:                        ep += 16;
                    389: 
                    390:                        continue;
                    391: 
                    392:                case '\\':
                    393:                        switch(c = GETC()) {
                    394: 
                    395:                        case '(':
                    396:                                if(nbra >= NBRA)
                    397:                                        ERROR(43);
                    398:                                *bracketp++ = nbra;
                    399:                                *ep++ = CBRA;
                    400:                                *ep++ = nbra++;
                    401:                                continue;
                    402: 
                    403:                        case ')':
                    404:                                if(bracketp <= bracket)
                    405:                                        ERROR(42);
                    406:                                *ep++ = CKET;
                    407:                                *ep++ = *--bracketp;
                    408:                                closed++;
                    409:                                continue;
                    410: 
                    411:                        case '{':
                    412:                                if(lastep == (char *) (0))
                    413:                                        goto defchar;
                    414:                                *lastep |= RNGE;
                    415:                                cflg = 0;
                    416:                        nlim:
                    417:                                c = GETC();
                    418:                                i = 0;
                    419:                                do {
                    420:                                        if ('0' <= c && c <= '9')
                    421:                                                i = 10 * i + c - '0';
                    422:                                        else
                    423:                                                ERROR(16);
                    424:                                } while(((c = GETC()) != '\\') && (c != ','));
                    425:                                if (i > 255)
                    426:                                        ERROR(11);
                    427:                                *ep++ = i;
                    428:                                if (c == ',') {
                    429:                                        if(cflg++)
                    430:                                                ERROR(44);
                    431:                                        if((c = GETC()) == '\\')
                    432:                                                *ep++ = 255;
                    433:                                        else {
                    434:                                                UNGETC(c);
                    435:                                                goto nlim; /* get 2'nd number */
                    436:                                        }
                    437:                                }
                    438:                                if(GETC() != '}')
                    439:                                        ERROR(45);
                    440:                                if(!cflg)       /* one number */
                    441:                                        *ep++ = i;
                    442:                                else if((ep[-1] & 0377) < (ep[-2] & 0377))
                    443:                                        ERROR(46);
                    444:                                continue;
                    445: 
                    446:                        case '\n':
                    447:                                ERROR(36);
                    448: 
                    449:                        case 'n':
                    450:                                c = '\n';
                    451:                                goto defchar;
                    452: 
                    453:                        default:
                    454:                                if(c >= '1' && c <= '9') {
                    455:                                        if((c -= '1') >= closed)
                    456:                                                ERROR(25);
                    457:                                        *ep++ = CBACK;
                    458:                                        *ep++ = c;
                    459:                                        continue;
                    460:                                }
                    461:                        }
                    462:                        /* Drop through to default to use \ to turn off special chars */
                    463: 
                    464:                defchar:
                    465:                default:
                    466:                        lastep = ep;
                    467:                        *ep++ = CCHR;
                    468:                        *ep++ = c;
                    469:                }
                    470:        }
                    471: }
                    472: 
                    473: step(p1, p2)
                    474: register char *p1, *p2;
                    475: {
                    476:        register c;
                    477: 
                    478:        if (circf) {
                    479:                loc1 = p1;
                    480:                return(advance(p1, p2));
                    481:        }
                    482:        /* fast check for first character */
                    483:        if (*p2==CCHR) {
                    484:                c = p2[1];
                    485:                do {
                    486:                        if (*p1 != c)
                    487:                                continue;
                    488:                        if (advance(p1, p2)) {
                    489:                                loc1 = p1;
                    490:                                return(1);
                    491:                        }
                    492:                } while (*p1++);
                    493:                return(0);
                    494:        }
                    495:                /* regular algorithm */
                    496:        do {
                    497:                if (advance(p1, p2)) {
                    498:                        loc1 = p1;
                    499:                        return(1);
                    500:                }
                    501:        } while (*p1++);
                    502:        return(0);
                    503: }
                    504: 
                    505: advance(lp, ep)
                    506: register char *lp, *ep;
                    507: {
                    508:        register char *curlp;
                    509:        char c;
                    510:        char *bbeg;
                    511:        int ct;
                    512: 
                    513:        for (;;) switch (*ep++) {
                    514: 
                    515:        case CCHR:
                    516:                if (*ep++ == *lp++)
                    517:                        continue;
                    518:                return(0);
                    519: 
                    520:        case CDOT:
                    521:                if (*lp++)
                    522:                        continue;
                    523:                return(0);
                    524: 
                    525:        case CDOL:
                    526:                if (*lp==0)
                    527:                        continue;
                    528:                return(0);
                    529: 
                    530:        case CEOF:
                    531:                loc2 = lp;
                    532:                return(1);
                    533: 
                    534:        case CCL:
                    535:                c = *lp++ & 0177;
                    536:                if(ISTHERE(c)) {
                    537:                        ep += 16;
                    538:                        continue;
                    539:                }
                    540:                return(0);
                    541:        case CBRA:
                    542:                braslist[*ep++] = lp;
                    543:                continue;
                    544: 
                    545:        case CKET:
                    546:                braelist[*ep++] = lp;
                    547:                continue;
                    548: 
                    549:        case CCHR|RNGE:
                    550:                c = *ep++;
                    551:                getrnge(ep);
                    552:                while(low--)
                    553:                        if(*lp++ != c)
                    554:                                return(0);
                    555:                curlp = lp;
                    556:                while(size--) 
                    557:                        if(*lp++ != c)
                    558:                                break;
                    559:                if(size < 0)
                    560:                        lp++;
                    561:                ep += 2;
                    562:                goto star;
                    563: 
                    564:        case CDOT|RNGE:
                    565:                getrnge(ep);
                    566:                while(low--)
                    567:                        if(*lp++ == '\0')
                    568:                                return(0);
                    569:                curlp = lp;
                    570:                while(size--)
                    571:                        if(*lp++ == '\0')
                    572:                                break;
                    573:                if(size < 0)
                    574:                        lp++;
                    575:                ep += 2;
                    576:                goto star;
                    577: 
                    578:        case CCL|RNGE:
                    579:                getrnge(ep + 16);
                    580:                while(low--) {
                    581:                        c = *lp++ & 0177;
                    582:                        if(!ISTHERE(c))
                    583:                                return(0);
                    584:                }
                    585:                curlp = lp;
                    586:                while(size--) {
                    587:                        c = *lp++ & 0177;
                    588:                        if(!ISTHERE(c))
                    589:                                break;
                    590:                }
                    591:                if(size < 0)
                    592:                        lp++;
                    593:                ep += 18;               /* 16 + 2 */
                    594:                goto star;
                    595: 
                    596:        case CBACK:
                    597:                bbeg = braslist[*ep];
                    598:                ct = braelist[*ep++] - bbeg;
                    599: 
                    600:                if(ecmp(bbeg, lp, ct)) {
                    601:                        lp += ct;
                    602:                        continue;
                    603:                }
                    604:                return(0);
                    605: 
                    606:        case CBACK|STAR:
                    607:                bbeg = braslist[*ep];
                    608:                ct = braelist[*ep++] - bbeg;
                    609:                curlp = lp;
                    610:                while(ecmp(bbeg, lp, ct))
                    611:                        lp += ct;
                    612: 
                    613:                while(lp >= curlp) {
                    614:                        if(advance(lp, ep))     return(1);
                    615:                        lp -= ct;
                    616:                }
                    617:                return(0);
                    618: 
                    619: 
                    620:        case CDOT|STAR:
                    621:                curlp = lp;
                    622:                while (*lp++);
                    623:                goto star;
                    624: 
                    625:        case CCHR|STAR:
                    626:                curlp = lp;
                    627:                while (*lp++ == *ep);
                    628:                ep++;
                    629:                goto star;
                    630: 
                    631:        case CCL|STAR:
                    632:                curlp = lp;
                    633:                do {
                    634:                        c = *lp++ & 0177;
                    635:                } while(ISTHERE(c));
                    636:                ep += 16;
                    637:                goto star;
                    638: 
                    639:        star:
                    640:                do {
                    641:                        if(--lp == locs)
                    642:                                break;
                    643:                        if (advance(lp, ep))
                    644:                                return(1);
                    645:                } while (lp > curlp);
                    646:                return(0);
                    647: 
                    648:        }
                    649: }
                    650: 
                    651: getrnge(str)
                    652: register char *str;
                    653: {
                    654:        low = *str++ & 0377;
                    655:        size = (*str & 0377) == 255 ? 20000 : (*str & 0377) - low;
                    656: }
                    657: 
                    658: ecmp(a, b, count)
                    659: register char  *a, *b;
                    660: register       count;
                    661: {
                    662:        if(a == b) /* should have been caught in compile() */
                    663:                error(51);
                    664:        while(count--)
                    665:                if(*a++ != *b++)        return(0);
                    666:        return(1);
                    667: }
                    668: 
                    669: static char *sccsid = "@(#)expr.y      4.8 (Berkeley) 6/24/90";
                    670: yyerror(s)
                    671: 
                    672: {
                    673:        fprintf(stderr, "%s\n", s);
                    674:        exit(2);
                    675: }

unix.superglobalmegacorp.com

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