V10/cmd/f77/exec.c

Compare this file to the similar file:
Show the results in this format:

#include "defs"

LOCAL int exar2(), popctl(), pushctl();

/*   Logical IF codes
*/


exif(p)
expptr p;
{
	pushctl(CTLIF);
	ctlstack->elselabel = newlabel();
	putif(p, ctlstack->elselabel);
}



exelif(p)
expptr p;
{
	if(ctlstack->ctltype == CTLIF)
	{
		if(ctlstack->endlabel == 0)
			ctlstack->endlabel = newlabel();
		putgoto(ctlstack->endlabel);
		putlabel(ctlstack->elselabel);
		ctlstack->elselabel = newlabel();
		putif(p, ctlstack->elselabel);
	}

	else	execerr("elseif out of place", CNULL);
}





exelse()
{
	if(ctlstack->ctltype==CTLIF)
	{
		if(ctlstack->endlabel == 0)
			ctlstack->endlabel = newlabel();
		putgoto( ctlstack->endlabel );
		putlabel(ctlstack->elselabel);
		ctlstack->ctltype = CTLELSE;
	}

	else	execerr("else out of place", CNULL);
}


exendif()
{
	if(ctlstack->ctltype == CTLIF)
	{
		putlabel(ctlstack->elselabel);
		if(ctlstack->endlabel)
			putlabel(ctlstack->endlabel);
		popctl();
	}
	else if(ctlstack->ctltype == CTLELSE)
	{
		putlabel(ctlstack->endlabel);
		popctl();
	}

	else
		execerr("endif out of place", CNULL);
}



LOCAL pushctl(code)
int code;
{
	register int i;

	if(++ctlstack >= lastctl)
		many("loops or if-then-elses", 'c', maxctl);
	ctlstack->ctltype = code;
	for(i = 0 ; i < 4 ; ++i)
		ctlstack->ctlabels[i] = 0;
	++blklevel;
}


LOCAL popctl()
{
	if( ctlstack-- < ctls )
		fatal("control stack empty");
	--blklevel;
}



LOCAL poplab()
{
	register struct Labelblock  *lp;

	for(lp = labeltab ; lp < highlabtab ; ++lp)
		if(lp->labdefined)
		{
			/* mark all labels in inner blocks unreachable */
			if(lp->blklevel > blklevel)
				lp->labinacc = YES;
		}
		else if(lp->blklevel > blklevel)
		{
			/* move all labels referred to in inner blocks out a level */
			lp->blklevel = blklevel;
		}
}



/*  BRANCHING CODE
*/

exgoto(lab)
struct Labelblock *lab;
{
	putgoto(lab->labelno);
}







exequals(lp, rp)
register struct Primblock *lp;
register expptr rp;
{
	if(lp->tag != TPRIM)
	{
		err("assignment to a non-variable");
		frexpr(lp);
		frexpr(rp);
	}
	else if(lp->namep->vclass!=CLVAR && lp->argsp)
	{
		if(parstate >= INEXEC)
			err("statement function amid executables");
		else
			mkstfunct(lp, rp);
	}
	else
	{
		if(parstate < INDATA)
			enddcl();
		puteq(mklhs(lp), fixtype(rp));
	}
}


long laststfcn = -1, thisstno;

mkstfunct(lp, rp)
struct Primblock *lp;
expptr rp;
{
	register struct Primblock *p;
	register Namep np;
	chainp args;

	laststfcn = thisstno;
	np = lp->namep;
	if(np->vclass == CLUNKNOWN)
		np->vclass = CLPROC;
	else
	{
		dclerr("redeclaration of statement function", np);
		return;
	}
	np->vprocclass = PSTFUNCT;
	np->vstg = STGSTFUNCT;
	impldcl(np);
	args = (lp->argsp ? lp->argsp->listp : CHNULL);
	np->varxptr.vstfdesc = mkchain(args , rp );

	for( ; args ; args = args->nextp)
		if( args->datap->tag!=TPRIM ||
		    (p = (struct Primblock *) (args->datap) )->argsp ||
		    p->fcharp || p->lcharp )
			err("non-variable argument in statement function definition");
		else
		{
			args->datap = (tagptr) (p->namep);
			vardcl(p->namep);
			free(p);
		}
}



