Annotation of researchv10no/cmd/ccom/common/pftn.c, revision 1.1

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

unix.superglobalmegacorp.com

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