248 lines
4.4 KiB
C
248 lines
4.4 KiB
C
|
#include "debug.h"
|
||
|
|
||
|
#include <alloc.h>
|
||
|
#include <assert.h>
|
||
|
#include <em.h>
|
||
|
|
||
|
#include "LLlex.h"
|
||
|
#include "chk_expr.h"
|
||
|
#include "def.h"
|
||
|
#include "desig.h"
|
||
|
#include "idf.h"
|
||
|
#include "main.h"
|
||
|
#include "node.h"
|
||
|
#include "scope.h"
|
||
|
#include "type.h"
|
||
|
|
||
|
|
||
|
AssignStat(left, right)
|
||
|
register struct node *left, *right;
|
||
|
{
|
||
|
register struct type *ltp, *rtp;
|
||
|
struct desig dsr;
|
||
|
|
||
|
if( !(ChkExpression(right) && ChkLhs(left)) )
|
||
|
return;
|
||
|
|
||
|
ltp = left->nd_type;
|
||
|
rtp = right->nd_type;
|
||
|
|
||
|
if( !TstAssCompat(ltp, rtp) ) {
|
||
|
node_error(left, "type incompatibility in assignment");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if( rtp == emptyset_type )
|
||
|
right->nd_type = ltp;
|
||
|
|
||
|
if( !err_occurred ) {
|
||
|
dsr = InitDesig;
|
||
|
CodeExpr(right, &dsr, NO_LABEL);
|
||
|
|
||
|
if( rtp->tp_fund & (T_ARRAY | T_RECORD) )
|
||
|
CodeAddress(&dsr);
|
||
|
else {
|
||
|
CodeValue(&dsr, rtp);
|
||
|
|
||
|
if( ltp == real_type && BaseType(rtp) == int_type )
|
||
|
Int2Real();
|
||
|
|
||
|
RangeCheck(ltp, rtp);
|
||
|
}
|
||
|
CodeMove(&dsr, left, rtp);
|
||
|
}
|
||
|
|
||
|
FreeNode(left);
|
||
|
FreeNode(right);
|
||
|
}
|
||
|
|
||
|
ProcStat(nd)
|
||
|
register struct node *nd;
|
||
|
{
|
||
|
if( !ChkCall(nd) ) return;
|
||
|
|
||
|
if( nd->nd_type ) {
|
||
|
node_error(nd, "procedure call expected");
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ChkForStat(nd)
|
||
|
register struct node *nd;
|
||
|
{
|
||
|
register struct def *df;
|
||
|
|
||
|
if( !(ChkVariable(nd) && ChkExpression(nd->nd_left) &&
|
||
|
ChkExpression(nd->nd_right)) )
|
||
|
return;
|
||
|
|
||
|
assert(nd->nd_class == Def);
|
||
|
|
||
|
df = nd->nd_def;
|
||
|
|
||
|
if( df->df_scope != BlockScope ) {
|
||
|
node_error(nd, "for loop: control variable must be local");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
assert(df->df_kind == D_VARIABLE);
|
||
|
|
||
|
if( df->df_scope != GlobalScope && df->var_off >= 0 ) {
|
||
|
node_error(nd,"for loop: control variable can't be a parameter");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if( !(df->df_type->tp_fund & T_ORDINAL) ) {
|
||
|
node_error(nd, "for loop: control variable must be ordinal");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if( !TstCompat(df->df_type, nd->nd_left->nd_type) )
|
||
|
node_error(nd,
|
||
|
"for loop: initial value incompatible with control variable");
|
||
|
|
||
|
if( !TstCompat(df->df_type, nd->nd_right->nd_type) )
|
||
|
node_error(nd,
|
||
|
"for loop: final value incompatible with control variable");
|
||
|
|
||
|
df->df_flags |= D_LOOPVAR;
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
arith
|
||
|
CodeInitFor(nd, priority)
|
||
|
register struct node *nd;
|
||
|
{
|
||
|
/* Push init-value or final-value, the value may only be evaluated
|
||
|
once, so generate a temporary for it, when not a constant.
|
||
|
*/
|
||
|
|
||
|
arith tmp;
|
||
|
|
||
|
CodePExpr(nd);
|
||
|
if( nd->nd_class != Value ) {
|
||
|
tmp = NewInt(priority);
|
||
|
C_dup(int_size);
|
||
|
C_stl(tmp);
|
||
|
return tmp;
|
||
|
}
|
||
|
return (arith) 0;
|
||
|
}
|
||
|
|
||
|
CodeFor(nd, stepsize, l1, l2, tmp1)
|
||
|
struct node *nd;
|
||
|
label l1, l2;
|
||
|
arith tmp1;
|
||
|
{
|
||
|
/* Test if loop has to be done */
|
||
|
if( stepsize == 1 ) /* TO */
|
||
|
C_bgt(l2);
|
||
|
else /* DOWNTO */
|
||
|
C_blt(l2);
|
||
|
|
||
|
/* Store init-value in control-variable */
|
||
|
if( tmp1 )
|
||
|
C_lol(tmp1);
|
||
|
else
|
||
|
CodePExpr(nd->nd_left);
|
||
|
|
||
|
/* Label at begin of the body */
|
||
|
C_df_ilb(l1);
|
||
|
|
||
|
RangeCheck(nd->nd_type, nd->nd_left->nd_type);
|
||
|
CodeDStore(nd);
|
||
|
}
|
||
|
|
||
|
CodeEndFor(nd, stepsize, l1, l2, tmp2)
|
||
|
struct node *nd;
|
||
|
label l1, l2;
|
||
|
arith tmp2;
|
||
|
{
|
||
|
/* Test if loop has to be done once more */
|
||
|
CodePExpr(nd);
|
||
|
C_dup(int_size);
|
||
|
if( tmp2 )
|
||
|
C_lol(tmp2);
|
||
|
else
|
||
|
CodePExpr(nd->nd_right);
|
||
|
C_beq(l2);
|
||
|
|
||
|
/* Increment/decrement the control-variable */
|
||
|
if( stepsize == 1 ) /* TO */
|
||
|
C_inc();
|
||
|
else /* DOWNTO */
|
||
|
C_dec();
|
||
|
C_bra(l1);
|
||
|
|
||
|
/* Exit label */
|
||
|
C_df_ilb(l2);
|
||
|
}
|
||
|
|
||
|
WithStat(nd)
|
||
|
struct node *nd;
|
||
|
{
|
||
|
struct withdesig *wds;
|
||
|
struct desig ds;
|
||
|
struct scopelist *scl;
|
||
|
|
||
|
if( nd->nd_type->tp_fund != T_RECORD ) {
|
||
|
node_error(nd, "record variable expected");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if( err_occurred ) return;
|
||
|
|
||
|
/* Generate code */
|
||
|
|
||
|
CodeDAddress(nd);
|
||
|
|
||
|
wds = new_withdesig();
|
||
|
wds->w_next = WithDesigs;
|
||
|
WithDesigs = wds;
|
||
|
wds->w_scope = nd->nd_type->rec_scope;
|
||
|
|
||
|
/* create a desig structure for the temporary */
|
||
|
ds.dsg_kind = DSG_FIXED;
|
||
|
ds.dsg_offset = NewPtr(1);
|
||
|
ds.dsg_name = 0;
|
||
|
|
||
|
/* need some pointertype to store pointer */
|
||
|
CodeStore(&ds, nil_type);
|
||
|
|
||
|
/* record is indirectly available */
|
||
|
ds.dsg_kind = DSG_PFIXED;
|
||
|
wds->w_desig = ds;
|
||
|
|
||
|
scl = new_scopelist();
|
||
|
scl->sc_scope = wds->w_scope;
|
||
|
scl->next = CurrVis;
|
||
|
CurrVis = scl;
|
||
|
}
|
||
|
|
||
|
EndWith(saved_scl, nd)
|
||
|
struct scopelist *saved_scl;
|
||
|
struct node *nd;
|
||
|
{
|
||
|
/* restore scope, and release structures */
|
||
|
struct scopelist *scl;
|
||
|
struct withdesig *wds;
|
||
|
|
||
|
while( CurrVis != saved_scl ) {
|
||
|
|
||
|
/* release scopelist */
|
||
|
scl = CurrVis;
|
||
|
CurrVis = CurrVis->next;
|
||
|
free_scopelist(scl);
|
||
|
|
||
|
/* release temporary */
|
||
|
FreePtr(WithDesigs->w_desig.dsg_offset);
|
||
|
|
||
|
/* release withdesig */
|
||
|
wds = WithDesigs;
|
||
|
WithDesigs = WithDesigs->w_next;
|
||
|
free_withdesig(wds);
|
||
|
}
|
||
|
FreeNode(nd);
|
||
|
}
|