excall(name, args, nstars, labels)
Namep name;
struct Listblock *args;
int nstars;
struct Labelblock *labels[ ];
{
	register expptr p;

	settype(name, TYSUBR, ENULL);
	p = mkfunct( mkprim(name, args, CHNULL) );
	p->exprblock.vtype = p->exprblock.leftp->headblock.vtype = TYINT;
	if(nstars > 0)
		putcmgo(p, nstars, labels);
	else putexpr(p);
}



exstop(stop, p)
int stop;
register expptr p;
{
	char *q;
	int n;
	expptr mkstrcon();

	if(p)
	{
		if( ! ISCONST(p) )
		{
			execerr("pause/stop argument must be constant", CNULL);
			frexpr(p);
			p = mkstrcon(0, CNULL);
		}
		else if( ISINT(p->constblock.vtype) )
		{
			q = convic(p->constblock.Const.ci);
			n = strlen(q);
			if(n > 0)
			{
				p->constblock.Const.ccp = copyn(n, q);
				p->constblock.vtype = TYCHAR;
				p->constblock.vleng = (expptr) ICON(n);
			}
			else
				p = (expptr) mkstrcon(0, CNULL);
		}
		else if(p->constblock.vtype != TYCHAR)
		{
			execerr("pause/stop argument must be integer or string", CNULL);
			p = (expptr) mkstrcon(0, CNULL);
		}
	}
	else	p = (expptr) mkstrcon(0, CNULL);

	putexpr( call1(TYSUBR, (stop ? "s_stop" : "s_paus"), p) );
}

/* DO LOOP CODE */

#define DOINIT	par[0]
#define DOLIMIT	par[1]
#define DOINCR	par[2]

#define VARSTEP	0
#define POSSTEP	1
#define NEGSTEP	2


