ack/lang/fortran/comp/misc.c

1042 lines
17 KiB
C
Raw Normal View History

1991-10-07 16:35:03 +00:00
/****************************************************************
Copyright 1990 by AT&T Bell Laboratories and Bellcore.
Permission to use, copy, modify, and distribute this software
and its documentation for any purpose and without fee is hereby
granted, provided that the above copyright notice appear in all
copies and that both that the copyright notice and this
permission notice and warranty disclaimer appear in supporting
documentation, and that the names of AT&T Bell Laboratories or
Bellcore or any of their entities not be used in advertising or
publicity pertaining to distribution of the software without
specific, written prior permission.
AT&T and Bellcore disclaim all warranties with regard to this
software, including all implied warranties of merchantability
and fitness. In no event shall AT&T or Bellcore be liable for
any special, indirect or consequential damages or any damages
whatsoever resulting from loss of use, data or profits, whether
in an action of contract, negligence or other tortious action,
arising out of or in connection with the use or performance of
this software.
****************************************************************/
#include "defs.h"
int oneof_stg (name, stg, mask)
Namep name;
int stg, mask;
{
if (stg == STGCOMMON && name) {
if ((mask & M(STGEQUIV)))
return name->vcommequiv;
if ((mask & M(STGCOMMON)))
return !name->vcommequiv;
}
return ONEOF(stg, mask);
}
/* op_assign -- given a binary opcode, return the associated assignment
operator */
int op_assign (opcode)
int opcode;
{
int retval = -1;
switch (opcode) {
case OPPLUS: retval = OPPLUSEQ; break;
case OPMINUS: retval = OPMINUSEQ; break;
case OPSTAR: retval = OPSTAREQ; break;
case OPSLASH: retval = OPSLASHEQ; break;
case OPMOD: retval = OPMODEQ; break;
case OPLSHIFT: retval = OPLSHIFTEQ; break;
case OPRSHIFT: retval = OPRSHIFTEQ; break;
case OPBITAND: retval = OPBITANDEQ; break;
case OPBITXOR: retval = OPBITXOREQ; break;
case OPBITOR: retval = OPBITOREQ; break;
default:
erri ("op_assign: bad opcode '%d'", opcode);
break;
} /* switch */
return retval;
} /* op_assign */
char *
Alloc(n) /* error-checking version of malloc */
/* ckalloc initializes memory to 0; Alloc does not */
int n;
{
char errbuf[32];
register char *rv;
rv = malloc(n);
if (!rv) {
sprintf(errbuf, "malloc(%d) failure!", n);
Fatal(errbuf);
}
return rv;
}
cpn(n, a, b)
register int n;
register char *a, *b;
{
while(--n >= 0)
*b++ = *a++;
}
eqn(n, a, b)
register int n;
register char *a, *b;
{
while(--n >= 0)
if(*a++ != *b++)
return(NO);
return(YES);
}
cmpstr(a, b, la, lb) /* compare two strings */
register char *a, *b;
ftnint la, lb;
{
register char *aend, *bend;
aend = a + la;
bend = b + lb;
if(la <= lb)
{
while(a < aend)
if(*a != *b)
return( *a - *b );
else
{
++a;
++b;
}
while(b < bend)
if(*b != ' ')
return(' ' - *b);
else
++b;
}
else
{
while(b < bend)
if(*a != *b)
return( *a - *b );
else
{
++a;
++b;
}
while(a < aend)
if(*a != ' ')
return(*a - ' ');
else
++a;
}
return(0);
}
/* hookup -- Same as LISP NCONC, that is a destructive append of two lists */
chainp hookup(x,y)
register chainp x, y;
{
register chainp p;
if(x == NULL)
return(y);
for(p = x ; p->nextp ; p = p->nextp)
;
p->nextp = y;
return(x);
}
struct Listblock *mklist(p)
chainp p;
{
register struct Listblock *q;
q = ALLOC(Listblock);
q->tag = TLIST;
q->listp = p;
return(q);
}
chainp mkchain(p,q)
register char * p;
register chainp q;
{
register chainp r;
if(chains)
{
r = chains;
chains = chains->nextp;
}
else
r = ALLOC(Chain);
r->datap = p;
r->nextp = q;
return(r);
}
chainp
revchain(next)
register chainp next;
{
register chainp p, prev = 0;
while(p = next) {
next = p->nextp;
p->nextp = prev;
prev = p;
}
return prev;
}
/* addunder -- turn a cvarname into an external name */
/* The cvarname may already end in _ (to avoid C keywords); */
/* if not, it has room for appending an _. */
char *
addunder(s)
register char *s;
{
register int c, i;
char *s0 = s;
i = 0;
while(c = *s++)
if (c == '_')
i++;
else
i = 0;
if (!i) {
*s-- = 0;
*s = '_';
}
return( s0 );
}
/* copyn -- return a new copy of the input Fortran-string */
char *copyn(n, s)
register int n;
register char *s;
{
register char *p, *q;
p = q = (char *) Alloc(n);
while(--n >= 0)
*q++ = *s++;
return(p);
}
/* copys -- return a new copy of the input C-string */
char *copys(s)
char *s;
{
return( copyn( strlen(s)+1 , s) );
}
/* convci -- Convert Fortran-string to integer; assumes that input is a
legal number, with no trailing blanks */
ftnint convci(n, s)
register int n;
register char *s;
{
ftnint sum;
sum = 0;
while(n-- > 0)
sum = 10*sum + (*s++ - '0');
return(sum);
}
/* convic - Convert Integer constant to string */
char *convic(n)
ftnint n;
{
static char s[20];
register char *t;
s[19] = '\0';
t = s+19;
do {
*--t = '0' + n%10;
n /= 10;
} while(n > 0);
return(t);
}
/* mkname -- add a new identifier to the environment, including the closed
hash table. */
Namep mkname(s)
register char *s;
{
struct Hashentry *hp;
register Namep q;
register int c, hash, i;
register char *t;
char *s0;
char errbuf[64];
hash = i = 0;
s0 = s;
while(c = *s++) {
hash += c;
if (c == '_')
i = 1;
}
hash %= maxhash;
/* Add the name to the closed hash table */
hp = hashtab + hash;
while(q = hp->varp)
if( hash == hp->hashval && !strcmp(s0,q->fvarname) )
return(q);
else if(++hp >= lasthash)
hp = hashtab;
if(++nintnames >= maxhash-1)
many("names", 'n', maxhash); /* Fatal error */
hp->varp = q = ALLOC(Nameblock);
hp->hashval = hash;
q->tag = TNAME; /* TNAME means the tag type is NAME */
c = s - s0;
if (c > 7 && noextflag) {
sprintf(errbuf, "\"%.35s%s\" over 6 characters long", s0,
c > 36 ? "..." : "");
errext(errbuf);
}
q->fvarname = strcpy(mem(c,0), s0);
t = q->cvarname = mem(c + i + 1, 0);
s = s0;
/* add __ to the end of any name containing _ */
while(*t = *s++)
t++;
if (i) {
t[0] = t[1] = '_';
t[2] = 0;
}
else if (in_vector(s0) >= 0) {
t[0] = '_';
t[1] = 0;
}
return(q);
}
struct Labelblock *mklabel(l)
ftnint l;
{
register struct Labelblock *lp;
if(l <= 0)
return(NULL);
for(lp = labeltab ; lp < highlabtab ; ++lp)
if(lp->stateno == l)
return(lp);
if(++highlabtab > labtabend)
many("statement labels", 's', maxstno);
lp->stateno = l;
lp->labelno = newlabel();
lp->blklevel = 0;
lp->labused = NO;
lp->fmtlabused = NO;
lp->labdefined = NO;
lp->labinacc = NO;
lp->labtype = LABUNKNOWN;
lp->fmtstring = 0;
return(lp);
}
newlabel()
{
return( ++lastlabno );
}
/* this label appears in a branch context */
struct Labelblock *execlab(stateno)
ftnint stateno;
{
register struct Labelblock *lp;
if(lp = mklabel(stateno))
{
if(lp->labinacc)
warn1("illegal branch to inner block, statement label %s",
convic(stateno) );
else if(lp->labdefined == NO)
lp->blklevel = blklevel;
if(lp->labtype == LABFORMAT)
err("may not branch to a format");
else
lp->labtype = LABEXEC;
}
else
execerr("illegal label %s", convic(stateno));
return(lp);
}
/* find or put a name in the external symbol table */
Extsym *mkext(f,s)
char *f, *s;
{
Extsym *p;
for(p = extsymtab ; p<nextext ; ++p)
if(!strcmp(s,p->cextname))
return( p );
if(nextext >= lastext)
many("external symbols", 'x', maxext);
nextext->fextname = strcpy(gmem(strlen(f)+1,0), f);
nextext->cextname = f == s
? nextext->fextname
: strcpy(gmem(strlen(s)+1,0), s);
nextext->extstg = STGUNKNOWN;
nextext->extp = 0;
nextext->allextp = 0;
nextext->extleng = 0;
nextext->maxleng = 0;
nextext->extinit = 0;
nextext->curno = nextext->maxno = 0;
return( nextext++ );
}
Addrp builtin(t, s, dbi)
int t, dbi;
char *s;
{
register Extsym *p;
register Addrp q;
extern chainp used_builtins;
p = mkext(s,s);
if(p->extstg == STGUNKNOWN)
p->extstg = STGEXT;
else if(p->extstg != STGEXT)
{
errstr("improper use of builtin %s", s);
return(0);
}
q = ALLOC(Addrblock);
q->tag = TADDR;
q->vtype = t;
q->vclass = CLPROC;
q->vstg = STGEXT;
q->memno = p - extsymtab;
q->dbl_builtin = dbi;
/* A NULL pointer here tells you to use memno to check the external
symbol table */
q -> uname_tag = UNAM_EXTERN;
/* Add to the list of used builtins */
if (dbi >= 0)
add_extern_to_list (q, &used_builtins);
return(q);
}
add_extern_to_list (addr, list_store)
Addrp addr;
chainp *list_store;
{
chainp last = CHNULL;
chainp list;
int memno;
if (list_store == (chainp *) NULL || addr == (Addrp) NULL)
return;
list = *list_store;
memno = addr -> memno;
for (;list; last = list, list = list -> nextp) {
Addrp this = (Addrp) (list -> datap);
if (this -> tag == TADDR && this -> uname_tag == UNAM_EXTERN &&
this -> memno == memno)
return;
} /* for */
if (*list_store == CHNULL)
*list_store = mkchain((char *)cpexpr((expptr)addr), CHNULL);
else
last->nextp = mkchain((char *)cpexpr((expptr)addr), CHNULL);
} /* add_extern_to_list */
frchain(p)
register chainp *p;
{
register chainp q;
if(p==0 || *p==0)
return;
for(q = *p; q->nextp ; q = q->nextp)
;
q->nextp = chains;
chains = *p;
*p = 0;
}
void
frexchain(p)
register chainp *p;
{
register chainp q, r;
if (q = *p) {
for(;;q = r) {
frexpr((expptr)q->datap);
if (!(r = q->nextp))
break;
}
q->nextp = chains;
chains = *p;
*p = 0;
}
}
tagptr cpblock(n,p)
register int n;
register char * p;
{
register ptr q;
memcpy((char *)(q = ckalloc(n)), (char *)p, n);
return( (tagptr) q);
}
ftnint lmax(a, b)
ftnint a, b;
{
return( a>b ? a : b);
}
ftnint lmin(a, b)
ftnint a, b;
{
return(a < b ? a : b);
}
maxtype(t1, t2)
int t1, t2;
{
int t;
t = t1 >= t2 ? t1 : t2;
if(t==TYCOMPLEX && (t1==TYDREAL || t2==TYDREAL) )
t = TYDCOMPLEX;
return(t);
}
/* return log base 2 of n if n a power of 2; otherwise -1 */
log_2(n)
ftnint n;
{
int k;
/* trick based on binary representation */
if(n<=0 || (n & (n-1))!=0)
return(-1);
for(k = 0 ; n >>= 1 ; ++k)
;
return(k);
}
frrpl()
{
struct Rplblock *rp;
while(rpllist)
{
rp = rpllist->rplnextp;
free( (charptr) rpllist);
rpllist = rp;
}
}
/* Call a Fortran function with an arbitrary list of arguments */
int callk_kludge;
expptr callk(type, name, args)
int type;
char *name;
chainp args;
{
register expptr p;
p = mkexpr(OPCALL,
(expptr)builtin(callk_kludge ? callk_kludge : type, name, 0),
(expptr)args);
p->exprblock.vtype = type;
return(p);
}
expptr call4(type, name, arg1, arg2, arg3, arg4)
int type;
char *name;
expptr arg1, arg2, arg3, arg4;
{
struct Listblock *args;
args = mklist( mkchain((char *)arg1,
mkchain((char *)arg2,
mkchain((char *)arg3,
mkchain((char *)arg4, CHNULL)) ) ) );
return( callk(type, name, (chainp)args) );
}
expptr call3(type, name, arg1, arg2, arg3)
int type;
char *name;
expptr arg1, arg2, arg3;
{
struct Listblock *args;
args = mklist( mkchain((char *)arg1,
mkchain((char *)arg2,
mkchain((char *)arg3, CHNULL) ) ) );
return( callk(type, name, (chainp)args) );
}
expptr call2(type, name, arg1, arg2)
int type;
char *name;
expptr arg1, arg2;
{
struct Listblock *args;
args = mklist( mkchain((char *)arg1, mkchain((char *)arg2, CHNULL) ) );
return( callk(type,name, (chainp)args) );
}
expptr call1(type, name, arg)
int type;
char *name;
expptr arg;
{
return( callk(type,name, (chainp)mklist(mkchain((char *)arg,CHNULL)) ));
}
expptr call0(type, name)
int type;
char *name;
{
return( callk(type, name, CHNULL) );
}
struct Impldoblock *mkiodo(dospec, list)
chainp dospec, list;
{
register struct Impldoblock *q;
q = ALLOC(Impldoblock);
q->tag = TIMPLDO;
q->impdospec = dospec;
q->datalist = list;
return(q);
}
/* ckalloc -- Allocate 1 memory unit of size n, checking for out of
memory error */
ptr ckalloc(n)
register int n;
{
register ptr p;
if( p = (ptr)calloc(1, (unsigned) n) )
return(p);
fprintf(stderr, "failing to get %d bytes\n",n);
Fatal("out of memory");
/* NOT REACHED */ return 0;
}
isaddr(p)
register expptr p;
{
if(p->tag == TADDR)
return(YES);
if(p->tag == TEXPR)
switch(p->exprblock.opcode)
{
case OPCOMMA:
return( isaddr(p->exprblock.rightp) );
case OPASSIGN:
case OPASSIGNI:
case OPPLUSEQ:
case OPMINUSEQ:
case OPSLASHEQ:
case OPMODEQ:
case OPLSHIFTEQ:
case OPRSHIFTEQ:
case OPBITANDEQ:
case OPBITXOREQ:
case OPBITOREQ:
return( isaddr(p->exprblock.leftp) );
}
return(NO);
}
isstatic(p)
register expptr p;
{
extern int useauto;
if(p->headblock.vleng && !ISCONST(p->headblock.vleng))
return(NO);
switch(p->tag)
{
case TCONST:
return(YES);
case TADDR:
if(ONEOF(p->addrblock.vstg,MSKSTATIC) &&
ISCONST(p->addrblock.memoffset) && !useauto)
return(YES);
default:
return(NO);
}
}
/* addressable -- return True iff it is a constant value, or can be
referenced by constant values */
addressable(p)
register expptr p;
{
switch(p->tag)
{
case TCONST:
return(YES);
case TADDR:
return( addressable(p->addrblock.memoffset) );
default:
return(NO);
}
}
/* isnegative_const -- returns true if the constant is negative. Returns
false for imaginary and nonnumeric constants */
int isnegative_const (cp)
struct Constblock *cp;
{
int retval;
if (cp == NULL)
return 0;
switch (cp -> vtype) {
case TYSHORT:
case TYLONG:
retval = cp -> Const.ci < 0;
break;
case TYREAL:
case TYDREAL:
retval = cp->vstg ? *cp->Const.cds[0] == '-'
: cp->Const.cd[0] < 0.0;
break;
default:
retval = 0;
break;
} /* switch */
return retval;
} /* isnegative_const */
negate_const(cp)
Constp cp;
{
if (cp == (struct Constblock *) NULL)
return;
switch (cp -> vtype) {
case TYSHORT:
case TYLONG:
cp -> Const.ci = - cp -> Const.ci;
break;
case TYCOMPLEX:
case TYDCOMPLEX:
if (cp->vstg)
switch(*cp->Const.cds[1]) {
case '-':
++cp->Const.cds[1];
break;
case '0':
break;
default:
--cp->Const.cds[1];
}
else
cp->Const.cd[1] = -cp->Const.cd[1];
/* no break */
case TYREAL:
case TYDREAL:
if (cp->vstg)
switch(*cp->Const.cds[0]) {
case '-':
++cp->Const.cds[0];
break;
case '0':
break;
default:
--cp->Const.cds[0];
}
else
cp->Const.cd[0] = -cp->Const.cd[0];
break;
case TYCHAR:
case TYLOGICAL:
erri ("negate_const: can't negate type '%d'", cp -> vtype);
break;
default:
erri ("negate_const: bad type '%d'",
cp -> vtype);
break;
} /* switch */
} /* negate_const */
ffilecopy (infp, outfp)
FILE *infp, *outfp;
{
while (!feof (infp)) {
register c = getc (infp);
if (!feof (infp))
putc (c, outfp);
} /* while */
} /* ffilecopy */
#define NOT_IN_VECTOR -1
/* in_vector -- verifies whether str is in c_keywords.
If so, the index is returned else NOT_IN_VECTOR is returned.
c_keywords must be in alphabetical order (as defined by strcmp).
*/
int in_vector(str)
char *str;
{
extern int n_keywords;
extern char *c_keywords[];
register int n = n_keywords;
register char **K = c_keywords;
register int n1, t;
do {
n1 = n >> 1;
if (!(t = strcmp(str, K[n1])))
return K - c_keywords + n1;
if (t < 0)
n = n1;
else {
n -= ++n1;
K += n1;
}
}
while(n > 0);
return NOT_IN_VECTOR;
} /* in_vector */
int is_negatable (Const)
Constp Const;
{
int retval = 0;
if (Const != (Constp) NULL)
switch (Const -> vtype) {
case TYSHORT:
retval = Const -> Const.ci >= -BIGGEST_SHORT;
break;
case TYLONG:
retval = Const -> Const.ci >= -BIGGEST_LONG;
break;
case TYREAL:
case TYDREAL:
case TYCOMPLEX:
case TYDCOMPLEX:
retval = 1;
break;
case TYLOGICAL:
case TYCHAR:
case TYSUBR:
default:
retval = 0;
break;
} /* switch */
return retval;
} /* is_negatable */
backup(fname, bname)
char *fname, *bname;
{
FILE *b, *f;
static char couldnt[] = "Couldn't open %.80s";
if (!(f = fopen(fname, binread))) {
warn1(couldnt, fname);
return;
}
if (!(b = fopen(bname, binwrite))) {
warn1(couldnt, bname);
return;
}
ffilecopy(f, b);
fclose(f);
fclose(b);
}
/* struct_eq -- returns YES if structures have the same field names and
types, NO otherwise */
int struct_eq (s1, s2)
chainp s1, s2;
{
struct Dimblock *d1, *d2;
Constp cp1, cp2;
if (s1 == CHNULL && s2 == CHNULL)
return YES;
for(; s1 && s2; s1 = s1->nextp, s2 = s2->nextp) {
register Namep v1 = (Namep) s1 -> datap;
register Namep v2 = (Namep) s2 -> datap;
if (v1 == (Namep) NULL || v1 -> tag != TNAME ||
v2 == (Namep) NULL || v2 -> tag != TNAME)
return NO;
if (v1->vtype != v2->vtype || v1->vclass != v2->vclass
|| strcmp(v1->fvarname, v2->fvarname))
return NO;
/* compare dimensions (needed for comparing COMMON blocks) */
if (d1 = v1->vdim) {
if (!(cp1 = (Constp)d1->nelt) || cp1->tag != TCONST)
return NO;
if (!(d2 = v2->vdim))
if (cp1->Const.ci == 1)
continue;
else
return NO;
if (!(cp2 = (Constp)d2->nelt) || cp2->tag != TCONST
|| cp1->Const.ci != cp2->Const.ci)
return NO;
}
else if ((d2 = v2->vdim) && (!(cp2 = (Constp)d2->nelt)
|| cp2->tag != TCONST
|| cp2->Const.ci != 1))
return NO;
} /* while s1 != CHNULL && s2 != CHNULL */
return s1 == CHNULL && s2 == CHNULL;
} /* struct_eq */