Annotation of researchv10no/cmd/ccom/common/trees.c, revision 1.1.1.1

1.1       root        1: /*     @(#) trees.c: 1.3 3/4/84        */
                      2: 
                      3: # include "mfile1.h"
                      4: 
                      5: /*     some special actions, used in finding the type of nodes */
                      6: # define NCVT 01
                      7: # define PUN 02
                      8: # define TYPL 04
                      9: # define TYPR 010
                     10: # define TYMATCH 040
                     11: # define LVAL 0100
                     12: # define CVTO 0200
                     13: # define CVTL 0400
                     14: # define CVTR 01000
                     15: # define PTMATCH 02000
                     16: # define OTHER 04000
                     17: # define NCVTR 010000
                     18: 
                     19: /* node conventions:
                     20: **
                     21: ** NAME:       rval>0 is stab index for external
                     22: ** rval<0 is -inlabel number
                     23: ** lval is offset in address units
                     24: ** (NAME really means "STATIC VARIABLE")
                     25: ** ICON:       lval has the value
                     26: ** rval has the STAB index, or - label number,
                     27: ** if a name whose address is in the constant
                     28: ** rval = NONAME means no name
                     29: ** VAUTO:      automatic name: lval has offset in address units
                     30: ** VPARAM:     parameter: lval has offset in address units
                     31: ** REG:        rval is reg. number
                     32: **
                     33: */
                     34: extern int maxarg;
                     35: int onlywarn;
                     36: 
                     37: TWORD
                     38: indtype( t )
                     39: register TWORD t;
                     40: {
                     41:        /* return the type of an intermediate expression of type t */
                     42:        switch( t )
                     43:        {
                     44: 
                     45:        case CHAR:
                     46:        case SHORT:
                     47:                return( INT );
                     48: 
                     49:        case UCHAR:
                     50:        case USHORT:
                     51:                return( UNSIGNED );
                     52: 
                     53: /*rick: no!    case FLOAT:
                     54: /*             return( DOUBLE );
                     55: */
                     56: 
                     57:        }
                     58:        return( t );
                     59: }
                     60: 
                     61: int bdebug = 0;
                     62: extern ddebug;
                     63: 
                     64: # ifndef XI
                     65: NODE *
                     66: xicolon( l, r1, r2 ) register NODE *l, *r1, *r2;
                     67: 
                     68: {
                     69:        uerror( "syntax error: colon in subscript" );
                     70:        tfree( r2 );
                     71:        return( buildtree( LB, l, r1 ) );
                     72: }
                     73: # endif
                     74: 
                     75: NODE *
                     76: buildtree( o, l, r )
                     77: register NODE *l, *r;
                     78: register o;
                     79: {
                     80:        register NODE *p, *q;
                     81:        register struct symtab *sp;
                     82:        register NODE *lr, *ll;
                     83:        register actions;
                     84:        register opty;
                     85:        int     i, tlineno;
                     86: 
                     87: # ifndef NODBG
                     88:        if( bdebug )
                     89:                printf( "buildtree( %s, %d, %d )\n", opst[o], l-node, r-node );
                     90: # endif
                     91: 
                     92:        /* special case to recognize subscripting explicitly */
                     93: 
                     94: # ifdef XI
                     95:        if( o == LB && l->tn.type == LONG )
                     96:                return( xicolon( l, r, (NODE *) 0 ) );
                     97: # endif
                     98: 
                     99:        if( o == LB ) return( buildtree( STAR, buildtree( PLUS, l, r ), NIL ) );
                    100:        opty = optype(o);
                    101: 
                    102:        /* check for constants */
                    103: 
                    104:        if( o == NOT && l->in.op == ICON && hflag )
                    105:        {
                    106:                werror( "constant argument to NOT" );
                    107:        }
                    108: 
                    109:        else if( o==UNARY MINUS && l->in.op==FCON )
                    110:        {
                    111:                l->fpn.dval = -l->fpn.dval;
                    112:                return(l);
                    113:        }
                    114: 
                    115:        else if( o==QUEST && l->in.op==ICON && l->tn.rval==NONAME )
                    116:        {
                    117:                l->in.op = FREE;
                    118:                r->in.op = FREE;
                    119:                if( l->tn.lval )
                    120:                {
                    121:                        tfree( r->in.right );
                    122:                        return( r->in.left );
                    123:                }
                    124:                else
                    125:                {
                    126:                        tfree( r->in.left );
                    127:                        return( r->in.right );
                    128:                }
                    129:        }
                    130: 
                    131:        else if( (o==ANDAND || o==OROR) && (l->in.op==ICON||r->in.op==ICON) )
                    132:                goto ccwarn;
                    133: 
                    134:        else if( opty == BITYPE && l->in.op == ICON && r->in.op == ICON )
                    135:        {
                    136: 
                    137:                switch( o )
                    138:                {
                    139: 
                    140:                case ULT:
                    141:                case UGT:
                    142:                case ULE:
                    143:                case UGE:
                    144:                case LT:
                    145:                case GT:
                    146:                case LE:
                    147:                case GE:
                    148:                case EQ:
                    149:                case NE:
                    150:                case ANDAND:
                    151:                case OROR:
                    152:                case CBRANCH:
                    153: 
                    154: ccwarn:
                    155:                        if( hflag ) werror( "constant in conditional context" );
                    156:                }
                    157:        }
                    158:        if (o == CBRANCH || o == QUEST) switch (BTYPE(l->in.type)) {
                    159:        case STRTY:
                    160:        case UNIONTY:
                    161:                if (!ISPTR(l->in.type))
                    162:                        uerror("struct/union in conditional");
                    163:        case ENUMTY:
                    164:                break;
                    165:        }
                    166: 
                    167:        /* we make a real node, and look for shrinking later */
                    168: 
                    169:        tlineno = lineno;
                    170:        if (l && l->ln.lineno && l->ln.lineno < tlineno)
                    171:                tlineno = l->ln.lineno;
                    172:        if (r && r->ln.lineno && r->ln.lineno < tlineno)
                    173:                tlineno = r->ln.lineno;
                    174:        p = block( o, l, r, INT, 0, INT );
                    175: 
                    176:        actions = opact(p);
                    177: # ifdef MYOPACT
                    178:        actions = MYOPACT(p,actions);
                    179: # endif
                    180: 
                    181:        if( actions&LVAL )
                    182:        {
                    183:                 /* check left descendent */
                    184:                if( notlval(p->in.left) )
                    185:                {
                    186:                        uerror( "illegal lhs of assignment operator" );
                    187:                }
                    188:        }
                    189: 
                    190:        if( actions & NCVTR )
                    191:        {
                    192:                p->in.left = pconvert( p->in.left );
                    193:        }
                    194:        else if( !(actions & NCVT ) )
                    195:        {
                    196:                switch( opty )
                    197:                {
                    198: 
                    199:                case BITYPE:
                    200:                        p->in.right = pconvert( p->in.right );
                    201:                case UTYPE:
                    202:                        if( !(actions&LVAL) )
                    203:                                p->in.left = pconvert( p->in.left );
                    204: 
                    205:                }
                    206:        }
                    207: 
                    208:        if( (actions&PUN) && (o!=CAST||cflag) )
                    209:        {
                    210:                chkpun(p);
                    211:        }
                    212: 
                    213:        if( actions & (TYPL|TYPR) )
                    214:        {
                    215: 
                    216:                q = (actions&TYPL) ? p->in.left : p->in.right;
                    217: 
                    218:                p->in.type = q->in.type;
                    219:                p->fn.cdim = q->fn.cdim;
                    220:                p->fn.csiz = q->fn.csiz;
                    221:        }
                    222: 
                    223:        if( actions & CVTL ) p = convert( p, CVTL );
                    224:        if( actions & CVTR ) p = convert( p, CVTR );
                    225: 
                    226:        if( actions & TYMATCH ) p = tymatch(p);
                    227:        if( actions & PTMATCH ) p = ptmatch(p);
                    228: 
                    229:        if( actions & SPFLG ) p = clocal(p);
                    230: 
                    231:        if( actions & OTHER )
                    232:        {
                    233:                l = p->in.left;
                    234:                r = p->in.right;
                    235: 
                    236:                switch(o)
                    237:                {
                    238: 
                    239:                case NAME:
                    240:                        sp = &stab[idname];
                    241:                        if( sp->stype == UNDEF )
                    242:                        {
                    243:                                uerror( "%s undefined", sp->sname );
                    244:                                /* make p look reasonable */
                    245:                                p->in.type = p->fn.cdim = p->fn.csiz = INT;
                    246:                                p->tn.rval = idname;
                    247:                                p->tn.lval = 0;
                    248:                                defid( p, SNULL );
                    249:                                break;
                    250:                        }
                    251: #if defined(M32B) && defined(IMPREGAL)
                    252:                        p->in.pad[0] = '@';     /*mark as un-raname()'d*/
                    253: #endif
                    254:                        p->in.type = sp->stype;
                    255:                        p->fn.cdim = sp->dimoff;
                    256:                        p->fn.csiz = sp->sizoff;
                    257:                        /* special case: MOETY is really an ICON... */
                    258:                        if( p->in.type == MOETY )
                    259:                        {
                    260:                                p->tn.rval = NONAME;
                    261:                                p->tn.lval = sp->offset;
                    262:                                p->fn.cdim = 0;
                    263:                                p->in.type = ENUMTY;
                    264:                                p->in.op = ICON;
                    265:                        }
                    266:                        else
                    267:                        {
                    268:                                switch( sp->sclass )
                    269:                                {
                    270: 
                    271:                                case AUTO:
                    272:                                        p->in.op = VAUTO;
                    273:                                        p->tn.rval = NONAME;
                    274:                                        p->tn.lval = BITOOR(sp->offset);
                    275:                                        break;
                    276: 
                    277:                                case PARAM:
                    278:                                        p->in.op = VPARAM;
                    279:                                        p->tn.rval = NONAME;
                    280:                                        p->tn.lval = BITOOR(sp->offset);
                    281:                                        break;
                    282: 
                    283:                                case REGISTER:
                    284:                                        p->in.op = REG;
                    285:                                        p->tn.lval = 0;
                    286:                                        p->tn.rval = sp->offset;
                    287:                                        break;
                    288: 
                    289:                                case ULABEL:
                    290:                                case LABEL:
                    291:                                case STATIC:
                    292:                                        if( sp->slevel != 0 )
                    293:                                        {
                    294:                                                p->tn.lval = 0;
                    295:                                                p->tn.rval = -sp->offset;
                    296:                                                break;
                    297:                                        }
                    298:                                        /* FALLTHRU */
                    299: 
                    300:                                default:
                    301:                                        p->tn.lval = 0;
                    302:                                        p->tn.rval = idname;
                    303:                                }
                    304:                        }
                    305:                        break;
                    306: 
                    307:                case ICON:
                    308:                        p->in.type = INT;
                    309:                        p->fn.cdim = 0;
                    310:                        p->fn.csiz = INT;
                    311:                        break;
                    312: 
                    313:                case STRING:
                    314:                        p->in.op = NAME;
                    315:                        p->in.type = CHAR+ARY;
                    316:                        p->tn.lval = 0;
                    317:                        p->tn.rval = NOLAB;
                    318:                        p->fn.cdim = curdim;
                    319:                        p->fn.csiz = CHAR;
                    320:                        break;
                    321: 
                    322:                case FCON:
                    323:                        p->tn.lval = 0;
                    324:                        p->tn.rval = 0;
                    325:                        p->in.type = DOUBLE;
                    326:                        p->fn.cdim = 0;
                    327:                        p->fn.csiz = DOUBLE;
                    328: #ifdef MYFCON
                    329:                        MYFCON(p);
                    330: #endif
                    331:                        break;
                    332: 
                    333:                case STREF:
                    334:                        /* p->x turned into *(p+offset) */
                    335:                        /* rhs must be a name; check correctness */
                    336: 
                    337:                        i = r->tn.rval;
                    338:                        if(i<0 || i>=SYMTSZ || ((sp= &stab[i])->sclass != MOS &&
                    339:                            sp->sclass != MOU && !(sp->sclass&FIELD)) )
                    340:                        {
                    341:                                uerror( "%s not struct/union member",
                    342:                                        sp->sname);
                    343:                                break;
                    344:                        }
                    345:                        else
                    346:                                /* if this name is non-unique, find right one */
                    347:                                if( sp->sflags&SNONUNIQ && (l->in.type==PTR+STRTY ||
                    348:                                    l->in.type == PTR+UNIONTY) &&
                    349:                                    (l->fn.csiz+1) >= 0 )
                    350:                        {
                    351:                                /* nonunique name && structure defined */
                    352:                                char * memnam, * tabnam;
                    353:                                register j;
                    354:                                int memi;
                    355:                                j=dimtab[l->fn.csiz+1];
                    356:                                for( ; (memi=dimtab[j]) >= 0; ++j )
                    357:                                {
                    358:                                        tabnam = stab[memi].sname;
                    359:                                        memnam = stab[i].sname;
                    360: # ifndef NODBG
                    361:                                        if( ddebug>1 )
                    362:                                        {
                    363:                                                printf("member %s==%s?\n",
                    364:                                                        memnam, tabnam);
                    365:                                        }
                    366: # endif
                    367:                                        if( stab[memi].sflags & SNONUNIQ )
                    368:                                        {
                    369:                                                if ( memnam != tabnam )
                    370:                                                        continue;
                    371:                                                r->tn.rval = i = memi;
                    372:                                                break;
                    373:                                        }
                    374:                                        continue;
                    375:                                }
                    376:                                if( memi < 0 )
                    377:                                        uerror("%s not in this struct/union",
                    378:                                        stab[i].sname);
                    379:                        }
                    380:                        else
                    381:                        {
                    382:                                register j;
                    383:                                if( l->in.type != PTR+STRTY &&
                    384:                                    l->in.type != PTR+UNIONTY )
                    385:                                {
                    386:                                        if( stab[i].sflags & SNONUNIQ )
                    387:                                        {
                    388:                                                uerror(
                    389:                                                "nonunique name (%s) demands struct/union or struct/union pointer", stab[i].sname);
                    390:                                        }
                    391:                                        else uerror("%s needs reference to struct/union", sp->sname);
                    392:                                }
                    393:                                else if( (j=l->fn.csiz+1)<0 )
                    394:                                        cerror( "undefined structure or union");
                    395:                                else if( !chkstr( i, dimtab[j],
                    396:                                        DECREF(l->in.type) ) )
                    397:                                {
                    398:                                        uerror("%s not in this struct/union",
                    399:                                                sp->sname );
                    400:                                }
                    401:                        }
                    402:                        p = stref( p );
                    403:                        break;
                    404: 
                    405:                case STAR:
                    406:                        if( l->in.op == UNARY AND )
                    407:                        {
                    408:                                p->in.op = l->in.op = FREE;
                    409:                                p = l->in.left;
                    410:                        }
                    411:                        if( !ISPTR(l->in.type) ) uerror("illegal indirection");
                    412:                        p->in.type = DECREF(l->in.type);
                    413:                        p->fn.cdim = l->fn.cdim;
                    414:                        p->fn.csiz = l->fn.csiz;
                    415:                        break;
                    416: 
                    417:                case UNARY AND:
                    418:                        switch( l->in.op ) {
                    419:                        case CONV:
                    420:                                uerror("can't take address of cast");
                    421:                                break;
                    422:                        case STAR:
                    423:                                p->in.op = l->in.op = FREE;
                    424:                                p = l->in.left;
                    425:                        case NAME:
                    426:                        case VAUTO:
                    427:                        case VPARAM:
                    428:                        case TEMP:
                    429:                        case STCALL:
                    430:                        case UNARY STCALL:
                    431: refinc:
                    432:                                p->in.type = INCREF( l->in.type );
                    433:                                p->fn.cdim = l->fn.cdim;
                    434:                                p->fn.csiz = l->fn.csiz;
                    435:                                break;
                    436:                        case RNODE:
                    437: #ifdef TMPSRET
                    438:                                /* don't bother, let optim clean up */
                    439:                                if (simpstr(l->fn.cdim, l->fn.csiz) != STRTY)
                    440:                                        goto refinc;
                    441:                                p->in.op = STAR;
                    442:                                p->in.type = STRTY;
                    443:                                p->fn.csiz = l->fn.csiz;
                    444:                                p->fn.cdim = l->fn.cdim;
                    445:                                l->in.op = FREE;
                    446:                                idname = lookup(hash(".stfake"), 0);
                    447:                                p->in.left = buildtree(NAME, NIL, NIL);
                    448:                                p->in.left->fn.csiz = p->fn.csiz;
                    449:                                p->in.left->fn.cdim = p->fn.cdim;
                    450:                                break;
                    451: #else
                    452:                                goto refinc;
                    453: #endif
                    454:                        case COMOP:
                    455:                                lr = buildtree( UNARY AND, l->in.right, NIL );
                    456:                                p->in.op = l->in.op = FREE;
                    457:                                p = buildtree( COMOP, l->in.left, lr );
                    458:                                break;
                    459:                        case QUEST:
                    460:                                lr = buildtree(UNARY AND,
                    461:                                        l->in.right->in.right, NIL);
                    462:                                ll = buildtree(UNARY AND,
                    463:                                        l->in.right->in.left, NIL);
                    464:                                p->in.op = l->in.op = l->in.right->in.op = FREE;
                    465:                                p = buildtree(QUEST, l->in.left,
                    466:                                        buildtree( COLON, ll, lr));
                    467:                                break;
                    468:                        default:
                    469:                                uerror( "unacceptable operand of &" );
                    470:                                break;
                    471:                        }
                    472:                        break;
                    473:                case RETURN:
                    474:                case ASSIGN:
                    475:                case CAST:
                    476:                        /* structure assignment */
                    477:                        /* take the addresses of the two sides; then make an
                    478:                        ** operator using STASG and
                    479:                        ** the addresses of left and right
                    480:                        */
                    481:                        {
                    482:                                register TWORD t;
                    483:                                register d, s, sz;
                    484: 
                    485:                                if( l->fn.csiz != r->fn.csiz )
                    486:                                        uerror(
                    487:                                        "assignment of different structures" );
                    488: 
                    489:                                r = buildtree( UNARY AND, r, NIL );
                    490:                                l = buildtree( UNARY AND, l, NIL );
                    491:                                t = r->in.type;
                    492:                                d = r->fn.cdim;
                    493:                                s = r->fn.csiz;
                    494: #ifdef ENDSTRUCT
                    495:                                sz = tsize( STRTY, d, s );
                    496:                                l = aadjust( l, sz );
                    497:                                r = aadjust( r, sz );
                    498: #endif
                    499: 
                    500:                                l = block( STASG, l, r, t, d, s );
                    501: 
                    502:                                if( o == RETURN )
                    503:                                {
                    504:                                        p->in.op = FREE;
                    505:                                        p = l;
                    506:                                        break;
                    507:                                }
                    508: 
                    509:                                p->in.op = STAR;
                    510:                                p->in.left = clocal(l);
                    511:                                p->in.right = NIL;
                    512:                                break;
                    513:                        }
                    514:                case COLON:
                    515:                        if( l->fn.csiz != r->fn.csiz )
                    516:                                uerror( "type clash in ?:" );
                    517:                        break;
                    518: 
                    519:                case CALL:
                    520:                        i = 0;
                    521:                        p->in.right = r = strargs( p->in.right, &i );
                    522:                        if( i > maxarg ) maxarg = i;
                    523:                case UNARY CALL:
                    524:                        if( !ISPTR(l->in.type)) uerror("call of non-function");
                    525:                        p->in.type = DECREF(l->in.type);
                    526:                        if( !ISFTN(p->in.type)) uerror("call of non-function");
                    527:                        p->in.type = DECREF( p->in.type );
                    528:                        p->fn.cdim = l->fn.cdim;
                    529:                        p->fn.csiz = l->fn.csiz;
                    530:                        if( l->in.op == UNARY AND &&
                    531:                            l->in.left->in.op == NAME &&
                    532:                            l->in.left->tn.rval >= 0 &&
                    533:                            l->in.left->tn.rval != NONAME &&
                    534:                            ((i=stab[l->in.left->tn.rval].sclass)==FORTRAN
                    535:                            || i == UFORTRAN ) )
                    536:                        {
                    537:                                p->in.op += (FORTCALL-CALL);
                    538:                        }
                    539:                        if( p->in.type == STRTY || p->in.type == UNIONTY )
                    540:                        {
                    541:                                /* function returning structure */
                    542:                                /*  function really returns ptr with * */
                    543: 
                    544: # ifdef ARGSRET
                    545:                                /* set aside argument space for return val */
                    546:                                i = tsize( p->in.type, p->fn.cdim, p->fn.csiz );
                    547:                                /* alignment??? */
                    548:                                if( i > maxarg ) maxarg = i;
                    549: # endif
                    550:                                p->in.op += STCALL-CALL;
                    551:                                p->in.type = INCREF( p->in.type );
                    552:                                p = buildtree( STAR, p, NIL );
                    553:                                break;
                    554:                        }
                    555: 
                    556:                        /* fix up type of return value from call */
                    557:                        {
                    558:                                register TWORD t;
                    559:                                t = indtype( p->tn.type );
                    560:                                if( t != p->tn.type )
                    561:                                {
                    562:                                        p->fn.csiz = p->tn.type = t;
                    563:                                }
                    564: /* pjw                         if (p->fn.type == FLOAT)
                    565:                                        p->fn.type = p->fn.csiz = DOUBLE;
                    566: */                     }
                    567: # ifdef ARGALLRET
                    568:                        /* set aside argument space for return val */
                    569:                        i = tsize( p->in.type, p->fn.cdim, p->fn.csiz );
                    570:                        /* alignment??? */
                    571:                        if( i > maxarg ) maxarg = i;
                    572: # endif
                    573:                        break;
                    574: 
                    575:                default:
                    576:                        cerror( "other code %d", o );
                    577:                }
                    578: 
                    579:        }
                    580: 
                    581:        if( actions & CVTO ) p = oconvert(p);
                    582:        p = clocal(conval(p));
                    583: 
                    584: # ifndef NODBG
                    585:        if( bdebug ) eprint(p);
                    586: # endif
                    587:        p->ln.lineno = tlineno;
                    588:        return(p);
                    589: 
                    590: }
                    591: 
                    592: NODE *
                    593: strargs( p, off )
                    594: register NODE *p;
                    595: register *off;
                    596: {
                    597:         /* rewrite arguments */
                    598:        /* allocate the arguments at *off, and update *off */
                    599:        register TWORD t;
                    600:        int workoff;
                    601: 
                    602:        t = p->fn.type;
                    603:        if( p->in.op == CM )
                    604:        {
                    605: #if defined(BACKARGS) && defined(BACKPARAM) || \
                    606:        !defined(BACKARGS) && !defined(BACKPARAM)
                    607: 
                    608:                p->in.left = strargs( p->in.left, off );
                    609:                p->in.right = strargs( p->in.right, off );
                    610: 
                    611: #else
                    612: 
                    613:                p->in.right = strargs(p->in.right, off);
                    614:                p->in.left = strargs(p->in.left, off);
                    615: 
                    616: #endif
                    617:                return( p );
                    618:        }
                    619: 
                    620:        if (t == FLOAT)
                    621:                p = makety(p, t = DOUBLE, 0, DOUBLE);
                    622:        p = block( STARG, p, NIL, t, p->fn.cdim, p->fn.csiz );
                    623:        if( t == STRTY || t == UNIONTY )
                    624:        {
                    625:                p->in.left = buildtree( UNARY AND, p->in.left, NIL );
                    626:        }
                    627:        else
                    628:        {
                    629:                p->in.op = FUNARG;
                    630:        }
                    631: 
                    632: /*     SETOFF( *off, talign( t, p->fn.csiz ) );        */
                    633:        SETOFF( *off, ALSTACK );
                    634:        workoff = tsize( t, p->fn.cdim, p->fn.csiz );
                    635: 
                    636: #ifdef BACKARGS
                    637:        *off += workoff;
                    638:        p->tn.rval = -*off;
                    639: #else
                    640:        p->tn.rval = *off;
                    641:        *off += workoff;
                    642: #endif
                    643:        return( clocal(p) );
                    644: }
                    645: 
                    646: chkstr( i, j, type )
                    647: register TWORD type;
                    648: register i,j;
                    649: {
                    650:        /* is the MOS or MOU at stab[i] OK for strict reference by a ptr */
                    651:        /* i has been checked to contain a MOS or MOU */
                    652:        /* j is the index in dimtab of the members... */
                    653:        register k, kk;
                    654: 
                    655:        extern int ddebug;
                    656: 
                    657: # ifndef NODBG
                    658:        if( ddebug > 1 ) printf( "chkstr( %s(%d), %d )\n", stab[i].sname, i, j );
                    659: # endif
                    660:        if( (k = j) < 0 ) uerror( "undefined struct/union using %s", stab[i].sname);
                    661:        else
                    662:        {
                    663:                for( ; (kk = dimtab[k] ) >= 0; ++k )
                    664:                {
                    665:                        if( kk >= SYMTSZ )
                    666:                        {
                    667:                                cerror( "gummy structure" );
                    668:                                return(1);
                    669:                        }
                    670:                        if( kk == i ) return( 1 );
                    671:                        switch( stab[kk].stype )
                    672:                        {
                    673: 
                    674:                        case STRTY:
                    675:                        case UNIONTY:
                    676:                                if( type == STRTY ) continue;  /* no recursive looking for strs */
                    677:                                if( hflag && chkstr( i, dimtab[stab[kk].sizoff+1], stab[kk].stype ) )
                    678:                                {
                    679:                                        if( stab[kk].sname[0] == '$' ) return(0);  /* $FAKE */
                    680:                                        werror(
                    681:                                        "illegal member use: perhaps %s.%s?",
                    682:                                                stab[kk].sname, stab[i].sname);
                    683:                                        return(1);
                    684:                                }
                    685:                        }
                    686:                }
                    687:        }
                    688:        return( 0 );
                    689: }
                    690: 
                    691: NODE *
                    692: conval( p )
                    693: register NODE *p;
                    694: {
                    695:        register NODE *l, *r;
                    696:        register o, i, f, u;
                    697:        register CONSZ val;
                    698: 
                    699:        f = (p->tn.type==FLOAT || p->tn.type == DOUBLE);
                    700:        u = ISUNSIGNED(p->tn.type);
                    701: 
                    702:        switch( optype(o = p->tn.op) )
                    703:        {
                    704:        case BITYPE:
                    705:                l = p->in.left;
                    706:                r = p->in.right;
                    707:                break;
                    708: 
                    709:        case UTYPE:
                    710:                r = l = p->in.left;
                    711:                break;
                    712: 
                    713:        case LTYPE:
                    714:                return( p );
                    715:        }
                    716: 
                    717:        if( l->tn.op != ( f ? FCON : ICON ) ) return( p );
                    718:        if( r->tn.op != ( f ? FCON : ICON ) ) return( p );
                    719: 
                    720:        if( !f )
                    721:        {
                    722:                /* weed out unprofitable cases */
                    723:                val = r->tn.lval;
                    724:                if( l->tn.rval != NONAME && r->tn.rval != NONAME ) return(p);
                    725:                if( r->tn.rval != NONAME && o!=PLUS ) return(p);
                    726:                if( l->tn.rval != NONAME && o!=PLUS && o!=MINUS ) return(p);
                    727:        }
                    728:        else if( logop(o) ) return( p );
                    729: 
                    730:        switch( o )
                    731:        {
                    732: 
                    733:        case PLUS:
                    734:                if( f )
                    735:                {
                    736:                        l->fpn.dval += r->fpn.dval;
                    737:                        break;
                    738:                }
                    739:                l->tn.lval += val;
                    740:                if( l->tn.rval == NONAME )
                    741:                {
                    742:                        l->tn.rval = r->tn.rval;
                    743:                        l->in.type = r->in.type;
                    744:                }
                    745:                break;
                    746: 
                    747:        case MINUS:
                    748:                if( f ) l->fpn.dval -= r->fpn.dval;
                    749:                else l->tn.lval -= val;
                    750:                break;
                    751: 
                    752:        case MUL:
                    753:                if( f ) l->fpn.dval *= r->fpn.dval;
                    754:                else l->tn.lval *= val;
                    755:                break;
                    756: 
                    757:        case DIV:
                    758:                if( f )
                    759:                {
                    760:                        if( r->fpn.dval == 0. ) uerror( "division by 0" );
                    761:                        else l->fpn.dval /= r->fpn.dval;
                    762:                }
                    763:                else
                    764:                {
                    765:                        if( val == 0 ) uerror( "division by 0" );
                    766:                        else if( u )
                    767:                                l->tn.lval = (unsigned long)l->tn.lval / val;
                    768:                        else l->tn.lval /= val;
                    769:                }
                    770:                break;
                    771: 
                    772:        case MOD:
                    773:                if( val == 0 ) uerror( "division by 0" );
                    774:                else if( u ) l->tn.lval = (unsigned long)l->tn.lval % val;
                    775:                else l->tn.lval %= val;
                    776:                break;
                    777: 
                    778:        case AND:
                    779:                l->tn.lval &= val;
                    780:                break;
                    781: 
                    782:        case OR:
                    783:                l->tn.lval |= val;
                    784:                break;
                    785: 
                    786:        case ER:
                    787:                l->tn.lval ^=  val;
                    788:                break;
                    789: 
                    790:        case LS:
                    791:                i = val;
                    792:                l->tn.lval = l->tn.lval << i;
                    793:                break;
                    794: 
                    795:        case RS:
                    796:                i = val;
                    797:                if(i > 32) {
                    798:                        uerror("illegal shift");
                    799:                        break;
                    800:                }
                    801:                if( u ) l->tn.lval = (unsigned long)l->tn.lval >> i;
                    802:                else l->tn.lval = l->tn.lval >> i;
                    803:                break;
                    804: 
                    805:        case UNARY MINUS:
                    806:                if( f ) l->fpn.dval = -l->fpn.dval;
                    807:                else l->tn.lval = - l->tn.lval;
                    808:                break;
                    809: 
                    810:        case COMPL:
                    811:                l->tn.lval = ~l->tn.lval;
                    812:                break;
                    813: 
                    814:        case NOT:
                    815:                if( l->tn.type == FLOAT || l->tn.type == DOUBLE )
                    816:                {
                    817:                        l->tn.op = ICON;
                    818:                        l->tn.lval = !l->fpn.dval;
                    819:                }
                    820:                else l->tn.lval = !l->tn.lval;
                    821:                break;
                    822: 
                    823:        case LT:
                    824:                l->tn.lval = l->tn.lval < val;
                    825:                break;
                    826: 
                    827:        case LE:
                    828:                l->tn.lval = l->tn.lval <= val;
                    829:                break;
                    830: 
                    831:        case GT:
                    832:                l->tn.lval = l->tn.lval > val;
                    833:                break;
                    834: 
                    835:        case GE:
                    836:                l->tn.lval = l->tn.lval >= val;
                    837:                break;
                    838: 
                    839:        case ULT:
                    840:                l->tn.lval = (l->tn.lval-val)<0;
                    841:                break;
                    842: 
                    843:        case ULE:
                    844:                l->tn.lval = (l->tn.lval-val)<=0;
                    845:                break;
                    846: 
                    847:        case UGE:
                    848:                l->tn.lval = (l->tn.lval-val)>=0;
                    849:                break;
                    850: 
                    851:        case UGT:
                    852:                l->tn.lval = (l->tn.lval-val)>0;
                    853:                break;
                    854: 
                    855:        case EQ:
                    856:                l->tn.lval = l->tn.lval == val;
                    857:                break;
                    858: 
                    859:        case NE:
                    860:                l->tn.lval = l->tn.lval != val;
                    861:                break;
                    862: 
                    863:        default:
                    864:                return(p);
                    865:        }
                    866: 
                    867:        if( l != r ) r->tn.op = FREE; /* don't clobber unary answer */
                    868:        l = makety( l, p->tn.type, p->fn.cdim, p->fn.csiz );
                    869:        p->tn.op = FREE;
                    870:        return( l );
                    871: }
                    872: 
                    873: chkpun(p)
                    874: register NODE *p;
                    875: {
                    876:        /* checks p for the existance of a pun */
                    877:        /* this is called when the op of p is ASSIGN, RETURN, CAST, COLON,
                    878:        ** or relational
                    879:        */
                    880: 
                    881:        /* one case is when enumerations are used: this applies only to lint */
                    882:        /* in the other case, one operand is a pointer, the other integer */
                    883:        /* we check that this integer is in fact a constant zero... */
                    884:        /* if it is, we redo the type of the 0 to match the other side */
                    885: 
                    886:        /* with ASSIGN, assignment of pointer to integer is illegal */
                    887:        /* this falls out, because the LHS is never 0 */
                    888: 
                    889:        register NODE *q1, *q2;
                    890:        register t1, t2;
                    891:        register d1, d2;
                    892: 
                    893:        q1 = p->in.left;
                    894:        q2 = p->in.right;
                    895:        t1 = q1->in.type;
                    896:        t2 = q2->in.type;
                    897: 
                    898:        if( t1==ENUMTY || t2==ENUMTY )
                    899:        {
                    900:                 /* check for enumerations */
                    901:                /*(pjw)if( logop( p->in.op ) && p->in.op != EQ && p->in.op != NE )
                    902:                {
                    903:                        uerror( "illegal comparison of enums" );
                    904:                        return;
                    905:                }*/
                    906:                if( t1==ENUMTY && t2==ENUMTY && q1->fn.csiz==q2->fn.csiz )
                    907:                        return;
                    908:                werror( "enumeration type clash, operator %s", opst[p->in.op] );
                    909:                return;
                    910:        }
                    911: 
                    912:        if( ISPTR(t1) || ISARY(t1) )
                    913:        {       /* switch roles: q1 should be non-pointer */
                    914:                q1 = q2;
                    915:                q2 = p->in.left;
                    916:        }
                    917: 
                    918:        if( !ISPTR(q1->in.type) && !ISARY(q1->in.type) )
                    919:        {
                    920:                if( q1->in.op == ICON && q1->tn.lval == 0 && q1->tn.rval == NONAME )
                    921:                {       /* make the 0 have the type of the other side */
                    922:                        q1->fn.type = q2->fn.type;
                    923:                        q1->fn.cdim = q2->fn.cdim;
                    924:                        q1->fn.csiz = q2->fn.type;
                    925:                }
                    926:                else
                    927:                {
                    928:                        combo( "pointer/integer", p );
                    929:                }
                    930:                return;
                    931:        }
                    932:        else
                    933:        {       /* q1 and q2 still mean left and right */
                    934:                d1 = q1->fn.cdim;
                    935:                d2 = q2->fn.cdim;
                    936:                for( ;; )
                    937:                {
                    938:                        if( t1 == t2 )
                    939:                        {
                    940:                                ;
                    941:                                if(q1->fn.csiz!=q2->fn.csiz)
                    942:                                {
                    943:                                        combo( "structure pointer", p );
                    944:                                }
                    945:                                return;
                    946:                        }
                    947: /****** LINT Change: Turn on only if LINT is to be used with this pcc2 ******/
                    948: /*                     /* complain about pointer casts if cflag is set
                    949: /*                     /* (this implies that pflag is set)
                    950: /*                     */
                    951: /*                     if (p->in.op == CAST)
                    952: /*                     {
                    953: /*                             if (ISPTR(tl) && ISPTR(t2))
                    954: /*                             {
                    955: /*                                     werror(
                    956: /*                                     "pointer casts may be troublesome");
                    957: /*                                     return;
                    958: /*                             }
                    959: /*                     }
                    960: /****************************************************************************/
                    961:                        if( ISARY(t1) || ISPTR(t1) )
                    962:                        {
                    963:                                if( !ISARY(t2) && !ISPTR(t2) ) break;
                    964:                                if( ISARY(t1) && ISARY(t2) &&
                    965:                                    dimtab[d1] != dimtab[d2] )
                    966:                                {
                    967:                                        combo( "array size", p );
                    968:                                        return;
                    969:                                }
                    970:                                if( ISARY(t1) ) ++d1;
                    971:                                if( ISARY(t2) ) ++d2;
                    972:                        }
                    973:                        else break;
                    974:                        t1 = DECREF(t1);
                    975:                        t2 = DECREF(t2);
                    976:                }
                    977:                combo( "pointer", p );
                    978:        }
                    979: }
                    980: 
                    981: combo( s, p )
                    982: char *s;
                    983: NODE *p;
                    984: {
                    985:        char buf[100];
                    986:        sprintf( buf, "illegal %s combination, op %s", s, opst[p->tn.op] );
                    987:        if(onlywarn)
                    988:                werror(buf);
                    989:        else
                    990:                uerror(buf);
                    991: }
                    992: 
                    993: NODE *
                    994: stref( p )
                    995: register NODE *p;
                    996: {
                    997:        register TWORD t;
                    998:        register d, s, dsc, align;
                    999:        register OFFSZ off;
                   1000:        register struct symtab *q;
                   1001: 
                   1002:        /* make p->x */
                   1003:        /* this is also used to reference automatic variables */
                   1004:        q = &stab[p->in.right->tn.rval];
                   1005:        q->suse = -lineno;
                   1006:        p->in.right->in.op = FREE;
                   1007:        p->in.op = FREE;
                   1008:        p = pconvert( p->in.left );
                   1009: 
                   1010:        /* make p look like ptr to x */
                   1011: 
                   1012:        if( !ISPTR(p->in.type))
                   1013:        {
                   1014:                p->in.type = PTR+UNIONTY;
                   1015:        }
                   1016: 
                   1017:        t = INCREF( q->stype );
                   1018:        d = q->dimoff;
                   1019:        s = q->sizoff;
                   1020: 
                   1021:        p = makety( p, t, d, s );
                   1022: 
                   1023:        /* compute the offset to be added */
                   1024: 
                   1025:        off = q->offset;
                   1026:        dsc = q->sclass;
                   1027: 
                   1028:        if( dsc & FIELD )
                   1029:        {
                   1030:                  /* normalize offset */
                   1031:                switch(q->stype)
                   1032:                {
                   1033: 
                   1034:                case CHAR:
                   1035:                case UCHAR:
                   1036:                        align = ALCHAR;
                   1037:                        s = CHAR;
                   1038:                        break;
                   1039: 
                   1040:                case SHORT:
                   1041:                case USHORT:
                   1042:                        align = ALSHORT;
                   1043:                        s = SHORT;
                   1044:                        break;
                   1045: 
                   1046:                case ENUMTY:
                   1047:                case INT:
                   1048:                case UNSIGNED:
                   1049:                        align = ALINT;
                   1050:                        s = INT;
                   1051:                        break;
                   1052: 
                   1053: # ifdef LONGFIELDS
                   1054:                case LONG:
                   1055:                case ULONG:
                   1056:                        align = ALLONG;
                   1057:                        s = LONG;
                   1058:                        break;
                   1059: # endif
                   1060: 
                   1061:                default:
                   1062:                        cerror( "undefined bit field type" );
                   1063:                }
                   1064:                off = (off/align)*align;
                   1065:        }
                   1066:        if( off != 0 )
                   1067:                p = clocal( block( PLUS, p, offcon( off, t, d, s ), t, d, s ) );
                   1068: 
                   1069:        p = buildtree( STAR, p, NIL );
                   1070: 
                   1071:        /* if field, build field info */
                   1072: 
                   1073:        if( dsc & FIELD )
                   1074:        {
                   1075:                p = block( FLD, p, NIL, q->stype, 0, q->sizoff );
                   1076:                p->tn.rval = PKFIELD( dsc&FLDSIZ, q->offset%align );
                   1077:        }
                   1078:        p = clocal(p);
                   1079:        return(p);
                   1080: }
                   1081: 
                   1082: notlval(p)
                   1083: register NODE *p;
                   1084: {
                   1085: 
                   1086:        /* return 0 if p an lvalue, 1 otherwise */
                   1087: 
                   1088: again:
                   1089:        switch( p->in.op )
                   1090:        {
                   1091: 
                   1092:        case FLD:
                   1093:                p = p->in.left;
                   1094:                goto again;
                   1095: 
                   1096:        case STAR:
                   1097:                /* is there a special case for structs? */
                   1098:        case NAME:
                   1099:        case VAUTO:
                   1100:        case VPARAM:
                   1101:        case RNODE:
                   1102:        case QNODE:
                   1103:        case SNODE:
                   1104:                if( ISARY(p->in.type) || ISFTN(p->in.type) ) return(1);
                   1105:        case REG:
                   1106:                return(0);
                   1107: 
                   1108:        default:
                   1109:                return(1);
                   1110: 
                   1111:        }
                   1112: 
                   1113: }
                   1114: 
                   1115: NODE *
                   1116: bcon( i )
                   1117: {
                   1118:         /* make a constant node with value i */
                   1119:        register NODE *p;
                   1120: 
                   1121:        p = block( ICON, NIL, NIL, INT, 0, INT );
                   1122:        p->tn.lval = i;
                   1123:        p->tn.rval = NONAME;
                   1124:        return( clocal(p) );
                   1125: }
                   1126: 
                   1127: NODE *
                   1128: bpsize(p)
                   1129: register NODE *p;
                   1130: {
                   1131:        return( offcon( psize(p), p->in.type, p->fn.cdim, p->fn.csiz ) );
                   1132: }
                   1133: 
                   1134: OFFSZ
                   1135: psize( p )
                   1136: register NODE *p;
                   1137: {
                   1138:        /* p is a node of type pointer; psize returns the
                   1139:        ** size of the thing pointed to
                   1140:        */
                   1141: 
                   1142:        if( !ISPTR(p->in.type) )
                   1143:        {
                   1144:                uerror( "pointer required");
                   1145:                return( SZINT );
                   1146:        }
                   1147:        /* note: no pointers to fields */
                   1148:        return( tsize( DECREF(p->in.type), p->fn.cdim, p->fn.csiz ) );
                   1149: }
                   1150: 
                   1151: NODE *
                   1152: unconvert( p, q )
                   1153: register NODE *p, *q;
                   1154: {
                   1155:        /* p is divided by the size of q */
                   1156:        /* q had better have type pointer */
                   1157: 
                   1158: # ifdef MYPDIV
                   1159:        return( clocal( block( PDIV, p, bpsize(q), INT, 0, INT ) ) );
                   1160: # else
                   1161:        p = makety( p, PTRTYPE, 0, PTRTYPE );
                   1162:        p = clocal( buildtree( DIV, p, bpsize(q) ) );
                   1163: 
                   1164:        return( makety( p, PTRTYPE, 0, PTRTYPE ) );
                   1165: # endif
                   1166: }
                   1167: 
                   1168: NODE *
                   1169: convert( p, f )
                   1170: register NODE *p;
                   1171: register f;
                   1172: {
                   1173:        /*  convert an operand of p
                   1174:        ** f is either CVTL or CVTR
                   1175:        ** operand has type int, and is converted by the size of the other side
                   1176:        */
                   1177: 
                   1178:        register NODE *q, *r;
                   1179: 
                   1180:        if( f == CVTL )
                   1181:        {
                   1182:                q = p->in.left;
                   1183:                r = bpsize( p->in.right );
                   1184:        }
                   1185:        else
                   1186:        {
                   1187:                q = p->in.right;
                   1188:                r = bpsize( p->in.left );
                   1189:        }
                   1190: # ifdef MYPMUL
                   1191:        r = clocal( block( PMUL, q, r, PTRTYPE, 0, PTRTYPE ) );
                   1192: # else
                   1193:        /* if PTRTYPE is defined, make q this size; then, the MUL will be */
                   1194:        q = makety( q, PTRTYPE, 0, PTRTYPE );
                   1195:        r = clocal( buildtree( MUL, q, r ) );
                   1196: # endif
                   1197:        if( f == CVTL )
                   1198:                p->in.left = r;
                   1199:        else
                   1200:                p->in.right = r;
                   1201:        return(p);
                   1202: 
                   1203: }
                   1204: 
                   1205: econvert( p )
                   1206: register NODE *p;
                   1207: {
                   1208: 
                   1209:        /* change enums to ints, or appropriate types */
                   1210: 
                   1211:        register TWORD ty;
                   1212: 
                   1213:        if( (ty=BTYPE(p->in.type)) == ENUMTY || ty == MOETY )
                   1214:        {
                   1215:                if( dimtab[ p->fn.csiz ] == SZCHAR ) ty = CHAR;
                   1216:                else if( dimtab[ p->fn.csiz ] == SZINT ) ty = INT;
                   1217:                else if( dimtab[ p->fn.csiz ] == SZSHORT ) ty = SHORT;
                   1218:                else ty = LONG;
                   1219:                ty = ctype( ty );
                   1220:                p->fn.csiz = ty;
                   1221:                MODTYPE(p->in.type,ty);
                   1222:                if( p->in.op == ICON && ty != LONG )
                   1223:                        p->in.type = p->fn.csiz = INT;
                   1224:        }
                   1225: }
                   1226: 
                   1227: NODE *
                   1228: pconvert( p )
                   1229: register NODE *p;
                   1230: {
                   1231:        register TWORD t;
                   1232: 
                   1233:        /* if p should be changed into a pointer, do so */
                   1234:        /* also, widen p so it can be used as an operand */
                   1235: 
                   1236:        if( ISARY( p->in.type) )
                   1237:        {
                   1238:                p->in.type = DECREF( p->in.type );
                   1239:                ++p->fn.cdim;
                   1240:                return( buildtree( UNARY AND, p, NIL ) );
                   1241:        }
                   1242:        if( ISFTN( p->in.type) )
                   1243:                return( buildtree( UNARY AND, p, NIL ) );
                   1244: 
                   1245:        t = indtype( p->tn.type );
                   1246:        if( t == p->tn.type ) return( p );
                   1247:        if( p->tn.op == CONV && cbigger( p ) )
                   1248:        {
                   1249:                p->tn.type = p->fn.csiz = t;
                   1250:        }
                   1251:        else p = makety( p, t, 0, (int) t );
                   1252:        return( p );
                   1253: }
                   1254: 
                   1255: bigsize( t )
                   1256: TWORD  t; 
                   1257: {      /* return a type size for t */
                   1258:        switch( t ) {
                   1259:        case CHAR:
                   1260:        case UCHAR:
                   1261:                return( SZCHAR );
                   1262:        case SHORT:
                   1263:        case USHORT:
                   1264:                return( SZSHORT );
                   1265:        case VOID:
                   1266:        case INT:
                   1267:        case UNSIGNED:
                   1268:        case ENUMTY:
                   1269:                return( SZINT );
                   1270:        case LONG:
                   1271:        case ULONG:
                   1272:                return( SZLONG );
                   1273:        case FLOAT:
                   1274:                return( SZLONG+1 );  /* must appear > long */
                   1275:        case DOUBLE:
                   1276:                return( SZLONG+2 );
                   1277:        case STRTY:
                   1278:                uerror("structure appears where arithmetic type required");
                   1279:                return(0);
                   1280:        case UNIONTY:
                   1281:                uerror("union appears where arithmetic type required");
                   1282:                return(0);
                   1283:        }
                   1284: 
                   1285:        if( ISPTR(t) || ISARY(t) ) return( SZPOINT );
                   1286:        
                   1287:        cerror( "bad bigsize: 0%o", t );
                   1288: /*NOTREACHED*/
                   1289: }
                   1290: 
                   1291: cbigger( p )
                   1292: register NODE *p; 
                   1293: {
                   1294:        /* returns 1 if the conversion op p makes things bigger */
                   1295: 
                   1296:        if( p->tn.op != CONV ) cerror( "cbigger" );
                   1297:        if(p->tn.type == STRTY || p->tn.type == UNIONTY || p->tn.type == VOID) {
                   1298:                uerror("trying to cast to a struct, void or union");
                   1299:                return(0);
                   1300:        }
                   1301:        return( bigsize( p->tn.type ) > bigsize( p->in.left->tn.type ) );
                   1302: }
                   1303: 
                   1304: NODE *
                   1305: oconvert(p)
                   1306: register NODE *p;
                   1307: 
                   1308: {
                   1309:        /* convert the result itself: used for pointer and unsigned */
                   1310:        register TWORD t;
                   1311: 
                   1312:        switch(p->in.op)
                   1313:        {
                   1314: 
                   1315:        case LE:
                   1316:        case LT:
                   1317:        case GE:
                   1318:        case GT:
                   1319:                t = p->in.left->in.type;
                   1320:                if( ISUNSIGNED(t) || ISPTR(t) || ISARY(t) )
                   1321:                {
                   1322:                        p->in.op += (ULE-LE);
                   1323:                }
                   1324:                else
                   1325:                {
                   1326:                        t = p->in.right->in.type;
                   1327:                        if( ISUNSIGNED(t) || ISPTR(t) || ISARY(t) )
                   1328:                        {
                   1329:                                p->in.op += (ULE-LE);
                   1330:                        }
                   1331:                }
                   1332:        case EQ:
                   1333:        case NE:
                   1334:        case ULE:
                   1335:        case ULT:
                   1336:        case UGE:
                   1337:        case UGT:
                   1338:                return( p );
                   1339: 
                   1340:        case MINUS:
                   1341:                return( unconvert( p, p->in.left ) );
                   1342:        }
                   1343: 
                   1344:        cerror( "illegal oconvert: %d", p->in.op );
                   1345: 
                   1346:        return(p);
                   1347: }
                   1348: 
                   1349: NODE *
                   1350: ptmatch(p)
                   1351: register NODE *p;
                   1352: {
                   1353: 
                   1354:        /* makes the operands of p agree; they are
                   1355:        ** either pointers or integers, by this time
                   1356:        */
                   1357:        /* with MINUS, the sizes must be the same */
                   1358:        /* with COLON, the types must be the same */
                   1359: 
                   1360:        register TWORD t1, t2, t;
                   1361:        register o, d2, d, s2, s;
                   1362: 
                   1363:        o = p->in.op;
                   1364:        t = t1 = p->in.left->in.type;
                   1365:        t2 = p->in.right->in.type;
                   1366:        d = p->in.left->fn.cdim;
                   1367:        d2 = p->in.right->fn.cdim;
                   1368:        s = p->in.left->fn.csiz;
                   1369:        s2 = p->in.right->fn.csiz;
                   1370: 
                   1371:        switch( o )
                   1372:        {
                   1373: 
                   1374:        case ASSIGN:
                   1375:        case RETURN:
                   1376:        case CAST:
                   1377:                break;
                   1378: 
                   1379:        case MINUS:
                   1380:                if( psize(p->in.left) != psize(p->in.right) )
                   1381:                {
                   1382:                        uerror( "illegal pointer subtraction");
                   1383:                }
                   1384:                p->tn.type = p->fn.csiz = PTRTYPE;
                   1385:                p->fn.cdim = 0;
                   1386:                return( clocal(p) );
                   1387: 
                   1388:        case COLON:
                   1389:                if( t1 != t2 ) uerror( "incompatible types in ?:");
                   1390:                break;
                   1391: 
                   1392:        default:
                   1393:                if( !ISPTR(t1) )
                   1394:                {
                   1395:                        t = t2;
                   1396:                        d = d2;
                   1397:                        s = s2;
                   1398:                        break;
                   1399:                }
                   1400:                if( !ISPTR(t2) )
                   1401:                {
                   1402:                        break;
                   1403:                }
                   1404: 
                   1405:                /* both are pointers */
                   1406:                if( talign(t2,s2) < talign(t,s) )
                   1407:                {
                   1408:                        t = t2;
                   1409:                        s = s2;
                   1410:                }
                   1411:                break;
                   1412:        }
                   1413: 
                   1414:        p->in.left = makety( p->in.left, t, d, s );
                   1415:        p->in.right = makety( p->in.right, t, d, s );
                   1416:        if( !logop(o) )
                   1417:        {
                   1418:                p->in.type = t;
                   1419:                p->fn.cdim = d;
                   1420:                p->fn.csiz = s;
                   1421:        }
                   1422: 
                   1423:        return(clocal(p));
                   1424: }
                   1425: 
                   1426: int tdebug = 0;
                   1427: 
                   1428: NODE *
                   1429: tymatch(p)
                   1430: register NODE *p;
                   1431: {
                   1432: 
                   1433:        /* satisfy the types of various arithmetic binary ops */
                   1434: 
                   1435:        /* rules are:
                   1436:        ** if any float or doubles, make double
                   1437:        ** no, rick changes this rule to
                   1438:        ** a. if any doubles, make double 
                   1439:        ** b. if both floats, make float
                   1440:        ** c. if any floats, make double
                   1441:        ** the rest proceed as usual
                   1442:        ** if any longs, make long
                   1443:        ** if any ints, make int
                   1444:        ** if any shorts, make short
                   1445:        ** otherwise, make char
                   1446:        ** if either operand is unsigned, the result is...
                   1447:        ** if a simple assignment, type = type of lhs
                   1448:        ** if an asg. op, type = type of original lhs
                   1449:        **      (under CONV, if any)
                   1450:        ** pjw observes this is not right, for int /= double is wrong
                   1451:        */
                   1452: 
                   1453:        register o;
                   1454:        register TWORD t1, t2;
                   1455:        TWORD t, tu;
                   1456:        register NODE *l, *r;
                   1457:        int u;
                   1458: 
                   1459:        o = p->in.op;
                   1460: 
                   1461:        t1 = (l=p->in.left)->in.type;
                   1462:        t2 = (r=p->in.right)->in.type;
                   1463: 
                   1464:        /* constants have a kind of "flexible" type */
                   1465: 
                   1466:        if( r->tn.op == ICON && r->tn.rval == NONAME &&
                   1467:            (t1==CHAR || t1==UCHAR || t1==SHORT || t1==USHORT) )
                   1468:        {
                   1469:                /* if the constant retains its value when cast to the
                   1470:                ** type of the lhs, assume it has the lhs type
                   1471:                */
                   1472:                if( r->tn.lval == ccast( r->tn.lval, t1 ) )
                   1473:                {
                   1474:                        r->in.type = t2 = t1;
                   1475:                        r->fn.cdim = l->fn.cdim;
                   1476:                        r->fn.csiz = l->fn.csiz;
                   1477:                }
                   1478:        }
                   1479: 
                   1480:        /* this is the opposite case from the above (it is too early
                   1481:                to assume that constants are on the right) */
                   1482: 
                   1483:        if( l->tn.op == ICON && l->tn.rval == NONAME &&
                   1484:            (t2==CHAR || t2==UCHAR || t2==SHORT || t2==USHORT) )
                   1485:        {
                   1486:                /* if the constant retains its value when cast to the
                   1487:                ** type of the rhs, assume it has the rhs type
                   1488:                */
                   1489:                if( l->tn.lval == ccast( l->tn.lval, t2 ) )
                   1490:                {
                   1491:                        l->in.type = t1 = t2;
                   1492:                        l->fn.cdim = r->fn.cdim;
                   1493:                        l->fn.csiz = r->fn.csiz;
                   1494:                }
                   1495:        }
                   1496:        if( (t1==VOID || t2==VOID) && o!=CAST )
                   1497:                uerror("void type illegal in expression");
                   1498: 
                   1499:        u = 0;
                   1500:        if( ISUNSIGNED(t1) )
                   1501:        {
                   1502:                u = 1;
                   1503:                t1 = DEUNSIGN(t1);
                   1504:        }
                   1505:        if( ISUNSIGNED(t2) )
                   1506:        {
                   1507:                u |= 2;
                   1508:                t2 = DEUNSIGN(t2);
                   1509:        }
                   1510: /*rick
                   1511: /*     if( t1==DOUBLE || t1==FLOAT || t2==DOUBLE || t2==FLOAT ) t = DOUBLE;
                   1512: */
                   1513: /*pjw says enums are ints dammit */
                   1514:        if(t1 == ENUMTY) t1 = INT;
                   1515:        if(t2 == ENUMTY) t2 = INT;
                   1516:        if (t1 == DOUBLE || t2 == DOUBLE) t = DOUBLE;
                   1517:        else if (t1 == FLOAT && t2 == FLOAT) t = FLOAT;
                   1518:        else if (t1 == FLOAT || t2 == FLOAT) t = DOUBLE;
                   1519:        else if( t1==LONG || t2==LONG ) t = LONG;
                   1520:        else if( t1==INT || t2==INT ) t = INT;
                   1521:        else if( t1==SHORT || t2==SHORT ) t = SHORT;
                   1522:        else t = CHAR;
                   1523: 
                   1524:        if( o == ASSIGN || o == CAST || o == RETURN )
                   1525:        {
                   1526:                tu = t1;
                   1527:                if( o == RETURN )
                   1528:                {
                   1529:                        if( tu==FLOAT ) tu = DOUBLE;
                   1530:                        else if( tu==CHAR || tu==SHORT ) tu = INT;
                   1531:                }
                   1532:                t = t1 = tu;  /* t1 set to avoid lhs conversion */
                   1533:                if( ISUNSIGNED(l->tn.type) ) u=1;
                   1534:                else u=0;
                   1535:                t2 = r->tn.type;  /* back to reality... */
                   1536:        }
                   1537: /* pjw says this is the wrong test, as int -= unsigned should be int
                   1538:        tu = (u && UNSIGNABLE(t))?ENUNSIGN(t):t; */
                   1539:        if(u && UNSIGNABLE(t) && !asgbinop(o))
                   1540:                tu = ENUNSIGN(t);
                   1541:        else if(asgbinop(o)) {
                   1542:                tu = (u & 1? ENUNSIGN(t1): t1);
                   1543:                /* a bunch of cases for x op= float */
                   1544:                if((t2 == DOUBLE || t2 == FLOAT) && t1 != FLOAT && t1 != DOUBLE)
                   1545:                        tu = t2;
                   1546:        }
                   1547:        else
                   1548:                tu = t;
                   1549:        if( tu != t1 ) p->in.left = makety( l, tu, 0, (int)tu );
                   1550:        if( tu != t2 ) p->in.right = makety( r, tu, 0, (int)tu );
                   1551: 
                   1552:        if( !logop(o) ) /* the type of the node = type of the operation */
                   1553:        {
                   1554:                p->in.type = tu;
                   1555:                p->fn.cdim = 0;
                   1556:                p->fn.csiz = t;
                   1557:        }
                   1558: 
                   1559:        if( asgop(o) && p->in.left->tn.op == CONV  )
                   1560:        {
                   1561:                /* original, unconverted type */
                   1562:                /* this will get rewritten to A = A op B, in most cases */
                   1563:                /* if the op is an unsigned op, change the lhs to unsigned */
                   1564:                l = p->in.left->in.left;
                   1565:                tu = l->in.type;
                   1566:                if( u && (UNSIGNABLE(tu)) ) tu = ENUNSIGN(tu);
                   1567:                p->in.type = l->in.type = tu;
                   1568:                p->fn.cdim = l->fn.cdim;
                   1569:                p->fn.csiz = l->fn.csiz;
                   1570:        }
                   1571: 
                   1572: # ifndef NODBG
                   1573:        if( tdebug )
                   1574:        {
                   1575:                printf( "tymatch(%o): %o '%s' %o => %o\n",p,t1,opst[o],t2,tu );
                   1576:                eprint(p);
                   1577:        }
                   1578: # endif
                   1579: 
                   1580:        return(p);
                   1581: }
                   1582: 
                   1583: NODE *
                   1584: makety( p, t, d, s )
                   1585: register NODE *p;
                   1586: register TWORD t;
                   1587: register d,s;
                   1588: {
                   1589:        /* make p into type t by inserting a conversion */
                   1590:        register TWORD pt;
                   1591: 
                   1592:        pt = p->in.type;
                   1593: 
                   1594:        if( pt == ENUMTY && p->in.op == ICON )
                   1595:        {
                   1596:                econvert(p);
                   1597:                pt = p->in.type;
                   1598:        }
                   1599:  
                   1600:        if( ISARY(pt) || ISFTN(pt) )
                   1601:        {
                   1602:                p = pconvert( p );
                   1603:                pt = p->in.type;
                   1604:        }
                   1605:  
                   1606:        if( t == pt || (p->tn.op == FCON && p->tn.rval == NONAME))
                   1607:        {
                   1608: rew:
                   1609:                p->fn.type = t;
                   1610:                p->fn.cdim = d;
                   1611:                p->fn.csiz = s;
                   1612:                return( p );
                   1613:        }
                   1614: 
                   1615:        if( t & TMASK )
                   1616:        {
                   1617:                /* non-simple type */
                   1618:                if( ISPTR(pt) && p->in.op != STASG      /*pjw*/
                   1619: # ifdef TWOPTRS
                   1620:                    && TWOPTRS(t) == TWOPTRS(pt)
                   1621: # endif
                   1622:                    )
                   1623:                {
                   1624:                        /* don't generate CONV: just rewrite type */
                   1625:                        goto rew;
                   1626:                }
                   1627: 
                   1628:                return( block( CONV, p, NIL, t, d, s ) );
                   1629:        }
                   1630: 
                   1631:        if( p->in.op == ICON && p->tn.rval==NONAME )
                   1632:        {
                   1633:                if( t==DOUBLE||t==FLOAT )
                   1634:                {
                   1635:                        p->in.op = FCON;
                   1636:                        if( ISUNSIGNED(pt) )
                   1637:                        {
                   1638:                                p->fpn.dval = /* (unsigned CONSZ) */ p->tn.lval;
                   1639:                        }
                   1640:                        else
                   1641:                        {
                   1642:                                p->fpn.dval = p->tn.lval;
                   1643:                        }
                   1644:                        goto rew;
                   1645:                }
                   1646:                p->tn.lval = ccast( p->tn.lval, t );
                   1647:                goto rew;
                   1648:        }
                   1649: 
                   1650:        /* double op ((double) (float op float)) => */
                   1651:        /* double op ((double) float op (double) float) */
                   1652:        /* recursively */
                   1653:        /* don't push a conversion down the lhs of an assign op */
                   1654: 
                   1655:        if (pt == FLOAT && t == DOUBLE && optype(p->in.op) == BITYPE &&
                   1656:            !asgop(p->in.op) && p->in.op != CALL)
                   1657:        {
                   1658:                p->in.left = makety(p->in.left, t, d, s);
                   1659:                p->in.right = makety(p->in.right, t, d, s);
                   1660:                goto rew;
                   1661:        }       
                   1662:        return( block( CONV, p, NIL, t, d, s ) );
                   1663: }
                   1664: 
                   1665: NODE *
                   1666: block( o, l, r, t, d, s )
                   1667: NODE *l, *r; TWORD t;
                   1668: {
                   1669: 
                   1670:        register NODE *p;
                   1671: 
                   1672:        p = talloc();
                   1673:        p->in.op = o;
                   1674:        p->in.left = l;
                   1675:        p->in.right = r;
                   1676:        p->in.type = t;
                   1677:        p->ln.lineno = 0;
                   1678:        p->fn.cdim = d;
                   1679:        p->fn.csiz = s;
                   1680: 
                   1681: /*     for really heavy debugging
                   1682:        printf( "block( %s, %d, %d, 0%o, %d, %d ) yields %d\n",
                   1683:                opst[o], l-node, r-node, t, d, s, p-node );
                   1684: */
                   1685: 
                   1686:        return(p);
                   1687: }
                   1688: 
                   1689: icons(p)
                   1690: register NODE *p;
                   1691: {
                   1692:        /* if p is an integer constant, return its value */
                   1693:        register val;
                   1694: 
                   1695:        if( p->in.op != ICON )
                   1696:        {
                   1697:                uerror( "integer constant expected");
                   1698:                val = 1;
                   1699:        }
                   1700:        else
                   1701:        {
                   1702:                val = p->tn.lval;
                   1703:                if( val != p->tn.lval )
                   1704:                        uerror( "constant too big for cross-compiler" );
                   1705:        }
                   1706:        tfree( p );
                   1707:        return(val);
                   1708: }
                   1709: 
                   1710: /*     the intent of this table is to examine the
                   1711: ** operators, and to check them for
                   1712: ** correctness.
                   1713: **
                   1714: ** The table is searched for the op and the
                   1715: ** modified type (where this is one of the
                   1716: ** types INT (includes char and short), LONG,
                   1717: ** DOUBLE (includes FLOAT), and POINTER
                   1718: **
                   1719: ** The default action is to make the node type integer
                   1720: **
                   1721: ** The actions taken include:
                   1722: ** PUN   check for puns
                   1723: ** CVTL          convert the left operand
                   1724: ** CVTR          convert the right operand
                   1725: ** TYPL          the type is determined by the left operand
                   1726: ** TYPR          the type is determined by the right operand
                   1727: ** TYMATCH       force type of left and right to match, by inserting conversions
                   1728: ** PTMATCH       like TYMATCH, but for pointers
                   1729: ** LVAL          left operand must be lval
                   1730: ** CVTO          convert the op
                   1731: ** NCVT          do not convert the operands
                   1732: ** OTHER         handled by code
                   1733: ** NCVTR         convert the left operand, not the right...
                   1734: **
                   1735: */
                   1736: 
                   1737: # define MINT 01  /* integer */
                   1738: # define MDBI 02   /* integer or double */
                   1739: # define MSTR 04  /* structure */
                   1740: # define MPTR 010  /* pointer */
                   1741: # define MPTI 020  /* pointer or integer */
                   1742: # define MENU 040 /* enumeration variable or member */
                   1743: 
                   1744: opact( p )
                   1745: register NODE *p;
                   1746: {
                   1747:        register mt12, mt1, mt2, o;
                   1748: 
                   1749:        mt12 = 0;
                   1750: 
                   1751:        switch( optype(o=p->in.op) )
                   1752:        {
                   1753: 
                   1754:        case BITYPE:
                   1755:                mt12=mt2 = moditype( p->in.right->in.type );
                   1756:        case UTYPE:
                   1757:                mt12 &= (mt1 = moditype( p->in.left->in.type ));
                   1758: 
                   1759:        }
                   1760: 
                   1761:        switch( o )
                   1762:        {
                   1763: 
                   1764:        case NAME :
                   1765:        case STRING :
                   1766:        case ICON :
                   1767:        case FCON :
                   1768:        case CALL :
                   1769:        case UNARY CALL:
                   1770:        case STAR:
                   1771:                {
                   1772:                          return( OTHER );
                   1773:                }
                   1774:        case UNARY MINUS:
                   1775:                if( mt1 & MDBI ) return( TYPL );
                   1776:                break;
                   1777: 
                   1778:        case COMPL:
                   1779:                if( mt1 & MINT ) return( TYPL );
                   1780:                break;
                   1781: 
                   1782:        case UNARY AND:
                   1783:                {
                   1784:                          return( NCVT+OTHER );
                   1785:                }
                   1786:        case INIT:
                   1787:        case CM:
                   1788:                return( 0 );
                   1789:        case NOT:
                   1790:        case CBRANCH:
                   1791:        case ANDAND:
                   1792:        case OROR:
                   1793:                return( NCVT );
                   1794: 
                   1795:        case MUL:
                   1796:        case DIV:
                   1797:                if( mt12 & MDBI ) return( TYMATCH );
                   1798:                break;
                   1799: 
                   1800:        case MOD:
                   1801:        case AND:
                   1802:        case OR:
                   1803:        case ER:
                   1804:        case LS:
                   1805:        case RS:
                   1806:                if( mt12 & MINT ) return( TYMATCH );
                   1807:                break;
                   1808: 
                   1809:        case EQ:
                   1810:        case NE:
                   1811:        case LT:
                   1812:        case LE:
                   1813:        case GT:
                   1814:        case GE:
                   1815:                if( (mt1&MENU)||(mt2&MENU) ) return( PTMATCH+PUN );
                   1816:                if( mt12 & MDBI ) return( TYMATCH+CVTO );
                   1817:                else if( mt12 & MPTR ) return( PTMATCH+PUN+CVTO );
                   1818:                else if( mt12 & MPTI ) return( PTMATCH+PUN+CVTO );
                   1819:                else break;
                   1820: 
                   1821:        case QUEST:
                   1822:        case COMOP:
                   1823:                return( TYPR+NCVTR );
                   1824: 
                   1825:        case STREF:
                   1826:                return( NCVTR+OTHER );
                   1827: 
                   1828:        case COLON:
                   1829:                if( mt12 & MENU ) return( NCVT+PUN+PTMATCH );
                   1830:                else if( mt12 & MDBI ) return( TYMATCH );
                   1831:                else if( mt12 & MPTR ) return( TYPL+PTMATCH+PUN );
                   1832:                else if( (mt1&MINT) && (mt2&MPTR) ) return( TYPR+PUN );
                   1833:                else if( (mt1&MPTR) && (mt2&MINT) ) return( TYPL+PUN );
                   1834:                else if( mt12 & MSTR ) return( NCVT+TYPL+OTHER );
                   1835:                break;
                   1836: 
                   1837:        case ASSIGN:
                   1838:        case RETURN:
                   1839:                if( mt12 & MSTR ) return( LVAL+NCVT+TYPL+OTHER );
                   1840:        case CAST:
                   1841:                if(o==CAST && mt1==0)return(TYPL+TYMATCH);
                   1842:                if( mt12 & MDBI ) return( TYPL+LVAL+TYMATCH );
                   1843:                else if( (mt1&MENU)||(mt2&MENU) )
                   1844:                {
                   1845:                        return( LVAL+NCVT+TYPL+PTMATCH+PUN );
                   1846:                }
                   1847:                else if( mt12 == 0 ) break;
                   1848:                else if( mt1 & MPTR ) return( LVAL+PTMATCH+PUN );
                   1849:                else if( mt12 & MPTI ) return( TYPL+LVAL+TYMATCH+PUN );
                   1850:                if(mt12 & MSTR) {
                   1851:                        uerror("no structure casts");
                   1852:                        return(NCVT);
                   1853:                }
                   1854:                break;
                   1855: 
                   1856:        case ASG MUL:
                   1857:        case ASG DIV:
                   1858:                if( mt12 & MDBI ) return( LVAL+TYMATCH );
                   1859:                break;
                   1860: 
                   1861:        case ASG MOD:
                   1862:        case ASG AND:
                   1863:        case ASG OR:
                   1864:        case ASG ER:
                   1865:        case ASG LS:
                   1866:        case ASG RS:
                   1867:                if( mt12 & MINT ) return( LVAL+TYMATCH );
                   1868:                break;
                   1869: 
                   1870:        case ASG PLUS:
                   1871:        case ASG MINUS:
                   1872:        case INCR:
                   1873:        case DECR:
                   1874:                if( mt12 & MDBI ) return( TYMATCH+LVAL );
                   1875:                else if( (mt1&MPTR) && (mt2&MINT) ) return( TYPL+LVAL+CVTR );
                   1876:                break;
                   1877: 
                   1878:        case MINUS:
                   1879:                if( mt12 & MPTR ) return( CVTO+PTMATCH+PUN );
                   1880:                if( mt2 & MPTR ) break;
                   1881:        case PLUS:
                   1882:                if( mt12 & MDBI ) return( TYMATCH );
                   1883:                else if( (mt1&MPTR) && (mt2&MINT) ) return( TYPL+CVTR );
                   1884:                else if( (mt1&MINT) && (mt2&MPTR) ) return( TYPR+CVTL );
                   1885:                else break;
                   1886:                /* special operators */
                   1887:        case UOP0:
                   1888:        case UOP1:
                   1889:        case UOP2:
                   1890:        case UOP3:
                   1891:        case UOP4:
                   1892:        case UOP5:
                   1893:        case UOP6:
                   1894:        case UOP7:
                   1895:        case UOP8:
                   1896:        case UOP9:
                   1897:                return( TYPL );
                   1898:        }
                   1899:        uerror( "operands of %s have incompatible types", opst[o] );
                   1900:        return( NCVT );
                   1901: }
                   1902: 
                   1903: moditype( ty )
                   1904: register TWORD ty;
                   1905: {
                   1906: 
                   1907:        switch( ty )
                   1908:        {
                   1909: 
                   1910:        case UNDEF:
                   1911:        case VOID:
                   1912:                return(0); /* type is void */
                   1913:        case ENUMTY:
                   1914:        case MOETY:
                   1915:                /* pjw says enum is int return( MENU ); */
                   1916:                return(MINT|MDBI|MPTI);
                   1917:        case STRTY:
                   1918:        case UNIONTY:
                   1919:                return( MSTR );
                   1920: 
                   1921:        case CHAR:
                   1922:        case SHORT:
                   1923:        case UCHAR:
                   1924:        case USHORT:
                   1925:                return( MINT|MPTI|MDBI );
                   1926: 
                   1927:        case UNSIGNED:
                   1928:        case ULONG:
                   1929:        case INT:
                   1930:        case LONG:
                   1931:                return( MINT|MDBI|MPTI );
                   1932: 
                   1933:        case FLOAT:
                   1934:        case DOUBLE:
                   1935:                return( MDBI );
                   1936: 
                   1937:        default:
                   1938:                return( MPTR|MPTI );
                   1939:        }
                   1940: }
                   1941: 
                   1942: # ifndef MYCCAST
                   1943: CONSZ
                   1944: ccast( v, t )
                   1945: register CONSZ v;
                   1946: register TWORD t;
                   1947: {
                   1948:        /* cast value v into simple type t */
                   1949:        /* must be done as it would be on the target machine */
                   1950: 
                   1951:        switch( t )
                   1952:        {
                   1953: 
                   1954:        case CHAR:
                   1955: # ifdef CHSIGN
                   1956:                if( v&ONEBIT(SZCHAR-1) )
                   1957:                {
                   1958:                        return( v | ~BITMASK(SZCHAR) );
                   1959:                }
                   1960: # endif
                   1961:        case UCHAR:
                   1962:                return( v & BITMASK(SZCHAR) );
                   1963: 
                   1964:        case SHORT:
                   1965:                if( v&ONEBIT(SZSHORT-1) )
                   1966:                {
                   1967:                        return( v | ~BITMASK(SZSHORT) );
                   1968:                }
                   1969:        case USHORT:
                   1970:                return( v & BITMASK(SZSHORT) );
                   1971: 
                   1972:        case INT:
                   1973:                if( v&ONEBIT(SZINT-1) )
                   1974:                {
                   1975:                        return( v | ~BITMASK(SZINT) );
                   1976:                }
                   1977:        case UNSIGNED:
                   1978:                return( v & BITMASK(SZINT) );
                   1979: 
                   1980:        default:
                   1981:                return( v );
                   1982:        }
                   1983: }
                   1984: # endif
                   1985: 
                   1986: 
                   1987: NODE *
                   1988: doszof( p )
                   1989: register NODE *p;
                   1990: {
                   1991:        /* do sizeof p */
                   1992:        register i;
                   1993: 
                   1994:        /* whatever is the meaning of this if it is a bitfield? */
                   1995:        i = tsize( p->in.type, p->fn.cdim, p->fn.csiz )/SZCHAR;
                   1996: 
                   1997:        tfree(p);
                   1998:        if( i <= 0 ) werror( "sizeof returns 0" );
                   1999:        p = bcon(i);
                   2000:        p->tn.type = UNSIGNED;  /* damn dmr anyhow! */
                   2001:        return( p );
                   2002: }
                   2003: 
                   2004: # ifndef NODBG
                   2005: eprint(p)
                   2006: NODE *p;
                   2007: {
                   2008:        printf("\n++++++++\n");
                   2009:        e1print(p,"T");
                   2010:        printf("---------\n");
                   2011: }
                   2012: e1print( p ,s)
                   2013: register NODE *p;
                   2014: char *s;
                   2015: {
                   2016:        register ty, d;
                   2017:        static down = 0;
                   2018: 
                   2019:        ty = optype( p->tn.op );
                   2020: 
                   2021:        if( ty == BITYPE )
                   2022:        {
                   2023:                ++down;
                   2024:                e1print( p->in.right ,"R");
                   2025:                --down;
                   2026:        }
                   2027: 
                   2028:        for( d=down; d>1; d -= 2 ) printf( "\t" );
                   2029:        if( d ) printf( "    " );
                   2030: 
                   2031:        printf("%s=%d) %s, ", s, (int) (p-node), opst[p->in.op] );
                   2032:        if( ty == LTYPE )
                   2033:        {
                   2034:                printf( "lval=%ld", p->tn.lval );
                   2035:                printf( ", rval=%d, ", p->tn.rval );
                   2036:        }
                   2037:        printf("type=");
                   2038:        tprint( p->in.type );
                   2039:        printf( ", dim=%d, siz=%d\n", p->fn.cdim, p->fn.csiz );
                   2040:        if( ty != LTYPE )
                   2041:        {
                   2042:                ++down;
                   2043:                e1print( p->in.left ,"L");
                   2044:                --down;
                   2045:        }
                   2046: }
                   2047: 
                   2048: tprint( t )
                   2049: register TWORD t; 
                   2050: {
                   2051:         /* output a nice description of the type of t */
                   2052:        static char * tnames[] = 
                   2053:        {
                   2054:                "undef",
                   2055:                "farg",
                   2056:                "char",
                   2057:                "short",
                   2058:                "int",
                   2059:                "long",
                   2060:                "float",
                   2061:                "double",
                   2062:                "strty",
                   2063:                "unionty",
                   2064:                "enumty",
                   2065:                "moety",
                   2066:                "uchar",
                   2067:                "ushort",
                   2068:                "unsigned",
                   2069:                "ulong",
                   2070:                "?", "?"
                   2071:        };
                   2072: 
                   2073:        for(;; t = DECREF(t) )
                   2074:        {
                   2075:                if( ISPTR(t) ) printf( "PTR " );
                   2076:                else if( ISFTN(t) ) printf( "FTN " );
                   2077:                else if( ISARY(t) ) printf( "ARY " );
                   2078:                else 
                   2079:                {
                   2080:                        printf( "%s", tnames[t] );
                   2081:                        return;
                   2082:                }
                   2083:        }
                   2084: }
                   2085: # endif
                   2086: 
                   2087: # ifndef MYLOCCTR
                   2088: locctr(l)
                   2089: register l;
                   2090: {
                   2091:        register temp, lt;
                   2092:        /* look in locnames; print out the location counter name */
                   2093:        /* null means use the next; all null, don't print */
                   2094:        for( lt=l; lt<=STRNG && !locnames[lt]; ++lt )
                   2095:        {
                   2096:                 /* EMPTY */
                   2097:        }
                   2098:        if( lt == curloc ) return( lt );
                   2099:        temp = curloc;
                   2100:        if( lt > STRNG ) lt = l;
                   2101:        else printx( locnames[lt] );
                   2102:        curloc = lt;
                   2103:        return( temp );
                   2104: }
                   2105: # endif
                   2106: 
                   2107: # ifndef NOFLOAT
                   2108: 
                   2109: prtdcon( p )
                   2110: register NODE *p;
                   2111: {
                   2112:        register i;
                   2113:        register TWORD t;
                   2114: 
                   2115:        if( p->in.op == FCON )
                   2116:        {
                   2117:                locctr( DATA );
                   2118:                t = p->tn.type;
                   2119:                defalign( t==DOUBLE?ALDOUBLE:ALFLOAT );
                   2120:                deflab( i = getlab() );
                   2121:                fincode( p->fpn.dval, t==DOUBLE?SZDOUBLE:SZFLOAT );
                   2122:                p->tn.lval = 0;
                   2123:                p->tn.rval = -i;
                   2124:                p->in.op = NAME;
                   2125:        }
                   2126:        if( (i = optype(p->in.op)) == BITYPE ) prtdcon( p->in.right );
                   2127:        if( i != LTYPE ) prtdcon( p->in.left );
                   2128: }
                   2129: # endif
                   2130: 
                   2131: # ifndef MYLABELS
                   2132: getlab()
                   2133: {
                   2134:        static crslab = 10;
                   2135:        return( ++crslab );
                   2136: }
                   2137: # endif
                   2138: 
                   2139: int edebug = 0;
                   2140: ecomp( p )
                   2141: register NODE *p;
                   2142: {
                   2143:        slineno = p->ln.lineno;
                   2144: # ifndef NODBG
                   2145:        if( edebug ) eprint(p);
                   2146: # endif
                   2147:        if( !reached )
                   2148:        {
                   2149:                werror( "statement not reached" );
                   2150:                reached = 1;
                   2151:        }
                   2152: # ifdef CLOCAL
                   2153:        p = clocal(p);
                   2154: # endif
                   2155:        p = optim(p);
                   2156: # ifndef NOFLOAT
                   2157:        prtdcon(p);
                   2158: # endif
                   2159:        locctr( PROG );
                   2160:        ecode(p);
                   2161:        tfree(p);
                   2162: }
                   2163: 
                   2164: # ifndef MYECODE
                   2165: ecode( p )
                   2166: register NODE *p;
                   2167: {
                   2168:        /* standard version of writing the tree nodes */
                   2169:        if( nerrors ) return;
                   2170: # ifdef GDEBUG
                   2171:        dbline();
                   2172: # endif
                   2173:        p2tree( p );
                   2174:        p2compile( p );
                   2175: }
                   2176: # endif
                   2177: 
                   2178: # ifndef MYPRTREE
                   2179: 
                   2180: # ifndef RNODNAME
                   2181: # define RNODNAME LABFMT
                   2182: # endif
                   2183: 
                   2184: p2tree(p)
                   2185: register NODE *p;
                   2186: {
                   2187:        register ty;
                   2188:        register NODE *l;
                   2189:        register o;
                   2190:        char temp[32];                  /* place to dump label stuff */
                   2191: 
                   2192: # ifdef MYP2TREE
                   2193:        MYP2TREE(p);  /* local action can be taken here; then return... */
                   2194: # endif
                   2195: 
                   2196:        /* this routine sits painfully between pass1 and pass2 */
                   2197:        ty = optype(o=p->in.op);
                   2198:        p->tn.goal = 0;  /* an illegal goal, just to clear it out */
                   2199:        p->tn.type = ttype( p->tn.type );  /* type gets second-pass (bits) */
                   2200: 
                   2201:        switch( o )
                   2202:        {
                   2203: 
                   2204:        case TEMP:
                   2205:        case NAME:
                   2206:        case ICON:
                   2207:        case VAUTO:
                   2208:        case VPARAM:
                   2209:                if( p->tn.rval == NONAME )
                   2210:                        p->in.name = (char *) 0;
                   2211:                else if( p->tn.rval >= 0 )
                   2212:                {
                   2213:                         /* copy name from exname */
                   2214:                        register char *cp;
                   2215:                        cp = exname( stab[p->tn.rval].sname );
                   2216:                        p->in.name = tstr( cp );
                   2217:                }
                   2218:                else if( p->tn.rval == - strftn )
                   2219:                {
                   2220:                        sprintf( temp, RNODNAME, -p->tn.rval );
                   2221:                        p->in.name = tstr( temp );
                   2222:                }
                   2223:                else
                   2224:                {
                   2225:                        sprintf( temp, LABFMT, -p->tn.rval );
                   2226:                        p->in.name = tstr( temp );
                   2227:                }
                   2228:                break;
                   2229: 
                   2230:        case STARG:
                   2231:        case STASG:
                   2232:        case STCALL:
                   2233:        case UNARY STCALL:
                   2234:                /* set up size parameters */
                   2235:                l = p->in.left;
                   2236:                p->stn.stsize = tsize(STRTY,l->fn.cdim,l->fn.csiz);
                   2237:                p->stn.stalign = talign(STRTY,l->fn.csiz);
                   2238:                break;
                   2239: 
                   2240:                /* this should do something only if temporary regs are
                   2241:                /* built into the tree by machine-dependent actions */
                   2242:        case REG:
                   2243:                rbusy( p->tn.rval, p->in.type );
                   2244:        default:
                   2245:                p->in.name = (char *) 0;
                   2246:        }
                   2247: 
                   2248:        if( ty != LTYPE ) p2tree( p->in.left );
                   2249:        if( ty == BITYPE ) p2tree( p->in.right );
                   2250: }
                   2251: 
                   2252: # endif

unix.superglobalmegacorp.com

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