exdo(range, spec)
int range;
chainp spec;
{
	register expptr p, q;
	expptr q1;
	register Namep np;
	chainp cp;
	register int i;
	int dotype, incsign;
	Addrp dovarp, dostgp;
	expptr par[3];

	pushctl(CTLDO);
	dorange = ctlstack->dolabel = range;
	np = (Namep) (spec->datap);
	ctlstack->donamep = NULL;
	if(np->vdovar)
	{
		errstr("nested loops with variable %s", varstr(VL,np->varname));
		ctlstack->donamep = NULL;
		return;
	}

	dovarp = mkplace(np);
	if( ! ONEOF(dovarp->vtype, MSKINT|MSKREAL) )
	{
		err("bad type on do variable");
		return;
	}
	ctlstack->donamep = np;

	np->vdovar = YES;
	if( enregister(np) )
	{
		/* stgp points to a storage version, varp to a register version */
		dostgp = dovarp;
		dovarp = mkplace(np);
	}
	else
		dostgp = NULL;
	dotype = dovarp->vtype;

	for(i=0 , cp = spec->nextp ; cp!=NULL && i<3 ; cp = cp->nextp)
	{
		p = par[i++] = fixtype(cp->datap);
		if( ! ONEOF(p->headblock.vtype, MSKINT|MSKREAL) )
		{
			err("bad type on DO parameter");
			return;
		}
	}

	frchain(&spec);
	switch(i)
	{
	case 0:
	case 1:
		err("too few DO parameters");
		return;

	default:
		err("too many DO parameters");
		return;

	case 2:
		DOINCR = (expptr) ICON(1);

	case 3:
		break;
	}

	ctlstack->endlabel = newlabel();
	ctlstack->dobodylabel = newlabel();

	if( ISCONST(DOLIMIT) )
		ctlstack->domax = mkconv(dotype, DOLIMIT);
	else
		ctlstack->domax = (expptr) mktemp(dotype, PNULL);

	if( ISCONST(DOINCR) )
	{
		ctlstack->dostep = mkconv(dotype, DOINCR);
		if( (incsign = conssgn(ctlstack->dostep)) == 0)
			err("zero DO increment");
		ctlstack->dostepsign = (incsign > 0 ? POSSTEP : NEGSTEP);
	}
	else
	{
		ctlstack->dostep = (expptr) mktemp(dotype, PNULL);
		ctlstack->dostepsign = VARSTEP;
		ctlstack->doposlabel = newlabel();
		ctlstack->doneglabel = newlabel();
	}

	if( ISCONST(ctlstack->domax) && ISCONST(DOINIT) && ctlstack->dostepsign!=VARSTEP)
	{
		puteq(cpexpr(dovarp), cpexpr(DOINIT));
		if( onetripflag )
			frexpr(DOINIT);
		else
		{
			q = mkexpr(OPMINUS, cpexpr(DOINIT),
				cpexpr(ctlstack->domax));
			if(incsign == (i = conssgn(q)) || !i && bugwarn & 2)
			{
				warn("DO range never executed");
				putgoto(ctlstack->endlabel);
			}
			else if (!i && bugwarn)
				warnb("old f77 never executed the DO range");
			frexpr(q);
		}
	}
	else if(ctlstack->dostepsign!=VARSTEP && !onetripflag)
	{
		if( ISCONST(ctlstack->domax) )
			q = (expptr) cpexpr(ctlstack->domax);
		else
			q = mkexpr(OPASSIGN, cpexpr(ctlstack->domax), DOLIMIT);

		q1 = mkexpr(OPASSIGN, cpexpr(dovarp), DOINIT);
		q = mkexpr( (ctlstack->dostepsign==POSSTEP ? OPLE : OPGE), q1, q);
		putif(q, ctlstack->endlabel);
	}
	else
	{
		if(! ISCONST(ctlstack->domax) )
			puteq( cpexpr(ctlstack->domax), DOLIMIT);
		q = DOINIT;
		if( ! onetripflag )
			q = mkexpr(OPMINUS, q,
			    mkexpr(OPASSIGN, cpexpr(ctlstack->dostep), DOINCR) );
		puteq( cpexpr(dovarp), q);
		if(onetripflag && ctlstack->dostepsign==VARSTEP)
			puteq( cpexpr(ctlstack->dostep), DOINCR);
	}

	if(ctlstack->dostepsign == VARSTEP)
	{
		if(onetripflag)
			putgoto(ctlstack->dobodylabel);
		else
			putif( mkexpr(OPGE, cpexpr(ctlstack->dostep), ICON(0)),
			    ctlstack->doneglabel );
		putlabel(ctlstack->doposlabel);
		putif( mkexpr(OPLE,
		    mkexpr(OPPLUSEQ, cpexpr(dovarp), cpexpr(ctlstack->dostep)),
		    cpexpr(ctlstack->domax) ),
		    ctlstack->endlabel);
	}
	putlabel(ctlstack->dobodylabel);
	if(dostgp)
		puteq(dostgp, cpexpr(dovarp));
	frexpr(dovarp);
}



