Annotation of 43BSDTahoe/lib/mip/pftn.c, revision 1.1

1.1     ! root        1: #ifndef lint
        !             2: static char *sccsid ="@(#)pftn.c       1.27 (Berkeley) 12/10/87";
        !             3: #endif lint
        !             4: 
        !             5: # include "pass1.h"
        !             6: 
        !             7: unsigned int offsz;
        !             8: 
        !             9: struct symtab *schain[MAXSCOPES];      /* sym chains for clearst */
        !            10: int chaintop;                          /* highest active entry */
        !            11: 
        !            12: struct instk {
        !            13:        int in_sz;   /* size of array element */
        !            14:        int in_x;    /* current index for structure member in structure initializations */
        !            15:        int in_n;    /* number of initializations seen */
        !            16:        int in_s;    /* sizoff */
        !            17:        int in_d;    /* dimoff */
        !            18:        TWORD in_t;    /* type */
        !            19:        int in_id;   /* stab index */
        !            20:        int in_fl;   /* flag which says if this level is controlled by {} */
        !            21:        OFFSZ in_off;  /* offset of the beginning of this level */
        !            22:        }
        !            23: instack[10],
        !            24: *pstk;
        !            25: 
        !            26:        /* defines used for getting things off of the initialization stack */
        !            27: 
        !            28: 
        !            29: struct symtab *relook();
        !            30: 
        !            31: 
        !            32: int ddebug = 0;
        !            33: 
        !            34: struct symtab * mknonuniq();
        !            35: 
        !            36: defid( q, class ) register NODE *q; register int class; {
        !            37:        register struct symtab *p;
        !            38:        int idp;
        !            39:        register TWORD type;
        !            40:        TWORD stp;
        !            41:        register int scl;
        !            42:        int dsym, ddef;
        !            43:        int slev, temp;
        !            44:        int changed;
        !            45: 
        !            46:        if( q == NIL ) return;  /* an error was detected */
        !            47: 
        !            48:        if( q < node || q >= &node[TREESZ] ) cerror( "defid call" );
        !            49: 
        !            50:        idp = q->tn.rval;
        !            51: 
        !            52:        if( idp < 0 ) cerror( "tyreduce" );
        !            53:        p = &stab[idp];
        !            54: 
        !            55: # ifndef BUG1
        !            56:        if( ddebug ){
        !            57: #ifndef FLEXNAMES
        !            58:                printf( "defid( %.8s (%d), ", p->sname, idp );
        !            59: #else
        !            60:                printf( "defid( %s (%d), ", p->sname, idp );
        !            61: #endif
        !            62:                tprint( q->in.type );
        !            63:                printf( ", %s, (%d,%d) ), level %d\n", scnames(class), q->fn.cdim, q->fn.csiz, blevel );
        !            64:                }
        !            65: # endif
        !            66: 
        !            67:        fixtype( q, class );
        !            68: 
        !            69:        type = q->in.type;
        !            70:        class = fixclass( class, type );
        !            71: 
        !            72:        stp = p->stype;
        !            73:        slev = p->slevel;
        !            74: 
        !            75: # ifndef BUG1
        !            76:        if( ddebug ){
        !            77:                printf( "       modified to " );
        !            78:                tprint( type );
        !            79:                printf( ", %s\n", scnames(class) );
        !            80:                printf( "       previous def'n: " );
        !            81:                tprint( stp );
        !            82:                printf( ", %s, (%d,%d) ), level %d\n", scnames(p->sclass), p->dimoff, p->sizoff, slev );
        !            83:                }
        !            84: # endif
        !            85: 
        !            86:        if( stp == FTN && p->sclass == SNULL )goto enter;
        !            87:        if( blevel==1 && stp!=FARG ) switch( class ){
        !            88: 
        !            89:                default:
        !            90: #ifndef FLEXNAMES
        !            91:                        if(!(class&FIELD)) uerror( "declared argument %.8s is missing", p->sname );
        !            92: #else
        !            93:                        if(!(class&FIELD)) uerror( "declared argument %s is missing", p->sname );
        !            94: #endif
        !            95:                case MOS:
        !            96:                case STNAME:
        !            97:                case MOU:
        !            98:                case UNAME:
        !            99:                case MOE:
        !           100:                case ENAME:
        !           101:                case TYPEDEF:
        !           102:                        ;
        !           103:                        }
        !           104:        if( stp == UNDEF|| stp == FARG ) goto enter;
        !           105: 
        !           106:        if( type != stp ) goto mismatch;
        !           107:        if( blevel > slev && (class == AUTO || class == REGISTER) )
        !           108:                /* new scope */
        !           109:                goto mismatch;
        !           110: 
        !           111:        /* test (and possibly adjust) dimensions */
        !           112:        dsym = p->dimoff;
        !           113:        ddef = q->fn.cdim;
        !           114:        changed = 0;
        !           115:        for( temp=type; temp&TMASK; temp = DECREF(temp) ){
        !           116:                if( ISARY(temp) ){
        !           117:                        if (dimtab[dsym] == 0) {
        !           118:                                dimtab[dsym] = dimtab[ddef];
        !           119:                                changed = 1;
        !           120:                                }
        !           121:                        else if (dimtab[ddef]!=0&&dimtab[dsym]!=dimtab[ddef]) {
        !           122:                                goto mismatch;
        !           123:                                }
        !           124:                        ++dsym;
        !           125:                        ++ddef;
        !           126:                        }
        !           127:                }
        !           128: 
        !           129:        if (changed) {
        !           130:                FIXDEF(p);
        !           131:                }
        !           132: 
        !           133:        /* check that redeclarations are to the same structure */
        !           134:        if( (temp==STRTY||temp==UNIONTY||temp==ENUMTY) && p->sizoff != q->fn.csiz
        !           135:                 && class!=STNAME && class!=UNAME && class!=ENAME ){
        !           136:                goto mismatch;
        !           137:                }
        !           138: 
        !           139:        scl = ( p->sclass );
        !           140: 
        !           141: # ifndef BUG1
        !           142:        if( ddebug ){
        !           143:                printf( "       previous class: %s\n", scnames(scl) );
        !           144:                }
        !           145: # endif
        !           146: 
        !           147:        if( class&FIELD ){
        !           148:                /* redefinition */
        !           149:                if( !falloc( p, class&FLDSIZ, 1, NIL ) ) {
        !           150:                        /* successful allocation */
        !           151:                        psave( idp );
        !           152:                        return;
        !           153:                        }
        !           154:                /* blew it: resume at end of switch... */
        !           155:                }
        !           156: 
        !           157:        else switch( class ){
        !           158: 
        !           159:        case EXTERN:
        !           160:                switch( scl ){
        !           161:                case STATIC:
        !           162:                case USTATIC:
        !           163:                        if( slev==0 ) return;
        !           164:                        break;
        !           165:                case EXTDEF:
        !           166:                case EXTERN:
        !           167:                case FORTRAN:
        !           168:                case UFORTRAN:
        !           169:                        return;
        !           170:                        }
        !           171:                break;
        !           172: 
        !           173:        case STATIC:
        !           174:                if( scl==USTATIC || (scl==EXTERN && blevel==0) ){
        !           175:                        p->sclass = STATIC;
        !           176:                        if( ISFTN(type) ) curftn = idp;
        !           177:                        return;
        !           178:                        }
        !           179:                break;
        !           180: 
        !           181:        case USTATIC:
        !           182:                if( scl==STATIC || scl==USTATIC ) return;
        !           183:                break;
        !           184: 
        !           185:        case LABEL:
        !           186:                if( scl == ULABEL ){
        !           187:                        p->sclass = LABEL;
        !           188:                        deflab( p->offset );
        !           189:                        return;
        !           190:                        }
        !           191:                break;
        !           192: 
        !           193:        case TYPEDEF:
        !           194:                if( scl == class ) return;
        !           195:                break;
        !           196: 
        !           197:        case UFORTRAN:
        !           198:                if( scl == UFORTRAN || scl == FORTRAN ) return;
        !           199:                break;
        !           200: 
        !           201:        case FORTRAN:
        !           202:                if( scl == UFORTRAN ){
        !           203:                        p->sclass = FORTRAN;
        !           204:                        if( ISFTN(type) ) curftn = idp;
        !           205:                        return;
        !           206:                        }
        !           207:                break;
        !           208: 
        !           209:        case MOU:
        !           210:        case MOS:
        !           211:                if( scl == class ) {
        !           212:                        if( oalloc( p, &strucoff ) ) break;
        !           213:                        if( class == MOU ) strucoff = 0;
        !           214:                        psave( idp );
        !           215:                        return;
        !           216:                        }
        !           217:                break;
        !           218: 
        !           219:        case MOE:
        !           220:                if( scl == class ){
        !           221:                        if( p->offset!= strucoff++ ) break;
        !           222:                        psave( idp );
        !           223:                        }
        !           224:                break;
        !           225: 
        !           226:        case EXTDEF:
        !           227:                if( scl == EXTERN ) {
        !           228:                        p->sclass = EXTDEF;
        !           229:                        if( ISFTN(type) ) curftn = idp;
        !           230:                        return;
        !           231:                        }
        !           232:                break;
        !           233: 
        !           234:        case STNAME:
        !           235:        case UNAME:
        !           236:        case ENAME:
        !           237:                if( scl != class ) break;
        !           238:                if( dimtab[p->sizoff] == 0 ) return;  /* previous entry just a mention */
        !           239:                break;
        !           240: 
        !           241:        case ULABEL:
        !           242:                if( scl == LABEL || scl == ULABEL ) return;
        !           243:        case PARAM:
        !           244:        case AUTO:
        !           245:        case REGISTER:
        !           246:                ;  /* mismatch.. */
        !           247: 
        !           248:                }
        !           249: 
        !           250:        mismatch:
        !           251:        /* allow nonunique structure/union member names */
        !           252: 
        !           253:        if( class==MOU || class==MOS || class & FIELD ){/* make a new entry */
        !           254:                register int *memp;
        !           255:                p->sflags |= SNONUNIQ;  /* old entry is nonunique */
        !           256:                /* determine if name has occurred in this structure/union */
        !           257:                if (paramno > 0) for( memp = &paramstk[paramno-1];
        !           258:                        /* while */ *memp>=0 && stab[*memp].sclass != STNAME
        !           259:                                && stab[*memp].sclass != UNAME;
        !           260:                        /* iterate */ --memp){ char *cname, *oname;
        !           261:                        if( stab[*memp].sflags & SNONUNIQ ){
        !           262:                                cname=p->sname;
        !           263:                                oname=stab[*memp].sname;
        !           264: #ifndef FLEXNAMES
        !           265:                                for(temp=1; temp<=NCHNAM; ++temp){
        !           266:                                        if(*cname++ != *oname)goto diff;
        !           267:                                        if(!*oname++)break;
        !           268:                                        }
        !           269: #else
        !           270:                                if (cname != oname) goto diff;
        !           271: #endif
        !           272:                                uerror("redeclaration of: %s",p->sname);
        !           273:                                break;
        !           274:                                diff: continue;
        !           275:                                }
        !           276:                        }
        !           277:                p = mknonuniq( &idp ); /* update p and idp to new entry */
        !           278:                goto enter;
        !           279:                }
        !           280:        if( blevel > slev && class != EXTERN && class != FORTRAN &&
        !           281:                class != UFORTRAN && !( class == LABEL && slev >= 2 ) ){
        !           282:                q->tn.rval = idp = hide( p );
        !           283:                p = &stab[idp];
        !           284:                goto enter;
        !           285:                }
        !           286: #ifndef FLEXNAMES
        !           287:        uerror( "redeclaration of %.8s", p->sname );
        !           288: #else
        !           289:        uerror( "redeclaration of %s", p->sname );
        !           290: #endif
        !           291:        if( class==EXTDEF && ISFTN(type) ) curftn = idp;
        !           292:        return;
        !           293: 
        !           294:        enter:  /* make a new entry */
        !           295: 
        !           296: # ifndef BUG1
        !           297:        if( ddebug ) printf( "  new entry made\n" );
        !           298: # endif
        !           299:        if( type == UNDEF ) uerror("void type for %s",p->sname);
        !           300:        p->stype = type;
        !           301:        p->sclass = class;
        !           302:        p->slevel = blevel;
        !           303:        p->offset = NOOFFSET;
        !           304:        p->suse = lineno;
        !           305:        if( class == STNAME || class == UNAME || class == ENAME ) {
        !           306:                p->sizoff = curdim;
        !           307:                dstash( 0 );  /* size */
        !           308:                dstash( -1 ); /* index to members of str or union */
        !           309:                dstash( ALSTRUCT );  /* alignment */
        !           310:                dstash( idp );
        !           311:                }
        !           312:        else {
        !           313:                switch( BTYPE(type) ){
        !           314:                case STRTY:
        !           315:                case UNIONTY:
        !           316:                case ENUMTY:
        !           317:                        p->sizoff = q->fn.csiz;
        !           318:                        break;
        !           319:                default:
        !           320:                        p->sizoff = BTYPE(type);
        !           321:                        }
        !           322:                }
        !           323: 
        !           324:        /* copy dimensions */
        !           325: 
        !           326:        p->dimoff = q->fn.cdim;
        !           327: 
        !           328:        /* allocate offsets */
        !           329:        if( class&FIELD ){
        !           330:                (void) falloc( p, class&FLDSIZ, 0, NIL );  /* new entry */
        !           331:                psave( idp );
        !           332:                }
        !           333:        else switch( class ){
        !           334: 
        !           335:        case AUTO:
        !           336:                (void) oalloc( p, &autooff );
        !           337:                break;
        !           338:        case STATIC:
        !           339:        case EXTDEF:
        !           340:                p->offset = getlab();
        !           341:                if( ISFTN(type) ) curftn = idp;
        !           342:                break;
        !           343:        case ULABEL:
        !           344:        case LABEL:
        !           345:                p->offset = getlab();
        !           346:                p->slevel = 2;
        !           347:                if( class == LABEL ){
        !           348:                        (void) locctr( PROG );
        !           349:                        deflab( p->offset );
        !           350:                        }
        !           351:                break;
        !           352: 
        !           353:        case EXTERN:
        !           354:        case UFORTRAN:
        !           355:        case FORTRAN:
        !           356:                p->offset = getlab();
        !           357:                p->slevel = 0;
        !           358:                break;
        !           359:        case MOU:
        !           360:        case MOS:
        !           361:                (void) oalloc( p, &strucoff );
        !           362:                if( class == MOU ) strucoff = 0;
        !           363:                psave( idp );
        !           364:                break;
        !           365: 
        !           366:        case MOE:
        !           367:                p->offset = strucoff++;
        !           368:                psave( idp );
        !           369:                break;
        !           370:        case REGISTER:
        !           371:                p->offset = regvar--;
        !           372:                if( blevel == 1 ) p->sflags |= SSET;
        !           373:                if( regvar < minrvar ) minrvar = regvar;
        !           374:                break;
        !           375:                }
        !           376: 
        !           377:        {
        !           378:                register int l = p->slevel;
        !           379: 
        !           380:                if( l >= MAXSCOPES )
        !           381:                        cerror( "scopes nested too deep" );
        !           382: 
        !           383:                p->snext = schain[l];
        !           384:                schain[l] = p;
        !           385:                if( l >= chaintop )
        !           386:                        chaintop = l + 1;
        !           387:                }
        !           388: 
        !           389:        /* user-supplied routine to fix up new definitions */
        !           390: 
        !           391:        FIXDEF(p);
        !           392: 
        !           393: # ifndef BUG1
        !           394:        if( ddebug ) printf( "  dimoff, sizoff, offset: %d, %d, %d\n", p->dimoff, p->sizoff, p->offset );
        !           395: # endif
        !           396: 
        !           397:        }
        !           398: 
        !           399: psave( i ){
        !           400:        if( paramno >= PARAMSZ ){
        !           401:                cerror( "parameter stack overflow");
        !           402:                }
        !           403:        paramstk[ paramno++ ] = i;
        !           404:        }
        !           405: 
        !           406: ftnend(){ /* end of function */
        !           407:        if( retlab != NOLAB && nerrors == 0 ){ /* inside a real function */
        !           408:                efcode();
        !           409:                }
        !           410:        checkst(0);
        !           411:        retstat = 0;
        !           412:        tcheck();
        !           413:        curclass = SNULL;
        !           414:        brklab = contlab = retlab = NOLAB;
        !           415:        flostat = 0;
        !           416:        if( nerrors == 0 ){
        !           417:                if( psavbc != & asavbc[0] ) cerror("bcsave error");
        !           418:                if( paramno != 0 ) cerror("parameter reset error");
        !           419:                if( swx != 0 ) cerror( "switch error");
        !           420:                }
        !           421:        psavbc = &asavbc[0];
        !           422:        paramno = 0;
        !           423:        autooff = AUTOINIT;
        !           424:        minrvar = regvar = MAXRVAR;
        !           425:        reached = 1;
        !           426:        swx = 0;
        !           427:        swp = swtab;
        !           428:        (void) locctr(DATA);
        !           429:        }
        !           430: 
        !           431: dclargs(){
        !           432:        register i, j;
        !           433:        register struct symtab *p;
        !           434:        register NODE *q;
        !           435:        argoff = ARGINIT;
        !           436: # ifndef BUG1
        !           437:        if( ddebug > 2) printf("dclargs()\n");
        !           438: # endif
        !           439:        for( i=0; i<paramno; ++i ){
        !           440:                if( (j = paramstk[i]) < 0 ) continue;
        !           441:                p = &stab[j];
        !           442: # ifndef BUG1
        !           443:                if( ddebug > 2 ){
        !           444:                        printf("\t%s (%d) ",p->sname, j);
        !           445:                        tprint(p->stype);
        !           446:                        printf("\n");
        !           447:                        }
        !           448: # endif
        !           449:                if( p->stype == FARG ) {
        !           450:                        q = block(FREE,NIL,NIL,INT,0,INT);
        !           451:                        q->tn.rval = j;
        !           452:                        defid( q, PARAM );
        !           453:                        }
        !           454:                FIXARG(p); /* local arg hook, eg. for sym. debugger */
        !           455:                oalloc( p, &argoff );  /* always set aside space, even for register arguments */
        !           456:                }
        !           457:        cendarg();
        !           458:        (void) locctr(PROG);
        !           459:        defalign(ALINT);
        !           460:        ftnno = getlab();
        !           461:        bfcode( paramstk, paramno );
        !           462:        paramno = 0;
        !           463:        }
        !           464: 
        !           465: NODE *
        !           466: rstruct( idn, soru ){ /* reference to a structure or union, with no definition */
        !           467:        register struct symtab *p;
        !           468:        register NODE *q;
        !           469:        p = &stab[idn];
        !           470:        switch( p->stype ){
        !           471: 
        !           472:        case UNDEF:
        !           473:        def:
        !           474:                q = block( FREE, NIL, NIL, 0, 0, 0 );
        !           475:                q->tn.rval = idn;
        !           476:                q->in.type = (soru&INSTRUCT) ? STRTY : ( (soru&INUNION) ? UNIONTY : ENUMTY );
        !           477:                defid( q, (soru&INSTRUCT) ? STNAME : ( (soru&INUNION) ? UNAME : ENAME ) );
        !           478:                break;
        !           479: 
        !           480:        case STRTY:
        !           481:                if( soru & INSTRUCT ) break;
        !           482:                goto def;
        !           483: 
        !           484:        case UNIONTY:
        !           485:                if( soru & INUNION ) break;
        !           486:                goto def;
        !           487: 
        !           488:        case ENUMTY:
        !           489:                if( !(soru&(INUNION|INSTRUCT)) ) break;
        !           490:                goto def;
        !           491: 
        !           492:                }
        !           493:        stwart = instruct;
        !           494:        return( mkty( p->stype, 0, p->sizoff ) );
        !           495:        }
        !           496: 
        !           497: moedef( idn ){
        !           498:        register NODE *q;
        !           499: 
        !           500:        q = block( FREE, NIL, NIL, MOETY, 0, 0 );
        !           501:        q->tn.rval = idn;
        !           502:        if( idn>=0 ) defid( q, MOE );
        !           503:        }
        !           504: 
        !           505: bstruct( idn, soru ){ /* begining of structure or union declaration */
        !           506:        register NODE *q;
        !           507: 
        !           508:        psave( instruct );
        !           509:        psave( curclass );
        !           510:        psave( strucoff );
        !           511:        strucoff = 0;
        !           512:        instruct = soru;
        !           513:        q = block( FREE, NIL, NIL, 0, 0, 0 );
        !           514:        q->tn.rval = idn;
        !           515:        if( instruct==INSTRUCT ){
        !           516:                curclass = MOS;
        !           517:                q->in.type = STRTY;
        !           518:                if( idn >= 0 ) defid( q, STNAME );
        !           519:                }
        !           520:        else if( instruct == INUNION ) {
        !           521:                curclass = MOU;
        !           522:                q->in.type = UNIONTY;
        !           523:                if( idn >= 0 ) defid( q, UNAME );
        !           524:                }
        !           525:        else { /* enum */
        !           526:                curclass = MOE;
        !           527:                q->in.type = ENUMTY;
        !           528:                if( idn >= 0 ) defid( q, ENAME );
        !           529:                }
        !           530:        psave( idn = q->tn.rval );
        !           531:        /* the "real" definition is where the members are seen */
        !           532:        if ( idn >= 0 ) stab[idn].suse = lineno;
        !           533:        return( paramno-4 );
        !           534:        }
        !           535: 
        !           536: NODE *
        !           537: dclstruct( oparam ){
        !           538:        register struct symtab *p;
        !           539:        register i, al, sa, j, sz, szindex;
        !           540:        register TWORD temp;
        !           541:        register high, low;
        !           542: 
        !           543:        /* paramstk contains:
        !           544:                paramstk[ oparam ] = previous instruct
        !           545:                paramstk[ oparam+1 ] = previous class
        !           546:                paramstk[ oparam+2 ] = previous strucoff
        !           547:                paramstk[ oparam+3 ] = structure name
        !           548: 
        !           549:                paramstk[ oparam+4, ... ]  = member stab indices
        !           550: 
        !           551:                */
        !           552: 
        !           553: 
        !           554:        if( (i=paramstk[oparam+3]) < 0 ){
        !           555:                szindex = curdim;
        !           556:                dstash( 0 );  /* size */
        !           557:                dstash( -1 );  /* index to member names */
        !           558:                dstash( ALSTRUCT );  /* alignment */
        !           559:                dstash( -lineno );      /* name of structure */
        !           560:                }
        !           561:        else {
        !           562:                szindex = stab[i].sizoff;
        !           563:                }
        !           564: 
        !           565: # ifndef BUG1
        !           566:        if( ddebug ){
        !           567: #ifndef FLEXNAMES
        !           568:                printf( "dclstruct( %.8s ), szindex = %d\n", (i>=0)? stab[i].sname : "??", szindex );
        !           569: #else
        !           570:                printf( "dclstruct( %s ), szindex = %d\n", (i>=0)? stab[i].sname : "??", szindex );
        !           571: #endif
        !           572:                }
        !           573: # endif
        !           574:        temp = (instruct&INSTRUCT)?STRTY:((instruct&INUNION)?UNIONTY:ENUMTY);
        !           575:        stwart = instruct = paramstk[ oparam ];
        !           576:        curclass = paramstk[ oparam+1 ];
        !           577:        dimtab[ szindex+1 ] = curdim;
        !           578:        al = ALSTRUCT;
        !           579: 
        !           580:        high = low = 0;
        !           581: 
        !           582:        for( i = oparam+4;  i< paramno; ++i ){
        !           583:                dstash( j=paramstk[i] );
        !           584:                if( j<0 || j>= SYMTSZ ) cerror( "gummy structure member" );
        !           585:                p = &stab[j];
        !           586:                if( temp == ENUMTY ){
        !           587:                        if( p->offset < low ) low = p->offset;
        !           588:                        if( p->offset > high ) high = p->offset;
        !           589:                        p->sizoff = szindex;
        !           590:                        continue;
        !           591:                        }
        !           592:                sa = talign( p->stype, p->sizoff );
        !           593:                if( p->sclass & FIELD ){
        !           594:                        sz = p->sclass&FLDSIZ;
        !           595:                        }
        !           596:                else {
        !           597:                        sz = tsize( p->stype, p->dimoff, p->sizoff );
        !           598:                        }
        !           599:                if( sz == 0 ){
        !           600: #ifndef FLEXNAMES
        !           601:                        werror( "illegal zero sized structure member: %.8s", p->sname );
        !           602: #else
        !           603:                        werror( "illegal zero sized structure member: %s", p->sname );
        !           604: #endif
        !           605:                        }
        !           606:                if( sz > strucoff ) strucoff = sz;  /* for use with unions */
        !           607:                SETOFF( al, sa );
        !           608:                /* set al, the alignment, to the lcm of the alignments of the members */
        !           609:                }
        !           610:        dstash( -1 );  /* endmarker */
        !           611:        SETOFF( strucoff, al );
        !           612: 
        !           613:        if( temp == ENUMTY ){
        !           614:                register TWORD ty;
        !           615: 
        !           616: # ifdef ENUMSIZE
        !           617:                ty = ENUMSIZE(high,low);
        !           618: # else
        !           619:                if( (char)high == high && (char)low == low ) ty = ctype( CHAR );
        !           620:                else if( (short)high == high && (short)low == low ) ty = ctype( SHORT );
        !           621:                else ty = ctype(INT);
        !           622: #endif
        !           623:                strucoff = tsize( ty, 0, (int)ty );
        !           624:                dimtab[ szindex+2 ] = al = talign( ty, (int)ty );
        !           625:                }
        !           626: 
        !           627:        if( strucoff == 0 ) uerror( "zero sized structure" );
        !           628:        dimtab[ szindex ] = strucoff;
        !           629:        dimtab[ szindex+2 ] = al;
        !           630:        dimtab[ szindex+3 ] = paramstk[ oparam+3 ];  /* name index */
        !           631: 
        !           632:        FIXSTRUCT( szindex, oparam ); /* local hook, eg. for sym debugger */
        !           633: # ifndef BUG1
        !           634:        if( ddebug>1 ){
        !           635:                printf( "\tdimtab[%d,%d,%d] = %d,%d,%d\n", szindex,szindex+1,szindex+2,
        !           636:                                dimtab[szindex],dimtab[szindex+1],dimtab[szindex+2] );
        !           637:                for( i = dimtab[szindex+1]; dimtab[i] >= 0; ++i ){
        !           638: #ifndef FLEXNAMES
        !           639:                        printf( "\tmember %.8s(%d)\n", stab[dimtab[i]].sname, dimtab[i] );
        !           640: #else
        !           641:                        printf( "\tmember %s(%d)\n", stab[dimtab[i]].sname, dimtab[i] );
        !           642: #endif
        !           643:                        }
        !           644:                }
        !           645: # endif
        !           646: 
        !           647:        strucoff = paramstk[ oparam+2 ];
        !           648:        paramno = oparam;
        !           649: 
        !           650:        return( mkty( temp, 0, szindex ) );
        !           651:        }
        !           652: 
        !           653:        /* VARARGS */
        !           654: yyerror( s ) char *s; { /* error printing routine in parser */
        !           655: 
        !           656:        uerror( s );
        !           657: 
        !           658:        }
        !           659: 
        !           660: yyaccpt(){
        !           661:        ftnend();
        !           662:        }
        !           663: 
        !           664: ftnarg( idn ) {
        !           665:        switch( stab[idn].stype ){
        !           666: 
        !           667:        case UNDEF:
        !           668:                /* this parameter, entered at scan */
        !           669:                break;
        !           670:        case FARG:
        !           671: #ifndef FLEXNAMES
        !           672:                uerror("redeclaration of formal parameter, %.8s",
        !           673: #else
        !           674:                uerror("redeclaration of formal parameter, %s",
        !           675: #endif
        !           676:                        stab[idn].sname);
        !           677:                /* fall thru */
        !           678:        case FTN:
        !           679:                /* the name of this function matches parm */
        !           680:                /* fall thru */
        !           681:        default:
        !           682:                idn = hide( &stab[idn]);
        !           683:                break;
        !           684:        case TNULL:
        !           685:                /* unused entry, fill it */
        !           686:                ;
        !           687:                }
        !           688:        stab[idn].stype = FARG;
        !           689:        stab[idn].sclass = PARAM;
        !           690:        psave( idn );
        !           691:        }
        !           692: 
        !           693: talign( ty, s) register unsigned ty; register s; {
        !           694:        /* compute the alignment of an object with type ty, sizeoff index s */
        !           695: 
        !           696:        register i;
        !           697:        if( s<0 && ty!=INT && ty!=CHAR && ty!=SHORT && ty!=UNSIGNED && ty!=UCHAR && ty!=USHORT 
        !           698: #ifdef LONGFIELDS
        !           699:                && ty!=LONG && ty!=ULONG
        !           700: #endif
        !           701:                                        ){
        !           702:                return( fldal( ty ) );
        !           703:                }
        !           704: 
        !           705:        for( i=0; i<=(SZINT-BTSHIFT-1); i+=TSHIFT ){
        !           706:                switch( (ty>>i)&TMASK ){
        !           707: 
        !           708:                case FTN:
        !           709:                        cerror( "compiler takes alignment of function");
        !           710:                case PTR:
        !           711:                        return( ALPOINT );
        !           712:                case ARY:
        !           713:                        continue;
        !           714:                case 0:
        !           715:                        break;
        !           716:                        }
        !           717:                }
        !           718: 
        !           719:        switch( BTYPE(ty) ){
        !           720: 
        !           721:        case UNIONTY:
        !           722:        case ENUMTY:
        !           723:        case STRTY:
        !           724:                return( (unsigned int) dimtab[ s+2 ] );
        !           725:        case CHAR:
        !           726:        case UCHAR:
        !           727:                return( ALCHAR );
        !           728:        case FLOAT:
        !           729:                return( ALFLOAT );
        !           730:        case DOUBLE:
        !           731:                return( ALDOUBLE );
        !           732:        case LONG:
        !           733:        case ULONG:
        !           734:                return( ALLONG );
        !           735:        case SHORT:
        !           736:        case USHORT:
        !           737:                return( ALSHORT );
        !           738:        default:
        !           739:                return( ALINT );
        !           740:                }
        !           741:        }
        !           742: 
        !           743: OFFSZ
        !           744: tsize( ty, d, s )  TWORD ty; {
        !           745:        /* compute the size associated with type ty,
        !           746:            dimoff d, and sizoff s */
        !           747:        /* BETTER NOT BE CALLED WHEN t, d, and s REFER TO A BIT FIELD... */
        !           748: 
        !           749:        int i;
        !           750:        OFFSZ mult;
        !           751: 
        !           752:        mult = 1;
        !           753: 
        !           754:        for( i=0; i<=(SZINT-BTSHIFT-1); i+=TSHIFT ){
        !           755:                switch( (ty>>i)&TMASK ){
        !           756: 
        !           757:                case FTN:
        !           758:                        /* cerror( "compiler takes size of function"); */
        !           759:                        uerror( "can't take size of function" );
        !           760:                        return( SZCHAR );
        !           761:                case PTR:
        !           762:                        return( SZPOINT * mult );
        !           763:                case ARY:
        !           764:                        mult *= (unsigned int) dimtab[ d++ ];
        !           765:                        continue;
        !           766:                case 0:
        !           767:                        break;
        !           768: 
        !           769:                        }
        !           770:                }
        !           771: 
        !           772:        if( dimtab[s]==0 ) {
        !           773:                if( ty == STRTY )
        !           774:                        uerror( "undefined structure" );
        !           775:                else
        !           776:                        uerror( "unknown size");
        !           777:                return( SZINT );
        !           778:                }
        !           779:        return( (unsigned int) dimtab[ s ] * mult );
        !           780:        }
        !           781: 
        !           782: inforce( n ) OFFSZ n; {  /* force inoff to have the value n */
        !           783:        /* inoff is updated to have the value n */
        !           784:        OFFSZ wb;
        !           785:        register rest;
        !           786:        /* rest is used to do a lot of conversion to ints... */
        !           787: 
        !           788:        if( inoff == n ) return;
        !           789:        if( inoff > n ) {
        !           790:                cerror( "initialization alignment error");
        !           791:                }
        !           792: 
        !           793:        wb = inoff;
        !           794:        SETOFF( wb, SZINT );
        !           795: 
        !           796:        /* wb now has the next higher word boundary */
        !           797: 
        !           798:        if( wb >= n ){ /* in the same word */
        !           799:                rest = n - inoff;
        !           800:                vfdzero( rest );
        !           801:                return;
        !           802:                }
        !           803: 
        !           804:        /* otherwise, extend inoff to be word aligned */
        !           805: 
        !           806:        rest = wb - inoff;
        !           807:        vfdzero( rest );
        !           808: 
        !           809:        /* now, skip full words until near to n */
        !           810: 
        !           811:        rest = (n-inoff)/SZINT;
        !           812:        zecode( rest );
        !           813: 
        !           814:        /* now, the remainder of the last word */
        !           815: 
        !           816:        rest = n-inoff;
        !           817:        vfdzero( rest );
        !           818:        if( inoff != n ) cerror( "inoff error");
        !           819: 
        !           820:        }
        !           821: 
        !           822: vfdalign( n ){ /* make inoff have the offset the next alignment of n */
        !           823:        OFFSZ m;
        !           824: 
        !           825:        m = inoff;
        !           826:        SETOFF( m, n );
        !           827:        inforce( m );
        !           828:        }
        !           829: 
        !           830: 
        !           831: int idebug = 0;
        !           832: 
        !           833: int ibseen = 0;  /* the number of } constructions which have been filled */
        !           834: 
        !           835: int ifull = 0; /* 1 if all initializers have been seen */
        !           836: 
        !           837: int iclass;  /* storage class of thing being initialized */
        !           838: 
        !           839: int ilocctr = 0;  /* location counter for current initialization */
        !           840: 
        !           841: beginit(curid){
        !           842:        /* beginning of initilization; set location ctr and set type */
        !           843:        register struct symtab *p;
        !           844: 
        !           845: # ifndef BUG1
        !           846:        if( idebug >= 3 ) printf( "beginit(), curid = %d\n", curid );
        !           847: # endif
        !           848: 
        !           849:        p = &stab[curid];
        !           850: 
        !           851:        iclass = p->sclass;
        !           852:        if( curclass == EXTERN || curclass == FORTRAN ) iclass = EXTERN;
        !           853:        switch( iclass ){
        !           854: 
        !           855:        case UNAME:
        !           856:        case EXTERN:
        !           857:                return;
        !           858:        case AUTO:
        !           859:        case REGISTER:
        !           860:                break;
        !           861:        case EXTDEF:
        !           862:        case STATIC:
        !           863:                ilocctr = ISARY(p->stype)?ADATA:DATA;
        !           864:                if( nerrors == 0 ){
        !           865:                        (void) locctr( ilocctr );
        !           866:                        defalign( talign( p->stype, p->sizoff ) );
        !           867:                        defnam( p );
        !           868:                        }
        !           869: 
        !           870:                }
        !           871: 
        !           872:        inoff = 0;
        !           873:        ibseen = 0;
        !           874:        ifull = 0;
        !           875: 
        !           876:        pstk = 0;
        !           877: 
        !           878:        instk( curid, p->stype, p->dimoff, p->sizoff, inoff );
        !           879: 
        !           880:        }
        !           881: 
        !           882: instk( id, t, d, s, off ) OFFSZ off; TWORD t; {
        !           883:        /* make a new entry on the parameter stack to initialize id */
        !           884: 
        !           885:        register struct symtab *p;
        !           886: 
        !           887:        for(;;){
        !           888: # ifndef BUG1
        !           889:                if( idebug ) printf( "instk((%d, %o,%d,%d, %d)\n", id, t, d, s, off );
        !           890: # endif
        !           891: 
        !           892:                /* save information on the stack */
        !           893: 
        !           894:                if( !pstk ) pstk = instack;
        !           895:                else ++pstk;
        !           896: 
        !           897:                pstk->in_fl = 0;        /* { flag */
        !           898:                pstk->in_id =  id ;
        !           899:                pstk->in_t =  t ;
        !           900:                pstk->in_d =  d ;
        !           901:                pstk->in_s =  s ;
        !           902:                pstk->in_n = 0;  /* number seen */
        !           903:                pstk->in_x =  t==STRTY ?dimtab[s+1] : 0 ;
        !           904:                pstk->in_off =  off;   /* offset at the beginning of this element */
        !           905:                /* if t is an array, DECREF(t) can't be a field */
        !           906:                /* INS_sz has size of array elements, and -size for fields */
        !           907:                if( ISARY(t) ){
        !           908:                        pstk->in_sz = tsize( DECREF(t), d+1, s );
        !           909:                        }
        !           910:                else if( stab[id].sclass & FIELD ){
        !           911:                        pstk->in_sz = - ( stab[id].sclass & FLDSIZ );
        !           912:                        }
        !           913:                else {
        !           914:                        pstk->in_sz = 0;
        !           915:                        }
        !           916: 
        !           917:                if( (iclass==AUTO || iclass == REGISTER ) &&
        !           918:                        (ISARY(t) || t==STRTY) ) uerror( "no automatic aggregate initialization" );
        !           919: 
        !           920:                /* now, if this is not a scalar, put on another element */
        !           921: 
        !           922:                if( ISARY(t) ){
        !           923:                        t = DECREF(t);
        !           924:                        ++d;
        !           925:                        continue;
        !           926:                        }
        !           927:                else if( t == STRTY ){
        !           928:                        if( dimtab[pstk->in_s] == 0 ){
        !           929:                                uerror( "can't initialize undefined structure" );
        !           930:                                iclass = -1;
        !           931:                                return;
        !           932:                                }
        !           933:                        id = dimtab[pstk->in_x];
        !           934:                        p = &stab[id];
        !           935:                        if( p->sclass != MOS && !(p->sclass&FIELD) ) cerror( "insane structure member list" );
        !           936:                        t = p->stype;
        !           937:                        d = p->dimoff;
        !           938:                        s = p->sizoff;
        !           939:                        off += p->offset;
        !           940:                        continue;
        !           941:                        }
        !           942:                else return;
        !           943:                }
        !           944:        }
        !           945: 
        !           946: NODE *
        !           947: getstr(){ /* decide if the string is external or an initializer, and get the contents accordingly */
        !           948: 
        !           949:        register l, temp;
        !           950:        register NODE *p;
        !           951: 
        !           952:        if( (iclass==EXTDEF||iclass==STATIC) && (pstk->in_t == CHAR || pstk->in_t == UCHAR) &&
        !           953:                        pstk!=instack && ISARY( pstk[-1].in_t ) ){
        !           954:                /* treat "abc" as { 'a', 'b', 'c', 0 } */
        !           955:                strflg = 1;
        !           956:                ilbrace();  /* simulate { */
        !           957:                inforce( pstk->in_off );
        !           958:                /* if the array is inflexible (not top level), pass in the size and
        !           959:                        be prepared to throw away unwanted initializers */
        !           960:                lxstr((pstk-1)!=instack?dimtab[(pstk-1)->in_d]:0);  /* get the contents */
        !           961:                irbrace();  /* simulate } */
        !           962:                return( NIL );
        !           963:                }
        !           964:        else { /* make a label, and get the contents and stash them away */
        !           965:                if( iclass != SNULL ){ /* initializing */
        !           966:                        /* fill out previous word, to permit pointer */
        !           967:                        vfdalign( ALPOINT );
        !           968:                        }
        !           969:                temp = locctr( blevel==0?ISTRNG:STRNG ); /* set up location counter */
        !           970:                deflab( l = getlab() );
        !           971:                strflg = 0;
        !           972:                lxstr(0); /* get the contents */
        !           973:                (void) locctr( blevel==0?ilocctr:temp );
        !           974:                p = buildtree( STRING, NIL, NIL );
        !           975:                p->tn.rval = -l;
        !           976:                return(p);
        !           977:                }
        !           978:        }
        !           979: 
        !           980: putbyte( v ){ /* simulate byte v appearing in a list of integer values */
        !           981:        register NODE *p;
        !           982:        p = bcon(v);
        !           983:        incode( p, SZCHAR );
        !           984:        tfree( p );
        !           985:        gotscal();
        !           986:        }
        !           987: 
        !           988: endinit(){
        !           989:        register TWORD t;
        !           990:        register d, s, n, d1;
        !           991: 
        !           992: # ifndef BUG1
        !           993:        if( idebug ) printf( "endinit(), inoff = %d\n", inoff );
        !           994: # endif
        !           995: 
        !           996:        switch( iclass ){
        !           997: 
        !           998:        case EXTERN:
        !           999:        case AUTO:
        !          1000:        case REGISTER:
        !          1001:        case -1:
        !          1002:                return;
        !          1003:                }
        !          1004: 
        !          1005:        pstk = instack;
        !          1006: 
        !          1007:        t = pstk->in_t;
        !          1008:        d = pstk->in_d;
        !          1009:        s = pstk->in_s;
        !          1010:        n = pstk->in_n;
        !          1011: 
        !          1012:        if( ISARY(t) ){
        !          1013:                d1 = dimtab[d];
        !          1014: 
        !          1015:                vfdalign( pstk->in_sz );  /* fill out part of the last element, if needed */
        !          1016:                n = inoff/pstk->in_sz;  /* real number of initializers */
        !          1017:                if( d1 >= n ){
        !          1018:                        /* once again, t is an array, so no fields */
        !          1019:                        inforce( tsize( t, d, s ) );
        !          1020:                        n = d1;
        !          1021:                        }
        !          1022:                if( d1!=0 && d1!=n ) uerror( "too many initializers");
        !          1023:                if( n==0 ) werror( "empty array declaration");
        !          1024:                dimtab[d] = n;
        !          1025:                if( d1==0 ) FIXDEF(&stab[pstk->in_id]);
        !          1026:                }
        !          1027: 
        !          1028:        else if( t == STRTY || t == UNIONTY ){
        !          1029:                /* clearly not fields either */
        !          1030:                inforce( tsize( t, d, s ) );
        !          1031:                }
        !          1032:        else if( n > 1 ) uerror( "bad scalar initialization");
        !          1033:        /* this will never be called with a field element... */
        !          1034:        else inforce( tsize(t,d,s) );
        !          1035: 
        !          1036:        paramno = 0;
        !          1037:        vfdalign( AL_INIT );
        !          1038:        inoff = 0;
        !          1039:        iclass = SNULL;
        !          1040: 
        !          1041:        }
        !          1042: 
        !          1043: fixinit(){
        !          1044:        /* called from the grammar if we must punt during initialization */
        !          1045:        /* stolen from endinit() */
        !          1046:        pstk = instack;
        !          1047:        paramno = 0;
        !          1048:        vfdalign( AL_INIT );
        !          1049:        inoff = 0;
        !          1050:        iclass = SNULL;
        !          1051:        }
        !          1052: 
        !          1053: doinit( p ) register NODE *p; {
        !          1054: 
        !          1055:        /* take care of generating a value for the initializer p */
        !          1056:        /* inoff has the current offset (last bit written)
        !          1057:                in the current word being generated */
        !          1058: 
        !          1059:        register sz, d, s;
        !          1060:        register TWORD t;
        !          1061:        int o;
        !          1062: 
        !          1063:        /* note: size of an individual initializer is assumed to fit into an int */
        !          1064: 
        !          1065:        if( iclass < 0 ) goto leave;
        !          1066:        if( iclass == EXTERN || iclass == UNAME ){
        !          1067:                uerror( "cannot initialize extern or union" );
        !          1068:                iclass = -1;
        !          1069:                goto leave;
        !          1070:                }
        !          1071: 
        !          1072:        if( iclass == AUTO || iclass == REGISTER ){
        !          1073:                /* do the initialization and get out, without regard 
        !          1074:                    for filing out the variable with zeros, etc. */
        !          1075:                bccode();
        !          1076:                idname = pstk->in_id;
        !          1077:                p = buildtree( ASSIGN, buildtree( NAME, NIL, NIL ), p );
        !          1078:                ecomp(p);
        !          1079:                return;
        !          1080:                }
        !          1081: 
        !          1082:        if( p == NIL ) return;  /* for throwing away strings that have been turned into lists */
        !          1083: 
        !          1084:        if( ifull ){
        !          1085:                uerror( "too many initializers" );
        !          1086:                iclass = -1;
        !          1087:                goto leave;
        !          1088:                }
        !          1089:        if( ibseen ){
        !          1090:                uerror( "} expected");
        !          1091:                goto leave;
        !          1092:                }
        !          1093: 
        !          1094: # ifndef BUG1
        !          1095:        if( idebug > 1 ) printf( "doinit(%o)\n", p );
        !          1096: # endif
        !          1097: 
        !          1098:        t = pstk->in_t;  /* type required */
        !          1099:        d = pstk->in_d;
        !          1100:        s = pstk->in_s;
        !          1101:        if( pstk->in_sz < 0 ){  /* bit field */
        !          1102:                sz = -pstk->in_sz;
        !          1103:                }
        !          1104:        else {
        !          1105:                sz = tsize( t, d, s );
        !          1106:                }
        !          1107: 
        !          1108:        inforce( pstk->in_off );
        !          1109: 
        !          1110:        p = buildtree( ASSIGN, block( NAME, NIL,NIL, t, d, s ), p );
        !          1111: #ifdef LINT
        !          1112:        /* force lint to treat this like an assignment */
        !          1113:        ecode(p);
        !          1114: #endif
        !          1115:        p->in.left->in.op = FREE;
        !          1116:        p->in.left = p->in.right;
        !          1117:        p->in.right = NIL;
        !          1118:        p->in.left = optim( p->in.left );
        !          1119:        o = p->in.left->in.op;
        !          1120:        if( o == UNARY AND ){
        !          1121:                o = p->in.left->in.op = FREE;
        !          1122:                p->in.left = p->in.left->in.left;
        !          1123:                }
        !          1124:        p->in.op = INIT;
        !          1125: 
        !          1126:        if( sz < SZINT ){ /* special case: bit fields, etc. */
        !          1127:                if( o != ICON || p->in.left->tn.rval != NONAME )
        !          1128:                        uerror( "illegal initialization" );
        !          1129:                else incode( p->in.left, sz );
        !          1130:                }
        !          1131:        else if( o == FCON ){
        !          1132:                fincode( p->in.left->fpn.fval, sz );
        !          1133:                }
        !          1134:        else if( o == DCON ){
        !          1135:                fincode( p->in.left->dpn.dval, sz );
        !          1136:                }
        !          1137:        else {
        !          1138:                p = optim(p);
        !          1139:                if( p->in.left->in.op != ICON ) uerror( "illegal initialization" );
        !          1140:                else cinit( p, sz );
        !          1141:                }
        !          1142: 
        !          1143:        gotscal();
        !          1144: 
        !          1145:        leave:
        !          1146:        tfree(p);
        !          1147:        }
        !          1148: 
        !          1149: gotscal(){
        !          1150:        register t, ix;
        !          1151:        register n, id;
        !          1152:        struct symtab *p;
        !          1153:        OFFSZ temp;
        !          1154: 
        !          1155:        for( ; pstk > instack; ) {
        !          1156: 
        !          1157:                if( pstk->in_fl ) ++ibseen;
        !          1158: 
        !          1159:                --pstk;
        !          1160:                
        !          1161:                t = pstk->in_t;
        !          1162: 
        !          1163:                if( t == STRTY ){
        !          1164:                        ix = ++pstk->in_x;
        !          1165:                        if( (id=dimtab[ix]) < 0 ) continue;
        !          1166: 
        !          1167:                        /* otherwise, put next element on the stack */
        !          1168: 
        !          1169:                        p = &stab[id];
        !          1170:                        instk( id, p->stype, p->dimoff, p->sizoff, p->offset+pstk->in_off );
        !          1171:                        return;
        !          1172:                        }
        !          1173:                else if( ISARY(t) ){
        !          1174:                        n = ++pstk->in_n;
        !          1175:                        if( n >= dimtab[pstk->in_d] && pstk > instack ) continue;
        !          1176: 
        !          1177:                        /* put the new element onto the stack */
        !          1178: 
        !          1179:                        temp = pstk->in_sz;
        !          1180:                        instk( pstk->in_id, (TWORD)DECREF(pstk->in_t), pstk->in_d+1, pstk->in_s,
        !          1181:                                pstk->in_off+n*temp );
        !          1182:                        return;
        !          1183:                        }
        !          1184: 
        !          1185:                }
        !          1186:        ifull = 1;
        !          1187:        }
        !          1188: 
        !          1189: ilbrace(){ /* process an initializer's left brace */
        !          1190:        register t;
        !          1191:        struct instk *temp;
        !          1192: 
        !          1193:        temp = pstk;
        !          1194: 
        !          1195:        for( ; pstk > instack; --pstk ){
        !          1196: 
        !          1197:                t = pstk->in_t;
        !          1198:                if( t != STRTY && !ISARY(t) ) continue; /* not an aggregate */
        !          1199:                if( pstk->in_fl ){ /* already associated with a { */
        !          1200:                        if( pstk->in_n ) uerror( "illegal {");
        !          1201:                        continue;
        !          1202:                        }
        !          1203: 
        !          1204:                /* we have one ... */
        !          1205:                pstk->in_fl = 1;
        !          1206:                break;
        !          1207:                }
        !          1208: 
        !          1209:        /* cannot find one */
        !          1210:        /* ignore such right braces */
        !          1211: 
        !          1212:        pstk = temp;
        !          1213:        }
        !          1214: 
        !          1215: irbrace(){
        !          1216:        /* called when a '}' is seen */
        !          1217: 
        !          1218: # ifndef BUG1
        !          1219:        if( idebug ) printf( "irbrace(): paramno = %d on entry\n", paramno );
        !          1220: # endif
        !          1221: 
        !          1222:        if( ibseen ) {
        !          1223:                --ibseen;
        !          1224:                return;
        !          1225:                }
        !          1226: 
        !          1227:        for( ; pstk > instack; --pstk ){
        !          1228:                if( !pstk->in_fl ) continue;
        !          1229: 
        !          1230:                /* we have one now */
        !          1231: 
        !          1232:                pstk->in_fl = 0;  /* cancel { */
        !          1233:                gotscal();  /* take it away... */
        !          1234:                return;
        !          1235:                }
        !          1236: 
        !          1237:        /* these right braces match ignored left braces: throw out */
        !          1238:        ifull = 1;
        !          1239: 
        !          1240:        }
        !          1241: 
        !          1242: upoff( size, alignment, poff ) register alignment, *poff; {
        !          1243:        /* update the offset pointed to by poff; return the
        !          1244:        /* offset of a value of size `size', alignment `alignment',
        !          1245:        /* given that off is increasing */
        !          1246: 
        !          1247:        register off;
        !          1248: 
        !          1249:        off = *poff;
        !          1250:        SETOFF( off, alignment );
        !          1251:        if( (offsz-off) <  size ){
        !          1252:                if( instruct!=INSTRUCT )cerror("too many local variables");
        !          1253:                else cerror("Structure too large");
        !          1254:                }
        !          1255:        *poff = off+size;
        !          1256:        return( off );
        !          1257:        }
        !          1258: 
        !          1259: oalloc( p, poff ) register struct symtab *p; register *poff; {
        !          1260:        /* allocate p with offset *poff, and update *poff */
        !          1261:        register al, off, tsz;
        !          1262:        int noff;
        !          1263: 
        !          1264:        al = talign( p->stype, p->sizoff );
        !          1265:        noff = off = *poff;
        !          1266:        tsz = tsize( p->stype, p->dimoff, p->sizoff );
        !          1267: #ifdef BACKAUTO
        !          1268:        if( p->sclass == AUTO ){
        !          1269:                if( (offsz-off) < tsz ) cerror("too many local variables");
        !          1270:                noff = off + tsz;
        !          1271:                SETOFF( noff, al );
        !          1272:                off = -noff;
        !          1273:                }
        !          1274:        else
        !          1275: #endif
        !          1276:                if( p->sclass == PARAM && ( tsz < SZINT ) ){
        !          1277:                        off = upoff( SZINT, ALINT, &noff );
        !          1278: # ifndef RTOLBYTES
        !          1279:                        off = noff - tsz;
        !          1280: #endif
        !          1281:                        }
        !          1282:                else
        !          1283:                {
        !          1284:                off = upoff( tsz, al, &noff );
        !          1285:                }
        !          1286: 
        !          1287:        if( p->sclass != REGISTER ){ /* in case we are allocating stack space for register arguments */
        !          1288:                if( p->offset == NOOFFSET ) p->offset = off;
        !          1289:                else if( off != p->offset ) return(1);
        !          1290:                }
        !          1291: 
        !          1292:        *poff = noff;
        !          1293:        return(0);
        !          1294:        }
        !          1295: 
        !          1296: falloc( p, w, new, pty )  register struct symtab *p; NODE *pty; {
        !          1297:        /* allocate a field of width w */
        !          1298:        /* new is 0 if new entry, 1 if redefinition, -1 if alignment */
        !          1299: 
        !          1300:        register al,sz,type;
        !          1301: 
        !          1302:        type = (new<0)? pty->in.type : p->stype;
        !          1303: 
        !          1304:        /* this must be fixed to use the current type in alignments */
        !          1305:        switch( new<0?pty->in.type:p->stype ){
        !          1306: 
        !          1307:        case ENUMTY:
        !          1308:                {
        !          1309:                        int s;
        !          1310:                        s = new<0 ? pty->fn.csiz : p->sizoff;
        !          1311:                        al = dimtab[s+2];
        !          1312:                        sz = dimtab[s];
        !          1313:                        break;
        !          1314:                        }
        !          1315: 
        !          1316:        case CHAR:
        !          1317:        case UCHAR:
        !          1318:                al = ALCHAR;
        !          1319:                sz = SZCHAR;
        !          1320:                break;
        !          1321: 
        !          1322:        case SHORT:
        !          1323:        case USHORT:
        !          1324:                al = ALSHORT;
        !          1325:                sz = SZSHORT;
        !          1326:                break;
        !          1327: 
        !          1328:        case INT:
        !          1329:        case UNSIGNED:
        !          1330:                al = ALINT;
        !          1331:                sz = SZINT;
        !          1332:                break;
        !          1333: #ifdef LONGFIELDS
        !          1334: 
        !          1335:        case LONG:
        !          1336:        case ULONG:
        !          1337:                al = ALLONG;
        !          1338:                sz = SZLONG;
        !          1339:                break;
        !          1340: #endif
        !          1341: 
        !          1342:        default:
        !          1343:                if( new < 0 ) {
        !          1344:                        uerror( "illegal field type" );
        !          1345:                        al = ALINT;
        !          1346:                        }
        !          1347:                else {
        !          1348:                        al = fldal( p->stype );
        !          1349:                        sz =SZINT;
        !          1350:                        }
        !          1351:                }
        !          1352: 
        !          1353:        if( w > sz ) {
        !          1354:                uerror( "field too big");
        !          1355:                w = sz;
        !          1356:                }
        !          1357: 
        !          1358:        if( w == 0 ){ /* align only */
        !          1359:                SETOFF( strucoff, al );
        !          1360:                if( new >= 0 ) uerror( "zero size field");
        !          1361:                return(0);
        !          1362:                }
        !          1363: 
        !          1364:        if( strucoff%al + w > sz ) SETOFF( strucoff, al );
        !          1365:        if( new < 0 ) {
        !          1366:                if( (offsz-strucoff) < w )
        !          1367:                        cerror("structure too large");
        !          1368:                strucoff += w;  /* we know it will fit */
        !          1369:                return(0);
        !          1370:                }
        !          1371: 
        !          1372:        /* establish the field */
        !          1373: 
        !          1374:        if( new == 1 ) { /* previous definition */
        !          1375:                if( p->offset != strucoff || p->sclass != (FIELD|w) ) return(1);
        !          1376:                }
        !          1377:        p->offset = strucoff;
        !          1378:        if( (offsz-strucoff) < w ) cerror("structure too large");
        !          1379:        strucoff += w;
        !          1380:        p->stype = type;
        !          1381:        fldty( p );
        !          1382:        return(0);
        !          1383:        }
        !          1384: 
        !          1385: nidcl( p ) NODE *p; { /* handle unitialized declarations */
        !          1386:        /* assumed to be not functions */
        !          1387:        register class;
        !          1388:        register commflag;  /* flag for labelled common declarations */
        !          1389: 
        !          1390:        commflag = 0;
        !          1391: 
        !          1392:        /* compute class */
        !          1393:        if( (class=curclass) == SNULL ){
        !          1394:                if( blevel > 1 ) class = AUTO;
        !          1395:                else if( blevel != 0 || instruct ) cerror( "nidcl error" );
        !          1396:                else { /* blevel = 0 */
        !          1397:                        class = noinit();
        !          1398:                        if( class == EXTERN ) commflag = 1;
        !          1399:                        }
        !          1400:                }
        !          1401: #ifdef LCOMM
        !          1402:        /* hack so stab will come out as LCSYM rather than STSYM */
        !          1403:        if (class == STATIC) {
        !          1404:                extern int stabLCSYM;
        !          1405:                stabLCSYM = 1;
        !          1406:        }
        !          1407: #endif
        !          1408: 
        !          1409:        defid( p, class );
        !          1410: 
        !          1411:        /* if an array is not initialized, no empty dimension */
        !          1412:        if( class!=EXTERN && class!=TYPEDEF &&
        !          1413:            ISARY(p->in.type) && dimtab[p->fn.cdim]==0 )
        !          1414:                uerror("null storage definition");
        !          1415: 
        !          1416: #ifndef LCOMM
        !          1417:        if( class==EXTDEF || class==STATIC )
        !          1418: #else
        !          1419:        if (class==STATIC) {
        !          1420:                register struct symtab *s = &stab[p->tn.rval];
        !          1421:                extern int stabLCSYM;
        !          1422:                int sz = tsize(s->stype, s->dimoff, s->sizoff)/SZCHAR;
        !          1423:                
        !          1424:                stabLCSYM = 0;
        !          1425:                if (sz % sizeof (int))
        !          1426:                        sz += sizeof (int) - (sz % sizeof (int));
        !          1427:                if (s->slevel > 1)
        !          1428:                        printf("        .lcomm  L%d,%d\n", s->offset, sz);
        !          1429:                else
        !          1430:                        printf("        .lcomm  %s,%d\n", exname(s->sname), sz);
        !          1431:        }else if (class == EXTDEF)
        !          1432: #endif
        !          1433:                {
        !          1434:                /* simulate initialization by 0 */
        !          1435:                beginit(p->tn.rval);
        !          1436:                endinit();
        !          1437:                }
        !          1438:        if( commflag ) commdec( p->tn.rval );
        !          1439:        }
        !          1440: 
        !          1441: TWORD
        !          1442: types( t1, t2, t3 ) TWORD t1, t2, t3; {
        !          1443:        /* return a basic type from basic types t1, t2, and t3 */
        !          1444: 
        !          1445:        TWORD t[3], noun, adj, unsg;
        !          1446:        register i;
        !          1447: 
        !          1448:        t[0] = t1;
        !          1449:        t[1] = t2;
        !          1450:        t[2] = t3;
        !          1451: 
        !          1452:        unsg = INT;  /* INT or UNSIGNED */
        !          1453:        noun = UNDEF;  /* INT, CHAR, or FLOAT */
        !          1454:        adj = INT;  /* INT, LONG, or SHORT */
        !          1455: 
        !          1456:        for( i=0; i<3; ++i ){
        !          1457:                switch( t[i] ){
        !          1458: 
        !          1459:                default:
        !          1460:                bad:
        !          1461:                        uerror( "illegal type combination" );
        !          1462:                        return( INT );
        !          1463: 
        !          1464:                case UNDEF:
        !          1465:                        continue;
        !          1466: 
        !          1467:                case UNSIGNED:
        !          1468:                        if( unsg != INT ) goto bad;
        !          1469:                        unsg = UNSIGNED;
        !          1470:                        continue;
        !          1471: 
        !          1472:                case LONG:
        !          1473:                case SHORT:
        !          1474:                        if( adj != INT ) goto bad;
        !          1475:                        adj = t[i];
        !          1476:                        continue;
        !          1477: 
        !          1478:                case INT:
        !          1479:                case CHAR:
        !          1480:                case FLOAT:
        !          1481:                        if( noun != UNDEF ) goto bad;
        !          1482:                        noun = t[i];
        !          1483:                        continue;
        !          1484:                        }
        !          1485:                }
        !          1486: 
        !          1487:        /* now, construct final type */
        !          1488:        if( noun == UNDEF ) noun = INT;
        !          1489:        else if( noun == FLOAT ){
        !          1490:                if( unsg != INT || adj == SHORT ) goto bad;
        !          1491:                return( adj==LONG ? DOUBLE : FLOAT );
        !          1492:                }
        !          1493:        else if( noun == CHAR && adj != INT ) goto bad;
        !          1494: 
        !          1495:        /* now, noun is INT or CHAR */
        !          1496:        if( adj != INT ) noun = adj;
        !          1497:        if( unsg == UNSIGNED ) return( noun + (UNSIGNED-INT) );
        !          1498:        else return( noun );
        !          1499:        }
        !          1500: 
        !          1501: NODE *
        !          1502: tymerge( typ, idp ) NODE *typ, *idp; {
        !          1503:        /* merge type typ with identifier idp  */
        !          1504: 
        !          1505:        register unsigned t;
        !          1506:        register i;
        !          1507:        extern int eprint();
        !          1508: 
        !          1509:        if( typ->in.op != TYPE ) cerror( "tymerge: arg 1" );
        !          1510:        if(idp == NIL ) return( NIL );
        !          1511: 
        !          1512: # ifndef BUG1
        !          1513:        if( ddebug > 2 ) fwalk( idp, eprint, 0 );
        !          1514: # endif
        !          1515: 
        !          1516:        idp->in.type = typ->in.type;
        !          1517:        idp->fn.cdim = curdim;
        !          1518:        tyreduce( idp );
        !          1519:        idp->fn.csiz = typ->fn.csiz;
        !          1520: 
        !          1521:        for( t=typ->in.type, i=typ->fn.cdim; t&TMASK; t = DECREF(t) ){
        !          1522:                if( ISARY(t) ) dstash( dimtab[i++] );
        !          1523:                }
        !          1524: 
        !          1525:        /* now idp is a single node: fix up type */
        !          1526: 
        !          1527:        idp->in.type = ctype( idp->in.type );
        !          1528: 
        !          1529:        if( (t = BTYPE(idp->in.type)) != STRTY && t != UNIONTY && t != ENUMTY ){
        !          1530:                idp->fn.csiz = t;  /* in case ctype has rewritten things */
        !          1531:                }
        !          1532: 
        !          1533:        return( idp );
        !          1534:        }
        !          1535: 
        !          1536: tyreduce( p ) register NODE *p; {
        !          1537: 
        !          1538:        /* build a type, and stash away dimensions, from a parse tree of the declaration */
        !          1539:        /* the type is build top down, the dimensions bottom up */
        !          1540:        register o, temp;
        !          1541:        register unsigned t;
        !          1542: 
        !          1543:        o = p->in.op;
        !          1544:        p->in.op = FREE;
        !          1545: 
        !          1546:        if( o == NAME ) return;
        !          1547: 
        !          1548:        t = INCREF( p->in.type );
        !          1549:        if( o == UNARY CALL ) t += (FTN-PTR);
        !          1550:        else if( o == LB ){
        !          1551:                t += (ARY-PTR);
        !          1552:                temp = p->in.right->tn.lval;
        !          1553:                p->in.right->in.op = FREE;
        !          1554:                if( temp == 0 && p->in.left->tn.op == LB )
        !          1555:                        uerror( "null dimension" );
        !          1556:                }
        !          1557: 
        !          1558:        p->in.left->in.type = t;
        !          1559:        tyreduce( p->in.left );
        !          1560: 
        !          1561:        if( o == LB ) dstash( temp );
        !          1562: 
        !          1563:        p->tn.rval = p->in.left->tn.rval;
        !          1564:        p->in.type = p->in.left->in.type;
        !          1565: 
        !          1566:        }
        !          1567: 
        !          1568: fixtype( p, class ) register NODE *p; {
        !          1569:        register unsigned t, type;
        !          1570:        register mod1, mod2;
        !          1571:        /* fix up the types, and check for legality */
        !          1572: 
        !          1573:        if( (type = p->in.type) == UNDEF ) return;
        !          1574:        if( mod2 = (type&TMASK) ){
        !          1575:                t = DECREF(type);
        !          1576:                while( mod1=mod2, mod2 = (t&TMASK) ){
        !          1577:                        if( mod1 == ARY && mod2 == FTN ){
        !          1578:                                uerror( "array of functions is illegal" );
        !          1579:                                type = 0;
        !          1580:                                }
        !          1581:                        else if( mod1 == FTN && ( mod2 == ARY || mod2 == FTN ) ){
        !          1582:                                uerror( "function returns illegal type" );
        !          1583:                                type = 0;
        !          1584:                                }
        !          1585:                        t = DECREF(t);
        !          1586:                        }
        !          1587:                }
        !          1588: 
        !          1589:        /* detect function arguments, watching out for structure declarations */
        !          1590:        /* for example, beware of f(x) struct { int a[10]; } *x; { ... } */
        !          1591:        /* the danger is that "a" will be converted to a pointer */
        !          1592: 
        !          1593:        if( class==SNULL && blevel==1 && !(instruct&(INSTRUCT|INUNION)) )
        !          1594:                class = PARAM;
        !          1595:        if( class == PARAM || ( class==REGISTER && blevel==1 ) ){
        !          1596:                if( type == FLOAT ) type = DOUBLE;
        !          1597:                else if( ISARY(type) ){
        !          1598: #ifdef LINT
        !          1599:                        if( hflag && dimtab[p->fn.cdim]!=0 )
        !          1600:                                werror("array[%d] type changed to pointer",
        !          1601:                                        dimtab[p->fn.cdim]);
        !          1602: #endif
        !          1603:                        ++p->fn.cdim;
        !          1604:                        type += (PTR-ARY);
        !          1605:                        }
        !          1606:                else if( ISFTN(type) ){
        !          1607:                        werror( "a function is declared as an argument" );
        !          1608:                        type = INCREF(type);
        !          1609:                        }
        !          1610: 
        !          1611:                }
        !          1612: 
        !          1613:        if( instruct && ISFTN(type) ){
        !          1614:                uerror( "function illegal in structure or union" );
        !          1615:                type = INCREF(type);
        !          1616:                }
        !          1617:        p->in.type = type;
        !          1618:        }
        !          1619: 
        !          1620: uclass( class ) register class; {
        !          1621:        /* give undefined version of class */
        !          1622:        if( class == SNULL ) return( EXTERN );
        !          1623:        else if( class == STATIC ) return( USTATIC );
        !          1624:        else if( class == FORTRAN ) return( UFORTRAN );
        !          1625:        else return( class );
        !          1626:        }
        !          1627: 
        !          1628: fixclass( class, type ) TWORD type; {
        !          1629: 
        !          1630:        /* first, fix null class */
        !          1631: 
        !          1632:        if( class == SNULL ){
        !          1633:                if( instruct&INSTRUCT ) class = MOS;
        !          1634:                else if( instruct&INUNION ) class = MOU;
        !          1635:                else if( blevel == 0 ) class = EXTDEF;
        !          1636:                else if( blevel == 1 ) class = PARAM;
        !          1637:                else class = AUTO;
        !          1638: 
        !          1639:                }
        !          1640: 
        !          1641:        /* now, do general checking */
        !          1642: 
        !          1643:        if( ISFTN( type ) ){
        !          1644:                switch( class ) {
        !          1645:                default:
        !          1646:                        uerror( "function has illegal storage class" );
        !          1647:                case AUTO:
        !          1648:                        class = EXTERN;
        !          1649:                case EXTERN:
        !          1650:                case EXTDEF:
        !          1651:                case FORTRAN:
        !          1652:                case TYPEDEF:
        !          1653:                case STATIC:
        !          1654:                case UFORTRAN:
        !          1655:                case USTATIC:
        !          1656:                        ;
        !          1657:                        }
        !          1658:                }
        !          1659: 
        !          1660:        if( class&FIELD ){
        !          1661:                if( !(instruct&INSTRUCT) ) uerror( "illegal use of field" );
        !          1662:                return( class );
        !          1663:                }
        !          1664: 
        !          1665:        switch( class ){
        !          1666: 
        !          1667:        case MOU:
        !          1668:                if( !(instruct&INUNION) ) uerror( "illegal class" );
        !          1669:                return( class );
        !          1670: 
        !          1671:        case MOS:
        !          1672:                if( !(instruct&INSTRUCT) ) uerror( "illegal class" );
        !          1673:                return( class );
        !          1674: 
        !          1675:        case MOE:
        !          1676:                if( instruct & (INSTRUCT|INUNION) ) uerror( "illegal class" );
        !          1677:                return( class );
        !          1678: 
        !          1679:        case REGISTER:
        !          1680:                if( blevel == 0 ) uerror( "illegal register declaration" );
        !          1681:                else if( regvar >= MINRVAR && cisreg( type ) ) return( class );
        !          1682:                if( blevel == 1 ) return( PARAM );
        !          1683:                else return( AUTO );
        !          1684: 
        !          1685:        case AUTO:
        !          1686:        case LABEL:
        !          1687:        case ULABEL:
        !          1688:                if( blevel < 2 ) uerror( "illegal class" );
        !          1689:                return( class );
        !          1690: 
        !          1691:        case PARAM:
        !          1692:                if( blevel != 1 ) uerror( "illegal class" );
        !          1693:                return( class );
        !          1694: 
        !          1695:        case UFORTRAN:
        !          1696:        case FORTRAN:
        !          1697: # ifdef NOFORTRAN
        !          1698:                        NOFORTRAN;    /* a condition which can regulate the FORTRAN usage */
        !          1699: # endif
        !          1700:                if( !ISFTN(type) ) uerror( "fortran declaration must apply to function" );
        !          1701:                else {
        !          1702:                        type = DECREF(type);
        !          1703:                        if( ISFTN(type) || ISARY(type) || ISPTR(type) ) {
        !          1704:                                uerror( "fortran function has wrong type" );
        !          1705:                                }
        !          1706:                        }
        !          1707:        case EXTERN:
        !          1708:        case STATIC:
        !          1709:        case EXTDEF:
        !          1710:        case TYPEDEF:
        !          1711:        case USTATIC:
        !          1712:                if( blevel == 1 ){
        !          1713:                        uerror( "illegal class" );
        !          1714:                        return( PARAM );
        !          1715:                        }
        !          1716:        case STNAME:
        !          1717:        case UNAME:
        !          1718:        case ENAME:
        !          1719:                return( class );
        !          1720: 
        !          1721:        default:
        !          1722:                cerror( "illegal class: %d", class );
        !          1723:                /* NOTREACHED */
        !          1724: 
        !          1725:                }
        !          1726:        }
        !          1727: 
        !          1728: struct symtab *
        !          1729: mknonuniq(idindex) int *idindex; {/* locate a symbol table entry for */
        !          1730:        /* an occurrence of a nonunique structure member name */
        !          1731:        /* or field */
        !          1732:        register i;
        !          1733:        register struct symtab * sp;
        !          1734:        char *q;
        !          1735: 
        !          1736:        sp = & stab[ i= *idindex ]; /* position search at old entry */
        !          1737:        while( sp->stype != TNULL ){ /* locate unused entry */
        !          1738:                if( ++i >= SYMTSZ ){/* wrap around symbol table */
        !          1739:                        i = 0;
        !          1740:                        sp = stab;
        !          1741:                        }
        !          1742:                else ++sp;
        !          1743:                if( i == *idindex ) cerror("Symbol table full");
        !          1744:                }
        !          1745:        sp->sflags = SNONUNIQ | SMOS;
        !          1746:        q = stab[*idindex].sname; /* old entry name */
        !          1747: #ifdef FLEXNAMES
        !          1748:        sp->sname = stab[*idindex].sname;
        !          1749: #endif
        !          1750: # ifndef BUG1
        !          1751:        if( ddebug ){
        !          1752:                printf("\tnonunique entry for %s from %d to %d\n",
        !          1753:                        q, *idindex, i );
        !          1754:                }
        !          1755: # endif
        !          1756:        *idindex = i;
        !          1757: #ifndef FLEXNAMES
        !          1758:        {
        !          1759:                char *p = sp->sname;
        !          1760:                for( i=1; i<=NCHNAM; ++i ) /* copy name */
        !          1761:                        if( *p++ = *q /* assign */ ) ++q;
        !          1762:                }
        !          1763: #endif
        !          1764:        return ( sp );
        !          1765:        }
        !          1766: 
        !          1767: lookup( name, s) char *name; { 
        !          1768:        /* look up name: must agree with s w.r.t. STAG, SMOS and SHIDDEN */
        !          1769: 
        !          1770:        register char *p, *q;
        !          1771:        int i, ii;
        !          1772: #ifndef FLEXNAMES
        !          1773:        int j;
        !          1774: #endif
        !          1775:        register struct symtab *sp;
        !          1776: 
        !          1777:        /* compute initial hash index */
        !          1778: # ifndef BUG1
        !          1779:        if( ddebug > 2 ){
        !          1780:                printf( "lookup( %s, %d ), stwart=%d, instruct=%d\n", name, s, stwart, instruct );
        !          1781:                }
        !          1782: # endif
        !          1783: 
        !          1784:        i = 0;
        !          1785: #ifndef FLEXNAMES
        !          1786:        for( p=name, j=0; *p != '\0'; ++p ){
        !          1787:                i += *p;
        !          1788:                if( ++j >= NCHNAM ) break;
        !          1789:                }
        !          1790: #else
        !          1791:        i = (int)name;
        !          1792: #endif
        !          1793:        i = i%SYMTSZ;
        !          1794:        sp = &stab[ii=i];
        !          1795: 
        !          1796:        for(;;){ /* look for name */
        !          1797: 
        !          1798:                if( sp->stype == TNULL ){ /* empty slot */
        !          1799:                        sp->sflags = s;  /* set STAG, SMOS if needed, turn off all others */
        !          1800: #ifndef FLEXNAMES
        !          1801:                        p = sp->sname;
        !          1802:                        for( j=0; j<NCHNAM; ++j ) if( *p++ = *name ) ++name;
        !          1803: #else
        !          1804:                        sp->sname = name;
        !          1805: #endif
        !          1806:                        sp->stype = UNDEF;
        !          1807:                        sp->sclass = SNULL;
        !          1808:                        return( i );
        !          1809:                        }
        !          1810:                if( (sp->sflags & (STAG|SMOS|SHIDDEN)) != s ) goto next;
        !          1811:                p = sp->sname;
        !          1812:                q = name;
        !          1813: #ifndef FLEXNAMES
        !          1814:                for( j=0; j<NCHNAM;++j ){
        !          1815:                        if( *p++ != *q ) goto next;
        !          1816:                        if( !*q++ ) break;
        !          1817:                        }
        !          1818:                return( i );
        !          1819: #else
        !          1820:                if (p == q)
        !          1821:                        return ( i );
        !          1822: #endif
        !          1823:        next:
        !          1824:                if( ++i >= SYMTSZ ){
        !          1825:                        i = 0;
        !          1826:                        sp = stab;
        !          1827:                        }
        !          1828:                else ++sp;
        !          1829:                if( i == ii ) cerror( "symbol table full" );
        !          1830:                }
        !          1831:        }
        !          1832: 
        !          1833: #ifndef checkst
        !          1834: /* if not debugging, make checkst a macro */
        !          1835: checkst(lev){
        !          1836:        register int s, i, j;
        !          1837:        register struct symtab *p, *q;
        !          1838: 
        !          1839:        for( i=0, p=stab; i<SYMTSZ; ++i, ++p ){
        !          1840:                if( p->stype == TNULL ) continue;
        !          1841:                j = lookup( p->sname, p->sflags&(SMOS|STAG) );
        !          1842:                if( j != i ){
        !          1843:                        q = &stab[j];
        !          1844:                        if( q->stype == UNDEF ||
        !          1845:                            q->slevel <= p->slevel ){
        !          1846: #ifndef FLEXNAMES
        !          1847:                                cerror( "check error: %.8s", q->sname );
        !          1848: #else
        !          1849:                                cerror( "check error: %s", q->sname );
        !          1850: #endif
        !          1851:                                }
        !          1852:                        }
        !          1853: #ifndef FLEXNAMES
        !          1854:                else if( p->slevel > lev ) cerror( "%.8s check at level %d", p->sname, lev );
        !          1855: #else
        !          1856:                else if( p->slevel > lev ) cerror( "%s check at level %d", p->sname, lev );
        !          1857: #endif
        !          1858:                }
        !          1859:        }
        !          1860: #endif
        !          1861: 
        !          1862: struct symtab *
        !          1863: relook(p) register struct symtab *p; {  /* look up p again, and see where it lies */
        !          1864: 
        !          1865:        register struct symtab *q;
        !          1866: 
        !          1867:        /* I'm not sure that this handles towers of several hidden definitions in all cases */
        !          1868:        q = &stab[lookup( p->sname, p->sflags&(STAG|SMOS|SHIDDEN) )];
        !          1869:        /* make relook always point to either p or an empty cell */
        !          1870:        if( q->stype == UNDEF ){
        !          1871:                q->stype = TNULL;
        !          1872:                return(q);
        !          1873:                }
        !          1874:        while( q != p ){
        !          1875:                if( q->stype == TNULL ) break;
        !          1876:                if( ++q >= &stab[SYMTSZ] ) q=stab;
        !          1877:                }
        !          1878:        return(q);
        !          1879:        }
        !          1880: 
        !          1881: clearst( lev ) register int lev; {
        !          1882:        register struct symtab *p, *q;
        !          1883:        register int temp;
        !          1884:        struct symtab *clist = 0;
        !          1885: 
        !          1886:        temp = lineno;
        !          1887:        aobeg();
        !          1888: 
        !          1889:        /* step 1: remove entries */
        !          1890:        while( chaintop-1 > lev ){
        !          1891:                p = schain[--chaintop];
        !          1892:                schain[chaintop] = 0;
        !          1893:                for( ; p; p = q ){
        !          1894:                        q = p->snext;
        !          1895:                        if( p->stype == TNULL || p->slevel <= lev )
        !          1896:                                cerror( "schain botch" );
        !          1897:                        lineno = p->suse < 0 ? -p->suse : p->suse;
        !          1898:                        if( p->stype==UNDEF || ( p->sclass==ULABEL && lev<2 ) ){
        !          1899:                                lineno = temp;
        !          1900: #ifndef FLEXNAMES
        !          1901:                                uerror( "%.8s undefined", p->sname );
        !          1902: #else
        !          1903:                                uerror( "%s undefined", p->sname );
        !          1904: #endif
        !          1905:                                }
        !          1906:                        else aocode(p);
        !          1907: # ifndef BUG1
        !          1908:                        if( ddebug ){
        !          1909: #ifndef FLEXNAMES
        !          1910:                                printf( "removing %.8s", p->sname );
        !          1911: #else
        !          1912:                                printf( "removing %s", p->sname );
        !          1913: #endif
        !          1914:                                printf( " from stab[%d], flags %o level %d\n",
        !          1915:                                        p-stab, p->sflags, p->slevel);
        !          1916:                                }
        !          1917: # endif
        !          1918:                        if( p->sflags & SHIDES )unhide( p );
        !          1919:                        p->stype = TNULL;
        !          1920:                        p->snext = clist;
        !          1921:                        clist = p;
        !          1922:                        }
        !          1923:                }
        !          1924: 
        !          1925:        /* step 2: fix any mishashed entries */
        !          1926:        p = clist;
        !          1927:        while( p ){
        !          1928:                register struct symtab *next, **t, *r;
        !          1929: 
        !          1930:                q = p;
        !          1931:                next = p->snext;
        !          1932:                for(;;){
        !          1933:                        if( ++q >= &stab[SYMTSZ] )q = stab;
        !          1934:                        if( q == p || q->stype == TNULL )break;
        !          1935:                        if( (r = relook(q)) != q ) {
        !          1936:                                /* move q in schain list */
        !          1937:                                t = &schain[q->slevel];
        !          1938:                                while( *t && *t != q )
        !          1939:                                        t = &(*t)->snext;
        !          1940:                                if( *t )
        !          1941:                                        *t = r;
        !          1942:                                else
        !          1943:                                        cerror("schain botch 2");
        !          1944:                                *r = *q;
        !          1945:                                q->stype = TNULL;
        !          1946:                                }
        !          1947:                        }
        !          1948:                p = next;
        !          1949:                }
        !          1950: 
        !          1951:        lineno = temp;
        !          1952:        aoend();
        !          1953:        }
        !          1954: 
        !          1955: hide( p ) register struct symtab *p; {
        !          1956:        register struct symtab *q;
        !          1957:        for( q=p+1; ; ++q ){
        !          1958:                if( q >= &stab[SYMTSZ] ) q = stab;
        !          1959:                if( q == p ) cerror( "symbol table full" );
        !          1960:                if( q->stype == TNULL ) break;
        !          1961:                }
        !          1962:        *q = *p;
        !          1963:        p->sflags |= SHIDDEN;
        !          1964:        q->sflags = (p->sflags&(SMOS|STAG)) | SHIDES;
        !          1965: #ifndef FLEXNAMES
        !          1966:        if( hflag ) werror( "%.8s redefinition hides earlier one", p->sname );
        !          1967: #else
        !          1968:        if( hflag ) werror( "%s redefinition hides earlier one", p->sname );
        !          1969: #endif
        !          1970: # ifndef BUG1
        !          1971:        if( ddebug ) printf( "  %d hidden in %d\n", p-stab, q-stab );
        !          1972: # endif
        !          1973:        return( idname = q-stab );
        !          1974:        }
        !          1975: 
        !          1976: unhide( p ) register struct symtab *p; {
        !          1977:        register struct symtab *q;
        !          1978:        register s;
        !          1979: 
        !          1980:        s = p->sflags & (SMOS|STAG);
        !          1981:        q = p;
        !          1982: 
        !          1983:        for(;;){
        !          1984: 
        !          1985:                if( q == stab ) q = &stab[SYMTSZ-1];
        !          1986:                else --q;
        !          1987: 
        !          1988:                if( q == p ) break;
        !          1989: 
        !          1990:                if( (q->sflags&(SMOS|STAG)) == s ){
        !          1991: #ifndef FLEXNAMES
        !          1992:                        register j;
        !          1993:                        for( j =0; j<NCHNAM; ++j ) if( p->sname[j] != q->sname[j] ) break;
        !          1994:                        if( j == NCHNAM ){ /* found the name */
        !          1995: #else
        !          1996:                        if (p->sname == q->sname) {
        !          1997: #endif
        !          1998:                                q->sflags &= ~SHIDDEN;
        !          1999: # ifndef BUG1
        !          2000:                                if( ddebug ) printf( "unhide uncovered %d from %d\n", q-stab,p-stab);
        !          2001: # endif
        !          2002:                                return;
        !          2003:                                }
        !          2004:                        }
        !          2005: 
        !          2006:                }
        !          2007:        cerror( "unhide fails" );
        !          2008:        }

unix.superglobalmegacorp.com

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