|
|
researchv10 Norman
/* @(#) pftn.c: 1.6 3/4/84 */
# include "mfile1.h"
unsigned int maxoffset;
struct instk
{
int in_sz; /* size of array element */
int in_x; /* current index for structure member in initializations */
int in_n; /* number of initializations seen */
int in_s; /* sizoff */
int in_d; /* dimoff */
TWORD in_t; /* type */
int in_id; /* stab index */
int in_fl; /* flag which says if this level is controlled by {} */
OFFSZ in_off; /* offset of the beginning of this level */
}instack[10],*pstk;
struct symtab *relook();
int ddebug = 0;
struct symtab * mknonuniq();
defid( q, class )
register NODE *q;
{
register struct symtab *p;
extern struct symtab *scopestack[];
int idp;
register TWORD type;
register TWORD stp;
register scl;
register dsym, ddef;
register slev, temp;
if( q == NIL ) return; /* an error was detected */
if( q < node || q >= &node[TREESZ] ) cerror( "defid call" );
idp = q->tn.rval;
if( idp < 0 ) cerror( "tyreduce" );
p = &stab[idp];
# ifndef NODBG
if( ddebug )
{
printf( "defid( %s (%d), ", p->sname, idp );
tprint( q->in.type );
printf( ", %s, (%d,%d) ), level %d\n", scnames(class),
q->fn.cdim, q->fn.csiz, blevel );
}
# endif
fixtype( q, class );
type = q->in.type;
class = fixclass( class, type );
stp = p->stype;
slev = p->slevel;
# ifndef NODBG
if( ddebug )
{
printf( " modified to " );
tprint( type );
printf( ", %s\n", scnames(class) );
printf( " previous def'n: " );
tprint( stp );
printf( ", %s, (%d,%d) ), level %d\n", scnames(p->sclass),
p->dimoff, p->sizoff, slev );
}
# endif
if( stp == FTN && p->sclass == SNULL )goto enter;
/* name encountered as function, not yet defined */
/* BUG here! can't incompatibly declare func. in inner block */
if( stp == UNDEF|| stp == FARG )
{
if( blevel==1 && stp!=FARG ) switch( class )
{
default:
if(!(class&FIELD))
uerror("declared argument %s is missing", p->sname);
case MOS:
case STNAME:
case MOU:
case UNAME:
case MOE:
case ENAME:
case TYPEDEF:
;
}
goto enter;
}
if( type != stp ) goto mismatch;
/* test (and possibly adjust) dimensions */
dsym = p->dimoff;
ddef = q->fn.cdim;
for( temp=type; temp&TMASK; temp = DECREF(temp) )
{
if( ISARY(temp) )
{
if( dimtab[dsym] == 0 ) dimtab[dsym] = dimtab[ddef];
else if( dimtab[ddef]!=0 && dimtab[dsym] != dimtab[ddef] )
{
goto mismatch;
}
++dsym;
++ddef;
}
}
/* check that redeclarations are to the same structure */
if( (temp==STRTY||temp==UNIONTY||temp==ENUMTY) && p->sizoff != q->fn.csiz
&& class!=STNAME && class!=UNAME && class!=ENAME )
{
goto mismatch;
}
scl = ( p->sclass );
# ifndef NODBG
if( ddebug )
{
printf( " previous class: %s\n", scnames(scl) );
}
# endif
if( class&FIELD )
{
/* redefinition */
if( !falloc( p, class&FLDSIZ, 1, NIL ) )
{
/* successful allocation */
psave( idp );
return;
}
/* blew it: resume at end of switch... */
}
else switch( class )
{
case EXTERN:
switch( scl )
{
case STATIC:
case USTATIC:
if( slev==0 ) return;
break;
case EXTDEF:
case EXTERN:
case FORTRAN:
case UFORTRAN:
return;
}
break;
case STATIC:
if( scl==USTATIC || (scl==EXTERN && blevel==0) )
{
p->sclass = STATIC;
if( ISFTN(type) ) curftn = idp;
return;
}
break;
case USTATIC:
if( scl==STATIC || scl==USTATIC ) return;
break;
case LABEL:
if( scl == ULABEL )
{
p->sclass = LABEL;
deflab( p->offset );
return;
}
break;
case TYPEDEF:
if( scl == class ) return;
break;
case UFORTRAN:
if( scl == UFORTRAN || scl == FORTRAN ) return;
break;
case FORTRAN:
if( scl == UFORTRAN )
{
p->sclass = FORTRAN;
if( ISFTN(type) ) curftn = idp;
return;
}
break;
case MOU:
case MOS:
if( scl == class )
{
if( oalloc( p, &strucoff ) ) break;
if( class == MOU ) strucoff = 0;
psave( idp );
return;
}
break;
case MOE:
if( scl == class )
{
if( p->offset!= strucoff++ ) break;
psave( idp );
}
break;
case EXTDEF:
if( scl == EXTERN )
{
p->sclass = EXTDEF;
if( ISFTN(type) ) curftn = idp;
return;
}
break;
case STNAME:
case UNAME:
case ENAME:
if( scl != class ) break;
if( dimtab[p->sizoff] == 0 )
return; /* previous entry just a mention */
break;
case ULABEL:
if( scl == LABEL || scl == ULABEL ) return;
case PARAM:
case AUTO:
case REGISTER:
; /* mismatch.. */
}
mismatch:
/* allow nonunique structure/union member names */
if( class==MOU || class==MOS || class & FIELD )
{
/* make a new entry */
register * memp;
p->sflags |= SNONUNIQ; /* old entry is nonunique */
/* determine if name has occurred in this structure/union */
for( memp = ¶mstk[paramno-1];
/* while */ *memp>=0 && stab[*memp].sclass != STNAME
&& stab[*memp].sclass != UNAME;
/* iterate */ --memp)
{
if( stab[*memp].sflags & SNONUNIQ )
{ int i;
if ( p->sname != stab[*memp].sname )
continue;
i = stab[*memp].suse;
if(i < 0)
i = -i;
uerror("redeclaration of: %s from line %d",
p->sname, i);
break;
}
}
p = mknonuniq( &idp ); /* update p and idp to new entry */
goto enter;
}
if( blevel > slev && class != EXTERN && class != FORTRAN &&
class != UFORTRAN && !( class == LABEL && slev >= 2 ) )
{
q->tn.rval = idp = hide( p );
p = &stab[idp];
goto enter;
}
uerror( "redeclaration of %s from some line %d", p->sname, p->suse);
if( class==EXTDEF && ISFTN(type) ) curftn = idp;
return;
enter: /* make a new entry */
# ifndef NODBG
if( ddebug ) printf( " new entry made\n" );
# endif
if( type == UNDEF ) uerror("void type for %s",p->sname);
p->stype = type;
p->sclass = class;
p->slevel = (class == LABEL || class == ULABEL) ? 2 : blevel;
p->offset = NOOFFSET;
p->suse = lineno;
if( class == STNAME || class == UNAME || class == ENAME )
{
p->sizoff = curdim;
dstash( 0 ); /* size */
dstash( -1 ); /* index to members of str or union */
dstash( ALSTRUCT ); /* alignment */
dstash( idp ); /* name index */
}
else
{
switch( BTYPE(type) )
{
case STRTY:
case UNIONTY:
case ENUMTY:
p->sizoff = q->fn.csiz;
break;
default:
p->sizoff = BTYPE(type);
}
}
/* copy dimensions */
p->dimoff = q->fn.cdim;
/* allocate offsets */
if( class&FIELD )
{
falloc( p, class&FLDSIZ, 0, NIL ); /* new entry */
psave( idp );
}
else switch( class )
{
case AUTO:
oalloc( p, &autooff );
break;
case STATIC:
case EXTDEF:
p->offset = getlab();
if( ISFTN(type) ) curftn = idp;
break;
case ULABEL:
case LABEL:
p->offset = getlab();
if( class == LABEL )
{
locctr( PROG );
deflab( p->offset );
}
break;
case EXTERN:
case UFORTRAN:
case FORTRAN:
p->offset = getlab();
break;
case MOU:
case MOS:
oalloc( p, &strucoff );
if( class == MOU ) strucoff = 0;
psave( idp );
break;
case MOE:
p->offset = strucoff++;
psave( idp );
break;
case REGISTER:
/* nextrvar is left set by cisreg when it returns 1 */
p->offset = nextrvar;
if( blevel == 1 ) p->sflags |= SSET;
break;
}
/* user-supplied routine to fix up new definitions */
# ifdef FIXDEF
#ifdef TMPSRET /* don't try to put out sdb stuff for .stfake! */
if (strcmp(p->sname, ".stfake"))
#endif
FIXDEF(p);
# endif
if (blevel >= MAXNEST)
{
cerror("too many nesting levels (%d>=%d)", blevel, MAXNEST);
/*NOTREACHED*/
}
p->scopelink = scopestack[slev = p->slevel];
scopestack[slev] = p;
/* change level of outerscope identifiers first appearing in
/* inner scopes
*/
switch( class )
{
case ULABEL:
case LABEL:
p->slevel = 2;
break;
case EXTERN:
case UFORTRAN:
case FORTRAN:
p->slevel = 0;
break;
}
# ifndef NODBG
if( ddebug )
printf( " dimoff, sizoff, offset: %d, %d, %d\n",
p->dimoff, p->sizoff, p->offset );
# endif
}
asave( i )
{
if( argno >= ARGSZ )
{
cerror( "too many arguments (%d>=%d)", argno, ARGSZ);
}
argstk[ argno++ ] = i;
}
psave( i )
{
if( paramno >= PARAMSZ )
{
cerror( "parameter stack overflow (%d>=%d)", paramno, PARAMSZ);
}
paramstk[ paramno++ ] = i;
}
/* maximium size of outgoing arguments */
int maxarg;
ftnend()
{
/* end of function */
if( retlab != NOLAB )
{
/* inside a real function */
efcode();
# ifndef TWOPASS
p2bend();
# endif
}
checkst(0);
retstat = 0;
tcheck();
curclass = SNULL;
brklab = contlab = retlab = NOLAB;
flostat = 0;
strftn = 0;
argno = 0;
if( nerrors == 0 )
{
if( psavbc != & asavbc[0] ) cerror("bcsave error");
if( paramno != 0 ) cerror("parameter reset error");
if( swx != 0 ) cerror( "switch error");
}
psavbc = &asavbc[0];
paramno = 0;
autooff = AUTOINIT;
maxarg = 0;
reached = 1;
swx = 0;
swp = swtab;
locctr(DATA);
}
dclargs()
{
register i, j;
register struct symtab *p;
register NODE *q;
register TWORD temp;
extern TWORD simpstr();
argoff = ARGINIT;
# ifndef NODBG
if( ddebug > 2) printf("dclargs()\n");
# endif
for( i=0; i<argno; ++i )
{
if( (j = argstk[i]) < 0 ) continue;
p = &stab[j];
# ifndef NODBG
if( ddebug > 2 )
{
printf("\t%s (%d) ",p->sname, j);
tprint(p->stype);
printf("\n");
}
# endif
if( p->stype == FARG )
{
q = block(FREE,NIL,NIL,INT,0,INT);
q->tn.rval = j;
defid( q, PARAM );
}
if( p->sclass == REGISTER )
{
/* must set aside space, fill argsoff */
int tmp = p->offset;
p->offset = NOOFFSET;
p->sclass = PARAM;
oalloc( p, &argoff );
argsoff[i] = p->offset;
argty[i] = p->stype;
p->sclass = REGISTER;
p->offset = tmp;
}
else
{
oalloc( p, &argoff ); /* always set aside space */
argsoff[i] = p->offset;
argty[i] = p->stype;
}
}
autooff = AUTOINIT;
# ifdef CENDARG
CENDARG();
# endif
locctr(PROG);
defalign(ALINT);
++ftnno;
p = &stab[curftn];
if( p->slevel>1 && p->sclass == STATIC ) deflab( p->offset );
else defnam( p );
temp = p->stype;
temp = DECREF( temp );
/* optimization: functions returning short structs */
/* strftn != 0 if function returns structure */
strftn = (temp==STRTY) || (temp==UNIONTY);
if( strftn && simpstr( p->dimoff, p->sizoff ) != STRTY )
strftn = 0;
bfcode(argstk, argno, p->sname);
# ifndef NOREGARGS
regargs();
# endif
/* code for the case where returning a structure uses a static */
if( strftn )
{
# ifdef STATSRET
/* scalars ok */
/* define something the proper size */
i = tsize( temp, p->dimoff, p->sizoff);
# ifdef SRETNAME
SRETNAME(i);
# else
locctr( DATA );
defalign( talign( temp, p->sizoff ) );
deflab ( strftn = getlab() );
zecode( (i+SZINT-1)/SZINT ); /* set aside integer zeros */
locctr( PROG );
# endif
# endif
# ifdef TMPSRET
/* code to stash auxiliary reg (return addr for struct) into auto */
idname = lookup(hash(".stfake"), 0);
q = bdty(NAME, NIL, idname);
q->tn.type = PTR|STRTY;
++blevel; defid(q, AUTO); --blevel; /* sleazy, aren't we? */
ecomp(buildtree(ASSIGN, buildtree(NAME, NIL, NIL),
block(REG, NIL, (NODE *)AUXREG, PTR|STRTY,
q->fn.cdim, q->fn.csiz)));
q->tn.op = FREE;
# endif
}
}
# ifndef NOREGARGS
regargs()
{
register i;
register NODE *p, *q;
register struct symtab *s;
/* output the copy assignements for register arguments */
for( i=0; i<argno; ++i )
{
s = &stab[argstk[i]];
if( s->sclass == REGISTER )
{
int temp;
idname = argstk[i];
p = buildtree( NAME, NIL, NIL );
temp = s->offset;
s->offset = argsoff[i];
s->sclass = PARAM;
q = buildtree( NAME, NIL, NIL );
p = buildtree( ASSIGN, p, q );
ecomp( p );
s->offset = temp;
s->sclass = REGISTER;
}
}
}
# endif
NODE *
rstruct( idn, soru )
register idn,soru;
{
/* reference to a struct or union, with no definition */
register struct symtab *p;
register NODE *q;
p = &stab[idn];
switch( p->stype )
{
case UNDEF:
def:
q = block( FREE, NIL, NIL, 0, 0, 0 );
q->tn.rval = idn;
q->in.type = (soru&INSTRUCT) ? STRTY : ( (soru&INUNION) ? UNIONTY : ENUMTY );
defid( q, (soru&INSTRUCT) ? STNAME : ( (soru&INUNION) ? UNAME : ENAME ) );
break;
case STRTY:
if( soru & INSTRUCT ) break;
goto def;
case UNIONTY:
if( soru & INUNION ) break;
goto def;
case ENUMTY:
if( !(soru&(INUNION|INSTRUCT)) ) break;
goto def;
}
stwart = instruct;
return( mkty( p->stype, 0, p->sizoff ) );
}
moedef( idn )
register idn;
{
register NODE *q;
q = block( FREE, NIL, NIL, MOETY, 0, 0 );
q->tn.rval = idn;
if( idn>=0 ) defid( q, MOE );
}
extern char *hash(), *index(), *rindex();
static int fakeprev, fakenum;
static char fakestub[32], fakename[64];
bstruct( idn, soru )
register idn,soru;
{
/* begining of structure or union declaration */
register NODE *q;
if (idn < 0) {
char *p;
if ((p = rindex(ftitle, '/')) == NULL)
p = ftitle;
strcpy(fakestub, ++p);
if (p = index(fakestub, '"'))
*p = '\0';
sprintf(fakename, "%s$%d.%d", fakestub, lineno, ++fakenum);
idn = lookup(hash(fakename), STAG);
}
psave( instruct );
psave( curclass );
psave( strucoff );
strucoff = 0;
instruct = soru;
q = block( FREE, NIL, NIL, 0, 0, 0 );
q->tn.rval = idn;
if( instruct==INSTRUCT )
{
curclass = MOS;
q->in.type = STRTY;
defid( q, STNAME );
}
else if( instruct == INUNION )
{
curclass = MOU;
q->in.type = UNIONTY;
defid( q, UNAME );
}
else
{
/* enum */
curclass = MOE;
q->in.type = ENUMTY;
defid( q, ENAME );
}
psave( idn = q->tn.rval );
/* the "real" definition is where the members are seen */
if( idn >= 0 ) stab[idn].suse = lineno;
return( paramno-4 );
}
# ifndef ENUMSIZE
# define ENUMSIZE(h,l) INT
# endif
NODE *
dclstruct( oparam )
register oparam;
{
register struct symtab *p;
register i, al, sa, j, sz, szindex;
register TWORD temp;
register high, low;
/* paramstack contains:
** paramstack[ oparam ] = previous instruct
** paramstack[ oparam+1 ] = previous class
** paramstk[ oparam+2 ] = previous strucoff
** paramstk[ oparam+3 ] = structure name
**
** paramstk[ oparam+4, ... ] = member stab indices
**
*/
if( (i=paramstk[oparam+3]) < 0 )
{
szindex = curdim;
dstash( 0 ); /* size */
dstash( -1 ); /* index to member names */
dstash( ALSTRUCT ); /* alignment */
dstash( -lineno ); /* name of structure */
}
else
{
szindex = stab[i].sizoff;
}
# ifndef NODBG
if( ddebug )
{
printf( "dclstruct( %szindex = %d\n",
(i>=0)? stab[i].sname : "??", szindex );
}
# endif
temp = (instruct&INSTRUCT)?STRTY:((instruct&INUNION)?UNIONTY:ENUMTY);
stwart = instruct = paramstk[ oparam ];
curclass = paramstk[ oparam+1 ];
dimtab[ szindex+1 ] = curdim;
al = ALSTRUCT;
high = low = 0;
for( i = oparam+4; i< paramno; ++i )
{
dstash( j=paramstk[i] );
if( j<0 || j>= SYMTSZ ) cerror( "gummy structure member" );
p = &stab[j];
if( temp == ENUMTY )
{
if( p->offset < low ) low = p->offset;
if( p->offset > high ) high = p->offset;
p->sizoff = szindex;
continue;
}
sa = talign( p->stype, p->sizoff );
if( p->sclass & FIELD )
{
sz = p->sclass&FLDSIZ;
}
else
{
sz = tsize( p->stype, p->dimoff, p->sizoff );
}
if( sz == 0 )
{
werror( "structure member has size 0: %s", p->sname );
}
if( sz > strucoff ) strucoff = sz; /* for use with unions */
SETOFF( al, sa );
/* al, the alignment, to the LCM of the members' alignments */
}
dstash( -1 ); /* endmarker */
SETOFF( strucoff, al );
if( temp == ENUMTY )
{
register TWORD ty;
ty = ENUMSIZE(high,low);
strucoff = tsize( ty, 0, (int)ty );
dimtab[ szindex+2 ] = al = talign( ty, (int)ty );
}
if( strucoff == 0 ) uerror( "zero sized structure" );
dimtab[ szindex ] = strucoff;
dimtab[ szindex+2 ] = al;
dimtab[ szindex+3 ] = paramstk[ oparam+3 ]; /* name index */
#ifdef FIXSTRUCT
FIXSTRUCT( szindex, oparam ); /* local hook for debugging */
#endif
# ifndef NODBG
if( ddebug>1 )
{
printf( "\tdimtab[%d,%d,%d,%d] = %d,%d,%d,%d\n",
szindex,szindex+1,szindex+2,szindex+3,
dimtab[szindex],dimtab[szindex+1],dimtab[szindex+2],
dimtab[szindex+3] );
for( i = dimtab[szindex+1]; dimtab[i] >= 0; ++i )
{
printf( "\tmember %s(%d)\n", stab[dimtab[i]].sname, dimtab[i] );
}
}
# endif
strucoff = paramstk[ oparam+2 ];
paramno = oparam;
return( mkty( temp, 0, szindex ) );
}
/* VARARGS */
yyerror( s, a, b, c, d, e, f, g ) char *s;
{
/* error printing routine in parser */
uerror( s, a, b, c, d, e, f, g );
}
yyaccpt()
{
ftnend();
}
ftnarg( idn )
register idn;
{
register struct symtab *p;
p = &stab[idn];
switch( p->stype )
{
case UNDEF:
/* this parameter, entered at scan */
break;
case FARG:
uerror("redeclaration of formal parameter, %s",p->sname);
/* fall thru */
case FTN:
/* the name of this function matches parm */
/* fall thru */
default:
idn = hide(p);
p = &stab[idn];
break;
case TNULL:
/* unused entry, fill it */
;
}
p->stype = FARG;
p->sclass = PARAM;
asave( idn );
}
talign( ty, s)
register unsigned ty;
register s;
{
/* compute the alignment of an object with type ty, sizeoff index s */
register i;
# ifdef FUNNYFLDS
if( s<0 && ty!=INT && ty!=CHAR && ty!=SHORT && ty!=UNSIGNED && ty!=UCHAR && ty!=USHORT
#ifdef LONGFIELDS
&& ty!=LONG && ty!=ULONG
#endif
)
{
return( fldal( ty ) );
}
# endif
for( i=0; i<=(SZINT-BTSHIFT-1); i+=TSHIFT )
{
switch( (ty>>i)&TMASK )
{
case FTN:
cerror( "compiler takes alignment of function");
case PTR:
return( ALPOINT );
case ARY:
continue;
case 0:
break;
}
}
switch( BTYPE(ty) )
{
case UNIONTY:
case ENUMTY:
case STRTY:
return( dimtab[ s+2 ] );
case CHAR:
case UCHAR:
return( ALCHAR );
case FLOAT:
return( ALFLOAT );
case DOUBLE:
return( ALDOUBLE );
case LONG:
case ULONG:
return( ALLONG );
case SHORT:
case USHORT:
return( ALSHORT );
default:
return( ALINT );
}
}
OFFSZ
tsize( ty, d, s )
register TWORD ty;
{
/* compute the size associated with type ty,
** dimoff d, and sizoff s
*/
/* BETTER NOT BE CALLED WHEN t, d, and s REFER TO A BIT FIELD... */
register i;
register OFFSZ mult;
mult = 1;
for( i=0; i<=(SZINT-BTSHIFT-1); i+=TSHIFT )
{
switch( (ty>>i)&TMASK )
{
case FTN:
uerror("illegal use of function pointer");
case PTR:
return( SZPOINT * mult );
case ARY:
mult *= dimtab[ d++ ];
continue;
case 0:
break;
}
}
if(ty == ENUMTY)
return(SZINT);
if(ty == VOID)
uerror("using a void value");
if( dimtab[s]==0 ) {
uerror( "unknown size");
return( SZINT );
}
return( dimtab[ s ] * mult );
}
inforce( n )
register OFFSZ n;
{
/* force inoff to have the value n */
/* inoff is updated to have the value n */
register OFFSZ wb;
register rest;
/* rest is used to do a lot of conversion to ints... */
if( inoff == n ) return;
if( inoff > n )
{
uerror("too many initializers (%d>%d)", inoff, n);
inoff = n;
return;
}
wb = inoff;
SETOFF( wb, SZINT );
/* wb now has the next higher word boundary */
if( wb >= n )
{
/* in the same word */
rest = n - inoff;
vfdzero( rest );
return;
}
/* otherwise, extend inoff to be word aligned */
rest = wb - inoff;
vfdzero( rest );
/* now, skip full words until near to n */
rest = (n-inoff)/SZINT;
zecode( rest );
/* now, the remainder of the last word */
rest = n-inoff;
vfdzero( rest );
if( inoff != n ) cerror( "inoff error");
}
vfdalign( n )
{
/* make inoff have the offset the next alignment of n */
register OFFSZ m;
m = inoff;
SETOFF( m, n );
inforce( m );
}
#ifndef NODBG
extern int idebug;
#endif
int ibseen = 0; /* the number of } constructions which have been filled */
int iclass; /* storage class of thing being initialized */
int ilocctr = 0; /* location counter for current initialization */
beginit(curid)
{
/* beginning of initilization; set location ctr and set type */
register struct symtab *p;
# ifndef NODBG
if( idebug >= 3 ) printf( "beginit(), curid = %d\n", curid );
# endif
p = &stab[curid];
iclass = p->sclass;
if( curclass == EXTERN || curclass == FORTRAN ) iclass = EXTERN;
switch( iclass )
{
case UNAME:
case EXTERN:
return;
case AUTO:
case REGISTER:
break;
case EXTDEF:
case STATIC:
ilocctr = ISARY(p->stype)?ADATA:DATA;
locctr( ilocctr );
defalign( talign( p->stype, p->sizoff ) );
if( p->slevel>1 && p->sclass == STATIC ) deflab( p->offset );
else defnam( p );
}
inoff = 0;
ibseen = 0;
pstk = 0;
instk( curid, p->stype, p->dimoff, p->sizoff, inoff );
}
instk( id, t, d, s, off )
register OFFSZ off;
register TWORD t;
{
/* make a new entry on the parameter stack to initialize id */
register struct symtab *p;
for(;;)
{
# ifndef NODBG
if( idebug ) printf( "instk((%d, %o,%d,%d, %d)\n", id, t, d, s, off );
# endif
/* save information on the stack */
if( !pstk ) pstk = instack;
else ++pstk;
pstk->in_fl = 0; /* { flag */
pstk->in_id = id ;
pstk->in_t = t ;
pstk->in_d = d ;
pstk->in_s = s ;
pstk->in_n = 0; /* number seen */
pstk->in_x = t==STRTY ?dimtab[s+1] : 0 ;
pstk->in_off = off; /* offset at beginning of this element */
/* if t is an array, DECREF(t) can't be a field */
/* INS_sz has size of array elements, and -size for fields */
if( ISARY(t) )
{
pstk->in_sz = tsize( DECREF(t), d+1, s );
}
else if( stab[id].sclass & FIELD )
{
pstk->in_sz = - ( stab[id].sclass & FLDSIZ );
}
else
{
pstk->in_sz = 0;
}
if( (iclass==AUTO || iclass == REGISTER ) &&
(ISARY(t) || t==STRTY) )
uerror( "no automatic aggregate initialization" );
/* now, if this is not a scalar, put on another element */
if( ISARY(t) )
{
t = DECREF(t);
++d;
continue;
}
else if( t == STRTY )
{
id = dimtab[pstk->in_x];
p = &stab[id];
if( p->sclass != MOS && !(p->sclass&FIELD) )
cerror( "insane structure member list" );
t = p->stype;
d = p->dimoff;
s = p->sizoff;
off += p->offset;
continue;
}
else return;
}
}
NODE *
getstr()
{
/* decide if the string is external or an initializer,
and get the contents accordingly */
register l, temp;
register NODE *p;
if( (iclass==EXTDEF||iclass==STATIC) &&
(pstk->in_t == CHAR || pstk->in_t == UCHAR) &&
pstk!=instack && ISARY( pstk[-1].in_t ) )
{
/* treat "abc" as { 'a', 'b', 'c', 0 } */
strflg = 1;
ilbrace(); /* simulate { */
inforce( pstk->in_off );
/* if the array is inflexible (not top level), pass in the size
** Be prepared to throw away unwanted initializers
*/
/* get the contents */
lxstr((pstk-1)!=instack?dimtab[(pstk-1)->in_d]:0);
irbrace(); /* simulate } */
return( NIL );
}
else
{
/* make a label, and get the contents and stash them away */
if( iclass != SNULL )
{
/* initializing */
/* fill out previous word, to permit pointer */
vfdalign( ALPOINT );
}
/* set up location counter */
temp = locctr( blevel==0?ISTRNG:STRNG );
deflab( l = getlab() );
strflg = 0;
lxstr(0); /* get the contents */
locctr( blevel==0?ilocctr:temp );
p = buildtree( STRING, NIL, NIL );
p->tn.rval = -l;
return(p);
}
}
putbyte( v )
{
/* simulate byte v appearing in a list of integer values */
register NODE *p;
p = bcon(v);
incode( p, SZCHAR );
tfree( p );
gotscal();
}
endinit()
{
register TWORD t;
register d, s, n, d1;
# ifndef NODBG
if( idebug ) printf( "endinit(), inoff = %d\n", inoff );
# endif
switch( iclass )
{
case EXTERN:
case AUTO:
case REGISTER:
iclass = SNULL;
return;
}
pstk = instack;
t = pstk->in_t;
d = pstk->in_d;
s = pstk->in_s;
n = pstk->in_n;
if( ISARY(t) )
{
d1 = dimtab[d];
/* fill out part of the last element, if needed */
vfdalign( pstk->in_sz );
n = inoff/pstk->in_sz; /* real number of initializers */
if( d1 >= n )
{
/* once again, t is an array, so no fields */
inforce( tsize( t, d, s ) );
n = d1;
}
if( d1!=0 && d1!=n ) uerror( "too many initializers (%d>%d)",
n, d1);
if( n==0 ) werror( "empty array declaration");
dimtab[d] = n;
}
else if( t == STRTY || t == UNIONTY )
{
/* clearly not fields either */
inforce( tsize( t, d, s ) );
}
else if( n > 1 ) uerror("%d initializers for a scalar", n);
/* this will never be called with a field element... */
else inforce( tsize(t,d,s) );
paramno = 0;
vfdalign( ALINIT );
inoff = 0;
iclass = SNULL;
}
doinit( p )
register NODE *p;
{
/* take care of generating a value for the initializer p */
/* inoff has the current offset (last bit written)
** in the current word being generated
*/
register sz, d, s;
register TWORD t;
/* note: size of an individual initializer assumed to fit into an int */
if(nerrors)
goto leave;
if( iclass < 0 ) goto leave;
if( iclass == EXTERN || iclass == UNAME )
{
uerror( "cannot initialize extern or union" );
iclass = -1;
goto leave;
}
if( iclass == AUTO || iclass == REGISTER )
{
/* do the initialization and get out, without regard
** for filing out the variable with zeros, etc.
*/
bccode();
idname = pstk->in_id;
p = buildtree( ASSIGN, buildtree( NAME, NIL, NIL ), p );
ecomp(p);
return;
}
if( p == NIL ) return; /* throw away strings already made into lists */
if( ibseen )
{
uerror( "} expected");
goto leave;
}
# ifndef NODBG
if( idebug > 1 ) printf( "doinit(%o)\n", p );
# endif
t = pstk->in_t; /* type required */
d = pstk->in_d;
s = pstk->in_s;
if( pstk->in_sz < 0 )
{
/* bit field */
sz = -pstk->in_sz;
}
else
{
sz = tsize( t, d, s );
}
inforce( pstk->in_off );
p = buildtree( ASSIGN, block( NAME, NIL,NIL, t, d, s ), p );
p->in.left->in.op = FREE;
p->in.left = p->in.right;
p->in.right = NIL;
p->in.left = optim( p->in.left );
if( p->in.left->in.op == UNARY AND )
{
p->in.left->in.op = FREE;
p->in.left = p->in.left->in.left;
}
p->in.op = INIT;
p = optim(p);
# ifndef NOFLOAT
if( p->in.left->in.op == FCON )
{
fincode( p->in.left->fpn.dval, sz );
}
else
# endif
if( p->in.left->in.op != ICON )
{
uerror( "initialization by non-constant" );
inoff += sz;
}
else
{
/* a constant is being used as initializer */
if( sz < SZINT )
{
/* special case: bit fields, etc. */
incode( p->in.left, sz );
}
else
{
# ifdef MYINIT
MYINIT( optim(p), sz );
# else
ecode( optim(p) );
inoff += sz;
# endif
}
}
gotscal();
leave:
tfree(p);
}
gotscal()
{
register t, ix;
register n, id;
register struct symtab *p;
register OFFSZ temp;
for( ; pstk > instack; )
{
if( pstk->in_fl ) ++ibseen;
--pstk;
t = pstk->in_t;
if( t == STRTY )
{
ix = ++pstk->in_x;
if( (id=dimtab[ix]) < 0 ) continue;
/* otherwise, put next element on the stack */
p = &stab[id];
instk( id, p->stype, p->dimoff, p->sizoff, p->offset+pstk->in_off );
return;
}
else if( ISARY(t) )
{
n = ++pstk->in_n;
if( n >= dimtab[pstk->in_d] && pstk > instack ) continue;
/* put the new element onto the stack */
temp = pstk->in_sz;
instk( pstk->in_id, (TWORD)DECREF(pstk->in_t), pstk->in_d+1, pstk->in_s,
pstk->in_off+n*temp );
return;
}
}
}
ilbrace()
{
/* process an initializer's left brace */
register t;
register struct instk *temp;
temp = pstk;
for( ; pstk > instack; --pstk )
{
t = pstk->in_t;
if( t != STRTY && !ISARY(t) ) continue; /* not an aggregate */
if( pstk->in_fl )
{
/* already associated with a { */
if( pstk->in_n ) uerror( "illegal {");
continue;
}
/* we have one ... */
pstk->in_fl = 1;
break;
}
/* cannot find one */
/* ignore such right braces */
pstk = temp;
}
irbrace()
{
/* called when a '}' is seen */
# ifndef NODBG
if( idebug ) printf( "irbrace(): paramno = %d on entry\n", paramno );
# endif
if( ibseen )
{
--ibseen;
return;
}
for( ; pstk > instack; --pstk )
{
if( !pstk->in_fl ) continue;
/* we have one now */
pstk->in_fl = 0; /* cancel { */
gotscal(); /* take it away... */
return;
}
/* these right braces match ignored left braces: throw out */
}
upoff( size, alignment, poff )
register alignment, *poff;
{
/* update the offset pointed to by poff; return the
** offset of a value of size `size', alignment `alignment',
** given that off is increasing
*/
register off;
off = *poff;
SETOFF( off, alignment );
if( (maxoffset-off) < size )
{
if( instruct!=INSTRUCT )cerror("local variables fill memory");
else cerror("structure larger than address space");
}
*poff = off+size;
return( off );
}
oalloc( p, poff )
register struct symtab *p;
register *poff;
{
/* allocate p with offset *poff, and update *poff */
register al, off, tsz;
int noff;
al = talign( p->stype, p->sizoff );
noff = off = *poff;
tsz = tsize( p->stype, p->dimoff, p->sizoff );
#ifdef BACKAUTO
if( p->sclass == AUTO )
{
if( (maxoffset-off) < tsz ) cerror("local variables fill memory");
noff = off + tsz;
SETOFF( noff, al );
off = -noff;
}
else
#endif
#ifdef BACKPARAM
/* this is rather nonstandard, and may be buggy in some cases */
/* in particular, won't work with ARGSRET and ARGALLRET */
if( p->sclass == PARAM )
{
if( (maxoffset-off) < tsz )
cerror("local variables fill memory");
/* if( tsz < SZINT ) */
{
noff = off + SZINT;
SETOFF( noff, ALINT );
off = -noff;
# ifndef RTOLBYTES
off += tsz;
#endif
}
/* else
/* {
/* noff = off + tsz;
/* SETOFF( noff, al );
/* off = -noff;
/* }
*/
}
else
#endif
if( p->sclass == PARAM && ( tsz % SZINT ) )
{ /* pjw stuff here. caller and callee must agree
that params are 4 byte aligned */
tsz = (tsz+SZINT-1)/SZINT;
off = upoff(tsz*SZINT, ALSTACK, &noff );
# ifndef RTOLBYTES
off = noff - tsz;????(see note above)
#endif
}
else
{
off = upoff( tsz, al, &noff );
}
if( p->sclass != REGISTER )
{
/* in case we are allocating stack space for register arguments */
if( p->offset == NOOFFSET ) p->offset = off;
else if( off != p->offset ) return(1);
}
*poff = noff;
return(0);
}
falloc( p, w, new, pty )
register struct symtab *p;
register NODE *pty;
{
/* allocate a field of width w */
/* new is 0 if new entry, 1 if redefinition, -1 if alignment */
register al,sz,type;
type = (new<0)? pty->in.type : p->stype;
/* this must be fixed to use the current type in alignments */
switch( new<0?pty->in.type:p->stype )
{
case ENUMTY:
{
register s;
s = new<0 ? pty->fn.csiz : p->sizoff;
al = dimtab[s+2];
sz = dimtab[s];
break;
}
case CHAR:
case UCHAR:
al = ALCHAR;
sz = SZCHAR;
break;
case SHORT:
case USHORT:
al = ALSHORT;
sz = SZSHORT;
break;
case INT:
case UNSIGNED:
al = ALINT;
sz = SZINT;
break;
#ifdef LONGFIELDS
case LONG:
case ULONG:
al = ALLONG;
sz = SZLONG;
break;
#endif
default:
# ifdef FUNNYFLDS
if( new < 0 )
{
# endif
uerror( "illegal field type" );
al = ALINT;
# ifdef FUNNYFLDS
}
else
{
al = fldal( p->stype );
sz =SZINT;
}
# endif
}
if( w > sz )
{
uerror( "field too big (%d>%d)", w, sz);
w = sz;
}
if( w == 0 )
{
/* align only */
SETOFF( strucoff, al );
if( new >= 0 ) uerror( "zero size field");
return(0);
}
if( strucoff%al + w > sz ) SETOFF( strucoff, al );
if( new < 0 )
{
if( (maxoffset-strucoff) < w )
cerror("structure fills memory");
strucoff += w; /* we know it will fit */
return(0);
}
/* establish the field */
if( new == 1 )
{
/* previous definition */
if( p->offset != strucoff || p->sclass != (FIELD|w) ) return(1);
}
p->offset = strucoff;
if( (maxoffset-strucoff) < w ) cerror("structure overflows memory");
strucoff += w;
p->stype = type;
return(0);
}
# ifndef EXCLASS
# define EXCLASS EXTERN
# endif
nidcl( p )
register NODE *p;
{
/* handle unitialized declarations */
/* assumed to be not functions */
register class;
register commflag; /* flag for labelled common declarations */
commflag = 0;
/* compute class */
if( (class=curclass) == SNULL )
{
if( blevel > 1 ) class = AUTO;
else if( blevel != 0 || instruct ) cerror( "nidcl error" );
else
{
/* blevel = 0 */
if( (class = EXCLASS) == EXTERN ) commflag = 1;
}
}
defid( p, class );
if( class==EXTDEF || class==STATIC )
{
# ifndef ALLCOMM
/* simulate initialization by 0 */
beginit( p->tn.rval );
endinit();
# else
commflag = 1;
# endif
}
if( commflag ) commdec( p->tn.rval );
}
TWORD
types( t1, t2, t3 )
register TWORD t1, t2, t3;
{
/* return a basic type from basic types t1, t2, and t3 */
TWORD t[3];
register TWORD noun, adj, unsg;
register i;
t[0] = t1;
t[1] = t2;
t[2] = t3;
unsg = INT; /* INT or UNSIGNED */
noun = UNDEF; /* INT, CHAR, or FLOAT */
adj = INT; /* INT, LONG, or SHORT */
for( i=0; i<3; ++i )
{
switch( t[i] )
{
default:
bad:
uerror( "illegal type combination" );
return( INT );
case UNDEF:
continue;
case UNSIGNED:
if( unsg != INT ) goto bad;
unsg = UNSIGNED;
continue;
case LONG:
case SHORT:
if( adj != INT ) goto bad;
adj = t[i];
continue;
case INT:
case CHAR:
case FLOAT:
if( noun != UNDEF ) goto bad;
noun = t[i];
continue;
}
}
/* now, construct final type */
if( noun == UNDEF ) noun = INT;
else if( noun == FLOAT )
{
if( unsg != INT || adj == SHORT ) goto bad;
return( adj==LONG ? DOUBLE : FLOAT );
}
else if( noun == CHAR && adj != INT ) goto bad;
/* now, noun is INT or CHAR */
if( adj != INT ) noun = adj;
if( unsg == UNSIGNED ) return( noun + (UNSIGNED-INT) );
else return( noun );
}
NODE *
tymerge( typ, idp )
register NODE *typ, *idp;
{
/* merge type typ with identifier idp */
register unsigned t;
register i;
if( typ->in.op != TYPE ) cerror( "tymerge: arg 1" );
if(idp == NIL ) return( NIL );
# ifndef NODBG
if( ddebug > 2 ) eprint(idp);
# endif
idp->in.type = typ->in.type;
idp->fn.cdim = curdim;
tyreduce( idp );
idp->fn.csiz = typ->fn.csiz;
for( t=typ->in.type, i=typ->fn.cdim; t&TMASK; t = DECREF(t) )
{
if( ISARY(t) ) dstash( dimtab[i++] );
}
/* now idp is a single node: fix up type */
idp->in.type = ctype( idp->in.type );
if( (t = BTYPE(idp->in.type)) != STRTY && t != UNIONTY && t != ENUMTY )
{
idp->fn.csiz = t; /* in case ctype has rewritten things */
}
return( idp );
}
tyreduce( p )
register NODE *p;
{
/* build a type, stash away dimensions, from a declaration parse tree */
/* the type is built top down, the dimensions bottom up */
register o, temp;
register unsigned t;
o = p->in.op;
p->in.op = FREE;
if( o == NAME ) return;
t = INCREF( p->in.type );
if( o == UNARY CALL ) t += (FTN-PTR);
else if( o == LB )
{
t += (ARY-PTR);
temp = p->in.right->tn.lval;
p->in.right->in.op = FREE;
}
p->in.left->in.type = t;
tyreduce( p->in.left );
if( o == LB ) dstash( temp );
p->tn.rval = p->in.left->tn.rval;
p->in.type = p->in.left->in.type;
}
fixtype( p, class )
register NODE *p;
register class;
{
register unsigned t, type;
register mod1, mod2;
/* fix up the types, and check for legality */
if( (type = p->in.type) == UNDEF ) return;
if( mod2 = (type&TMASK) )
{
t = DECREF(type);
while( mod1=mod2, mod2 = (t&TMASK) )
{
if( mod1 == ARY && mod2 == FTN )
{
uerror( "array of functions is illegal" );
type = 0;
}
else if( mod1==FTN && ( mod2==ARY || mod2==FTN ) )
{
uerror( "function returns array or function" );
type = 0;
}
t = DECREF(t);
}
}
/* detect function arguments, watching out for structure declarations */
/* for example, beware of f(x) struct [ int a[10]; } *x; { ... } */
/* the danger is that "a" will be converted to a pointer */
if( class==SNULL && blevel==1 && !(instruct&(INSTRUCT|INUNION)) )
class = PARAM;
if( class == PARAM || ( class==REGISTER && blevel==1 ) )
{
if( type == FLOAT ) type = DOUBLE;
else if( ISARY(type) )
{
++p->fn.cdim;
type += (PTR-ARY);
}
else if( ISFTN(type) )
{
werror( "a function is declared as an argument" );
type = INCREF(type);
}
}
if( instruct && ISFTN(type) )
{
uerror( "function illegal in structure or union" );
type = INCREF(type);
}
p->in.type = type;
}
# ifndef MYCTYPE
TWORD
ctype( t )
register TWORD t;
{
register TWORD bt;
bt = BTYPE(t);
# ifdef NOFLOAT
if( bt==FLOAT || bt==DOUBLE ) cerror( "ctype:f" );
# endif
# ifdef NOSHORT
# ifdef NOUNSIGNED
if( bt==SHORT || bt==USHORT ) MODTYPE(t,INT);
# else
if( bt==SHORT ) MODTYPE(t,INT);
else if( bt==USHORT) MODTYPE(t,UNSIGNED);
# endif
# endif
# ifdef NOLONG
# ifdef NOUNSIGNED
if( bt==LONG || bt==ULONG ) MODTYPE(t,INT);
# else
if( bt==LONG ) MODTYPE(t,INT);
else if( bt==ULONG) MODTYPE(t,UNSIGNED);
# endif
# endif
#if defined(M32B) && defined(IMPREGAL)
radbl( t );
#endif
return( t );
}
# endif
uclass( class )
register class;
{
/* give undefined version of class */
if( class == SNULL ) return( EXTERN );
else if( class == STATIC ) return( USTATIC );
else if( class == FORTRAN ) return( UFORTRAN );
else return( class );
}
fixclass( class, type )
register TWORD type;
register class;
{
/* first, fix null class */
if( class == SNULL )
{
if( instruct&INSTRUCT ) class = MOS;
else if( instruct&INUNION ) class = MOU;
else if( blevel == 0 ) class = EXTDEF;
else if( blevel == 1 ) class = PARAM;
else class = AUTO;
}
/* now, do general checking */
if(blevel == 1 && (class == STATIC || class == EXTERN))
uerror("parameter declared with weird storage classs");
if( ISFTN( type ) )
{
switch( class )
{
default:
uerror( "function has illegal storage class" );
case AUTO:
class = EXTERN;
case EXTERN:
case EXTDEF:
case FORTRAN:
case TYPEDEF:
case STATIC:
case UFORTRAN:
case USTATIC:
;
}
}
if( class&FIELD )
{
if( !(instruct&INSTRUCT) ) uerror( "field not in structure" );
return( class );
}
switch( class )
{
case MOU:
if( !(instruct&INUNION) ) uerror( "illegal class" );
return( class );
case MOS:
if( !(instruct&INSTRUCT) ) uerror( "illegal class" );
return( class );
case MOE:
if( instruct & (INSTRUCT|INUNION) ) uerror( "illegal class" );
return( class );
case REGISTER:
if( blevel == 0 ) uerror( "register declaration outside a fncn" );
/* if cisreg returns 1, nextrvar has the reg. number */
else if( cisreg( type ) ) return( class );
if( blevel == 1 ) return( PARAM );
else return( AUTO );
case AUTO:
case LABEL:
case ULABEL:
if( blevel < 2 ) uerror( "auto outside function" );
return( class );
case PARAM:
if( blevel != 1 ) uerror( "illegal class" );
return( class );
case UFORTRAN:
case FORTRAN:
# ifdef NOFORTRAN
NOFORTRAN; /* a condition which can regulate the FORTRAN usage */
# endif
if( !ISFTN(type) ) uerror( "fortran declaration must apply to function" );
else
{
type = DECREF(type);
if( ISFTN(type) || ISARY(type) || ISPTR(type) )
{
uerror( "fortran function has wrong type" );
}
}
case STNAME:
case UNAME:
case ENAME:
case EXTERN:
case STATIC:
case EXTDEF:
case TYPEDEF:
case USTATIC:
return( class );
default:
cerror( "illegal class: %d", class );
/* NOTREACHED */
}
}
struct symtab *
mknonuniq(idindex)
register *idindex;
{
/* locate a symbol table entry for */
/* an occurrence of a nonunique structure member name */
/* or field */
register i;
register struct symtab * sp;
sp = & stab[ i= *idindex ]; /* position search at old entry */
while( sp->stype != TNULL )
{
/* locate unused entry */
if( ++i >= SYMTSZ )
{
/* wrap around symbol table */
i = 0;
sp = stab;
}
else ++sp;
if( i == *idindex ) cerror("Symbol table full");
}
sp->sflags = SNONUNIQ | SMOS;
sp->sname = stab[*idindex].sname; /* old entry name */
*idindex = i;
# ifndef NODBG
if( ddebug )
{
printf( "\tnonunique entry for %s from %d to %d\n",
sp->sname, *idindex, i );
}
# endif
return ( sp );
}
#ifndef checkst
/* if not debugging, make checkst a macro */
checkst(lev)
{
register int s, i, j;
register struct symtab *p, *q;
for( i=0, p=stab; i<SYMTSZ; ++i, ++p )
{
if( p->stype == TNULL ) continue;
j = lookup( p->sname, p->sflags&(SMOS|STAG) );
if( j != i )
{
q = &stab[j];
if( q->stype == UNDEF ||
q->slevel <= p->slevel )
{
cerror( "check error: %s", q->sname );
}
}
else if( p->slevel > lev ) cerror( "%s check at level %d", p->sname, lev );
}
}
#endif
struct symtab *
relook(p)
register struct symtab *p;
{
/* look up p again, and see where it lies */
register struct symtab *q;
/* I'm not sure that this handles towers of several hidden definitions in all cases */
q = &stab[lookup( p->sname, p->sflags&(STAG|SMOS|SHIDDEN) )];
/* make relook always point to either p or an empty cell */
if( q->stype == UNDEF )
{
q->stype = TNULL;
return(q);
}
while( q != p )
{
if( q->stype == TNULL ) break;
if( ++q >= &stab[SYMTSZ] ) q=stab;
}
return(q);
}
uplevel()
{
++blevel;
#ifdef GDEBUG
dblbrac();
#endif
}
clearst( lev )
{
/* clear entries of internal scope from the symbol table */
extern struct symtab *scopestack[];
register struct symtab *p, *q, *r;
# ifdef GDEBUG
/* do this first, so structure members don't get clobbered
** before they are printed out...
*/
aobeg();
p = scopestack[blevel];
while (p)
{
aocode(p);
p = p->scopelink;
}
aoend();
if (lev != 1) /* param level */
dbrbrac();
# endif
/* q is to collect enteries which properly belong to
an outer scope */
q = (struct symtab *) NULL;
/* first, find an empty slot to prevent newly hashed entries
** from being slopped into...
*/
p = scopestack[lev];
while (p)
{
# ifndef NODBG
if (ddebug)
printf("removing %s = stab[ %d], flags %o level %d\n",
p->sname,p-stab,p->sflags,p->slevel);
# endif
/* if this is still undefined, complain */
if (p->stype == UNDEF ||
( p->sclass == ULABEL && lev <=2) ) {
uerror("%s undefined", p->sname);
}
if( p->sflags & SHIDES )
unhide(p);
if( (r=relook(p)) != p )
*r = *p;
if ( lev > p->slevel ) {
/* collect outer scope entries first
appearing in inner scope */
r = p;
p = p->scopelink;
r->scopelink = q;
q = r;
} else {
p->stype = TNULL;
p = p->scopelink;
}
}
scopestack[lev] = (struct symtab *) NULL;
/* move outer scope entries into holes, and up a level */
while( q )
{
r = &stab[ lookup( q->sname, q->sflags ) ];
if ( r != q )
{
*r = *q;
q->stype = TNULL;
}
q = q->scopelink;
r->scopelink = scopestack[ lev-1 ];
scopestack[ lev-1 ] = r;
}
}
hide( p )
register struct symtab *p;
{
register struct symtab *q;
for( q=p+1; ; ++q )
{
if( q >= &stab[SYMTSZ] ) q = stab;
if( q == p ) cerror( "symbol table full" );
if( q->stype == TNULL ) break;
}
*q = *p;
p->sflags |= SHIDDEN;
q->sflags = (p->sflags&(SMOS|STAG)) | SHIDES;
if( hflag ) werror( "%s redefinition hides earlier one", p->sname );
# ifndef NODBG
if( ddebug ) printf( " %d hidden in %d\n", p-stab, q-stab );
# endif
return( idname = q-stab );
}
unhide( p )
register struct symtab *p;
{
register struct symtab *q;
register s;
s = p->sflags & (SMOS|STAG);
q = p;
for(;;)
{
if( q == stab ) q = &stab[SYMTSZ-1];
else --q;
if( q == p ) break;
if( (q->sflags&(SMOS|STAG)) == s )
{
if ( p->sname == q->sname )
{
/* found the name */
q->sflags &= ~SHIDDEN;
# ifndef NODBG
if( ddebug ) printf( "unhide uncovered %d from %d\n", q-stab,p-stab);
# endif
return;
}
}
}
cerror( "unhide fails" );
}
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.