enddo(here)
int here;
{
	register struct Ctlframe *q;
	register expptr t;
	Namep np;
	Addrp ap;
	register int i;

	while(here == dorange)
	{
		if(np = ctlstack->donamep)
		{
			t = mkexpr(OPPLUSEQ, mkplace(ctlstack->donamep),
			    cpexpr(ctlstack->dostep) );

			if(ctlstack->dostepsign == VARSTEP)
			{
				putif( mkexpr(OPLE, cpexpr(ctlstack->dostep), ICON(0)), ctlstack->doposlabel);
				putlabel(ctlstack->doneglabel);
				putif( mkexpr(OPLT, t, ctlstack->domax), ctlstack->dobodylabel);
			}
			else
				putif( mkexpr( (ctlstack->dostepsign==POSSTEP ? OPGT : OPLT),
				    t, ctlstack->domax),
				    ctlstack->dobodylabel);
			putlabel(ctlstack->endlabel);
			if(ap = memversion(np))
				puteq(ap, mkplace(np));
			for(i = 0 ; i < 4 ; ++i)
				ctlstack->ctlabels[i] = 0;
			deregister(ctlstack->donamep);
			ctlstack->donamep->vdovar = NO;
			frexpr(ctlstack->dostep);
		}

		popctl();
		poplab();
		dorange = 0;
		for(q = ctlstack ; q>=ctls ; --q)
			if(q->ctltype == CTLDO)
			{
				dorange = q->dolabel;
				break;
			}
	}
}

 chainp Lblfudgelist;

 expptr
labelfudge(t, newno)
 register int t;
{
	register chainp cp;
	register Addrp A;

	for(cp = Lblfudgelist; cp; cp = cp->nextp->nextp) 
		if ((int)cp->datap == t)
			break;
	if (cp) {
		A = (Addrp)cp->nextp->datap;
		if (newno)
			cp->datap = (tagptr)newno;
		}
	else {
		if (newno)
			return 0;
		A = ALLOC(Addrblock);
		A->tag = TADDR;
		A->vtype = TYLONG;
		A->vclass = CLVAR;
		A->vstg = STGINIT;
		A->memno = ++lastvarno;
		A->memoffset = ICON(0);
		Lblfudgelist = mkchain((tagptr)t,
			mkchain((tagptr)A, Lblfudgelist));
		}
	return (expptr)cpexpr((tagptr)A);
	}

exassign(vname, labelval)
Namep vname;
struct Labelblock *labelval;
{
	Addrp p;
	expptr mkaddcon();

	p = mkplace(vname);
	if( ! ONEOF(p->vtype, MSKINT|MSKADDR) )
		err("noninteger assign variable");
	else
		puteq(p, labelval->labtype == LABUNKNOWN
			? labelfudge(labelval->labelno,0)
			: mkaddcon(labelval->labelno) );
}



exarif(expr, neglab, zerlab, poslab)
expptr expr;
struct Labelblock *neglab, *zerlab, *poslab;
{
	register int lm, lz, lp;

	lm = neglab->labelno;
	lz = zerlab->labelno;
	lp = poslab->labelno;
	expr = fixtype(expr);

	if( ! ONEOF(expr->headblock.vtype, MSKINT|MSKREAL) )
	{
		err("invalid type of arithmetic if expression");
		frexpr(expr);
	}
	else
	{
		if(lm == lz)
			exar2(OPLE, expr, lm, lp);
		else if(lm == lp)
			exar2(OPNE, expr, lm, lz);
		else if(lz == lp)
			exar2(OPGE, expr, lz, lm);
		else
			prarif(expr, lm, lz, lp);
	}
}



LOCAL exar2(op, e, l1, l2)
int op;
expptr e;
int l1, l2;
{
	putif( mkexpr(op, e, ICON(0)), l2);
	putgoto(l1);
}


exreturn(p)
register expptr p;
{
	if(procclass != CLPROC)
		warn("RETURN statement in main or block data");
	if(p && (proctype!=TYSUBR || procclass!=CLPROC) )
	{
		err("alternate return in nonsubroutine");
		p = 0;
	}

	if(p)
	{
		putforce(TYINT, p);
		putgoto(retlabel);
	}
	else
		putgoto(proctype==TYSUBR ? ret0label : retlabel);
}



exasgoto(labvar)
struct Hashentry *labvar;
{
	register Addrp p;

	p = mkplace(labvar);
	if( ! ISINT(p->vtype) )
		err("assigned goto variable must be integer");
	else
		putbranch(p);
}