ack/lang/pc/comp/chk_expr.c

1345 lines
28 KiB
C

/* E X P R E S S I O N C H E C K I N G */
/* Check expressions, and try to evaluate them as far as possible.
*/
#include "debug.h"
#include <alloc.h>
#include <assert.h>
#include <em_arith.h>
#include <em_label.h>
#include "LLlex.h"
#include "Lpars.h"
#include "chk_expr.h"
#include "const.h"
#include "def.h"
#include "idf.h"
#include "main.h"
#include "misc.h"
#include "node.h"
#include "required.h"
#include "scope.h"
#include "type.h"
extern char *symbol2str();
STATIC
Xerror(nd, mess)
register struct node *nd;
char *mess;
{
if( nd->nd_class == Def && nd->nd_def ) {
if( nd->nd_def->df_kind != D_ERROR )
node_error(nd,"\"%s\": %s",
nd->nd_def->df_idf->id_text, mess);
}
else node_error(nd, "%s", mess);
}
struct node *
ZeroParam()
{
register struct node *nd;
nd = MkLeaf(Value, &dot);
nd->nd_type = int_type;
nd->nd_symb = INTEGER;
nd->nd_INT = (arith) 0;
nd = MkNode(Link, nd, NULLNODE, &dot);
nd->nd_symb = ',';
return nd;
}
MarkUsed(nd)
register struct node *nd;
{
while( nd && nd->nd_class != Def ) {
if( (nd->nd_class == Arrsel) || (nd->nd_class == LinkDef) )
nd = nd->nd_left;
else if( nd->nd_class == Arrow)
nd = nd->nd_right;
else break;
}
if( nd && nd->nd_class == Def ) {
register struct def *df = nd->nd_def;
if( df->df_kind != D_FIELD ) {
if( !(df->df_flags & (D_SET|D_VARPAR)) &&
(df->df_scope == CurrentScope) )
if( !is_anon_idf(df->df_idf) ) {
warning("\"%s\" used before set",
df->df_idf->id_text);
}
df->df_flags |= (D_USED | D_SET);
}
}
}
int
ChkConstant(expp)
register struct node *expp;
{
register struct node *nd;
if( !(nd = expp->nd_right) ) nd = expp;
if( nd->nd_class == Name && !ChkLinkOrName(nd) ) return 0;
if( nd->nd_class != Value || expp->nd_left ) {
Xerror(nd, "constant expected");
return 0;
}
if( expp->nd_class == Uoper )
return ChkUnOper(expp);
else if( nd != expp ) {
Xerror(expp, "constant expected");
return 0;
}
return 1;
}
int
ChkVariable(expp)
register struct node *expp;
{
/* Check that "expp" indicates an item that can be accessed */
if( !ChkLhs(expp) ) return 0;
if( expp->nd_class == Def && expp->nd_def->df_kind == D_FUNCTION ) {
Xerror(expp, "illegal use of function name");
return 0;
}
return 1;
}
int
ChkLhs(expp)
register struct node *expp;
{
int class;
/* Check that "expp" indicates an item that can be the lhs
of an assignment.
*/
if( !ChkVarAccess(expp) ) return 0;
class = expp->nd_class;
/* a constant is replaced by it's value in ChkLinkOrName, check here !,
* the remaining classes are checked by ChkVarAccess
*/
if( class == Value ) {
node_error(expp, "can't access a value");
return 0;
}
if( class == Def &&
!(expp->nd_def->df_kind & (D_FIELD | D_FUNCTION | D_VARIABLE)) ) {
Xerror(expp, "variable expected");
return 0;
}
/* assignment to function name */
if( class == Def && expp->nd_def->df_kind == D_FUNCTION )
if( expp->nd_def->prc_res )
expp->nd_type = ResultType(expp->nd_def->df_type);
else {
Xerror(expp, "illegal assignment to function-name");
return 0;
}
return 1;
}
#ifdef DEBUG
STATIC int
ChkValue(expp)
register struct node *expp;
{
switch( expp->nd_symb ) {
case INTEGER:
case REAL:
case STRING:
case NIL:
return 1;
default:
crash("(ChkValue)");
}
/*NOTREACHED*/
}
#endif
int
ChkLinkOrName(expp)
register struct node *expp;
{
register struct def *df;
expp->nd_type = error_type;
if( expp->nd_class == Name ) {
expp->nd_def = lookfor(expp, CurrVis, 1);
expp->nd_class = Def;
expp->nd_type = expp->nd_def->df_type;
}
else if( expp->nd_class == Link ) {
/* a selection from a record */
register struct node *left = expp->nd_left;
assert(expp->nd_symb == '.');
if( !ChkVariable(left) ) return 0;
if( left->nd_type->tp_fund != T_RECORD ) {
Xerror(left, "illegal selection");
return 0;
}
if( !(df = lookup(expp->nd_IDF, left->nd_type->rec_scope, D_INUSE)) ) {
id_not_declared(expp);
return 0;
}
else {
expp->nd_def = df;
expp->nd_type = df->df_type;
expp->nd_class = LinkDef;
}
return 1;
}
assert(expp->nd_class == Def);
df = expp->nd_def;
if( df->df_kind & (D_ENUM | D_CONST) ) {
MarkUsed(expp);
/* Replace an enum-literal or a CONST identifier by its value.
*/
if( df->df_kind == D_ENUM ) {
expp->nd_class = Value;
expp->nd_INT = df->enm_val;
expp->nd_symb = INTEGER;
}
else {
unsigned int ln = expp->nd_lineno;
assert(df->df_kind == D_CONST);
*expp = *(df->con_const);
expp->nd_lineno = ln;
}
}
return df->df_kind != D_ERROR;
}
STATIC int
ChkExLinkOrName(expp)
register struct node *expp;
{
if( !ChkLinkOrName(expp) ) return 0;
if( expp->nd_class != Def ) return 1;
if( !(expp->nd_def->df_kind & D_VALUE) ) {
Xerror(expp, "value expected");
}
return 1;
}
STATIC int
ChkUnOper(expp)
register struct node *expp;
{
/* Check an unary operation.
*/
register struct node *right = expp->nd_right;
register struct type *tpr;
if( !ChkExpression(right) ) return 0;
MarkUsed(right);
expp->nd_type = tpr = BaseType(right->nd_type);
switch( expp->nd_symb ) {
case '+':
if( tpr->tp_fund & T_NUMERIC ) {
*expp = *right;
free_node(right);
return 1;
}
break;
case '-':
if( tpr->tp_fund == T_INTEGER || tpr->tp_fund == T_LONG ) {
if( right->nd_class == Value )
cstunary(expp);
return 1;
}
if( tpr->tp_fund == T_REAL ) {
if( right->nd_class == Value ) {
expp->nd_token.tk_data.tk_real = right->nd_RIV;
expp->nd_class = Value;
expp->nd_symb = REAL;
FreeNode(right);
expp->nd_right = NULLNODE;
}
return 1;
}
break;
case NOT:
if( tpr == bool_type ) {
if( right->nd_class == Value )
cstunary(expp);
return 1;
}
break;
case '(':
/* Delete the brackets */
*expp = *right;
free_node(right);
return 1;
default:
crash("(ChkUnOper)");
}
node_error(expp, "\"%s\": illegal operand", symbol2str(expp->nd_symb));
return 0;
}
STATIC struct type *
ResultOfOperation(operator, tpl, tpr)
struct type *tpl, *tpr;
{
/* Return the result type of the binary operation "operator",
with operand types "tpl" and "tpr".
*/
switch( operator ) {
case '=' :
case NOTEQUAL :
case '<' :
case '>' :
case LESSEQUAL :
case GREATEREQUAL:
case IN :
return bool_type;
case '+' :
case '-' :
case '*' :
if( tpl == real_type || tpr == real_type )
return real_type;
if( tpl == long_type || tpr == long_type)
return long_type;
return tpl;
case '/' :
return real_type;
}
if (tpr == long_type && tpl == int_type) return tpr;
return tpl;
}
STATIC int
AllowedTypes(operator)
{
/* Return a bit mask indicating the allowed operand types for
binary operator "operator".
*/
switch( operator ) {
case '+' :
case '-' :
case '*' :
return T_NUMERIC | T_SET;
case '/' :
return T_NUMERIC;
case DIV :
case MOD :
return T_INTEGER | T_LONG;
case OR :
case AND :
return T_ENUMERATION;
case '=' :
case NOTEQUAL :
return T_ENUMERATION | T_CHAR | T_NUMERIC |
T_SET | T_POINTER | T_STRINGCONST |
T_STRING;
case LESSEQUAL :
case GREATEREQUAL:
return T_ENUMERATION | T_CHAR | T_NUMERIC |
T_SET | T_STRINGCONST;
case '<' :
case '>' :
return T_ENUMERATION | T_CHAR | T_NUMERIC |
T_STRINGCONST;
default :
crash("(AllowedTypes)");
}
/*NOTREACHED*/
}
STATIC int
Boolean(operator)
{
return operator == OR || operator == AND;
}
STATIC int
ChkBinOper(expp)
register struct node *expp;
{
/* Check a binary operation.
*/
register struct node *left, *right;
struct type *tpl, *tpr;
int retval, allowed;
left = expp->nd_left;
right = expp->nd_right;
retval = ChkExpression(left);
retval &= ChkExpression(right);
MarkUsed(left);
MarkUsed(right);
tpl = BaseType(left->nd_type);
tpr = BaseType(right->nd_type);
expp->nd_type = ResultOfOperation(expp->nd_symb, tpl ,tpr);
/* Check that the application of the operator is allowed on the type
of the operands.
There are some needles and pins:
- Boolean operators are only allowed on boolean operands, but the
"allowed-mask" of "AllowedTypes" can only indicate an enumeration
type.
- The IN-operator has as right-hand-side operand a set.
- Strings and packed arrays can be equivalent.
- In some cases, integers must be converted to reals.
- If one of the operands is the empty set then the result doesn't
have to be the empty set.
*/
if( expp->nd_symb == IN ) {
if( tpr->tp_fund != T_SET ) {
node_error(expp, "\"IN\": right operand must be a set");
return 0;
}
if( !TstAssCompat(tpl, ElementType(tpr)) ) {
node_error(expp, "\"IN\": incompatible types");
return 0;
}
if( left->nd_class == Value && right->nd_class == Set )
cstset(expp);
return retval;
}
if( !retval ) return 0;
allowed = AllowedTypes(expp->nd_symb);
if( !(tpr->tp_fund & allowed) || !(tpl->tp_fund & allowed) ) {
arith ub;
extern arith IsString();
if( allowed & T_STRINGCONST && (ub = IsString(tpl)) ) {
if( ub == IsString(tpr) )
return 1;
else {
node_error(expp, "\"%s\": incompatible types",
symbol2str(expp->nd_symb));
return 0;
}
}
else if( allowed & T_STRING && tpl->tp_fund == T_STRING )
return 1;
node_error(expp, "\"%s\": illegal operand type(s)",
symbol2str(expp->nd_symb));
return 0;
}
if( Boolean(expp->nd_symb) && tpl != bool_type ) {
node_error(expp, "\"%s\": illegal operand type(s)",
symbol2str(expp->nd_symb));
return 0;
}
if( allowed & T_NUMERIC ) {
if( (tpl == int_type || tpl == long_type) &&
(tpr == real_type || expp->nd_symb == '/') ) {
expp->nd_left =
MkNode(Cast, NULLNODE, expp->nd_left, &dot);
expp->nd_left->nd_type = tpl = real_type;
}
if( tpl == real_type &&
(tpr == int_type || tpr == long_type)) {
expp->nd_right =
MkNode(Cast, NULLNODE, expp->nd_right, &dot);
expp->nd_right->nd_type = tpr = real_type;
}
if( tpl == int_type && tpr == long_type) {
expp->nd_left =
MkNode(IntCoerc, NULLNODE, expp->nd_left, &dot);
expp->nd_left->nd_type = long_type;
}
else if( tpl == long_type && tpr == int_type) {
expp->nd_right =
MkNode(IntCoerc, NULLNODE, expp->nd_right, &dot);
expp->nd_right->nd_type = long_type;
}
}
/* Operands must be compatible */
if( !TstCompat(tpl, tpr) ) {
node_error(expp, "\"%s\": incompatible types",
symbol2str(expp->nd_symb));
return 0;
}
if( tpl->tp_fund & T_SET ) {
if( tpl == emptyset_type )
left->nd_type = tpr;
else if( tpr == emptyset_type )
right->nd_type = tpl;
if( expp->nd_type == emptyset_type )
expp->nd_type = tpr;
if( left->nd_class == Set && right->nd_class == Set )
cstset(expp);
}
else if( tpl->tp_fund != T_REAL &&
left->nd_class == Value && right->nd_class == Value )
cstbin(expp);
return 1;
}
STATIC int
ChkElement(expp, tp, set, cnt)
register struct node *expp;
register struct type **tp;
arith **set;
unsigned *cnt;
{
/* Check elements of a set. This routine may call itself
recursively. Also try to compute the set!
*/
register struct node *left = expp->nd_left;
register struct node *right = expp->nd_right;
register int i;
extern char *Malloc();
if( expp->nd_class == Link && expp->nd_symb == UPTO ) {
/* [ ... , expr1 .. expr2, ... ]
First check expr1 and expr2, and try to compute them.
*/
if( !ChkElement(left, tp, set, cnt) ||
!ChkElement(right, tp, set, cnt) )
return 0;
if( left->nd_class == Value &&
right->nd_class == Value && *set ) {
if( left->nd_INT > right->nd_INT ) {
/* Remove lower and upper bound of the range.
*/
*cnt -= 2;
(*set)[left->nd_INT/wrd_bits] &=
~(1 << (left->nd_INT%wrd_bits));
(*set)[right->nd_INT/wrd_bits] &=
~(1 << (right->nd_INT%wrd_bits));
}
else
/* We have a constant range. Put all elements
in the set.
*/
for( i = left->nd_INT + 1; i < right->nd_INT; i++ )
(*set)[i/wrd_bits] |= ( 1 << (i%wrd_bits) );
}
return 1;
}
/* Here, a single element is checked
*/
if( !ChkExpression(expp) ) return 0;
MarkUsed(expp);
if( *tp == emptyset_type ) {
/* first element in set determines the type of the set */
unsigned size;
*tp = set_type(expp->nd_type, 0);
size = (*tp)->tp_size * (sizeof(arith) / word_size);
*set = (arith *) Malloc(size);
clear((char *) *set, size);
}
else if( !TstCompat(ElementType(*tp), expp->nd_type) ) {
node_error(expp, "set element has incompatible type");
return 0;
}
if( expp->nd_class == Value ) {
/* a constant element
*/
i = expp->nd_INT;
if( expp->nd_type == int_type ) {
/* Check only integer base-types because they are not
equal to the integer host-type. The other base-types
are equal to their host-types.
*/
if( i < 0 || i > max_intset ) {
node_error(expp, "set element out of range");
return 0;
}
}
if( *set ) (*set)[i/wrd_bits] |= ( 1 << (i%wrd_bits));
(*cnt)++;
}
else if( *set ) {
free((char *) *set);
*set = (arith *) 0;
}
return 1;
}
STATIC int
ChkSet(expp)
register struct node *expp;
{
/* Check the legality of a SET aggregate, and try to evaluate it
compile time. Unfortunately this is all rather complicated.
*/
register struct node *nd = expp->nd_right;
arith *set = (arith *) 0;
unsigned cnt = 0;
assert(expp->nd_symb == SET);
expp->nd_type = emptyset_type;
/* Now check the elements given, and try to compute a constant set.
First allocate room for the set, but only if it isn't empty.
*/
if( !nd ) {
/* The resulting set IS empty, so we just return
*/
expp->nd_class = Set;
expp->nd_set = (arith *) 0;
return 1;
}
/* Now check the elements, one by one
*/
while( nd ) {
assert(nd->nd_class == Link && nd->nd_symb == ',');
if( !ChkElement(nd->nd_left, &(expp->nd_type), &set, &cnt) )
return 0;
nd = nd->nd_right;
}
if( set ) {
/* Yes, it was a constant set, and we managed to compute it!
Notice that at the moment there is no such thing as
partial evaluation. Either we evaluate the set, or we
don't (at all). Improvement not neccesary (???)
??? sets have a contant part and a variable part ???
*/
expp->nd_class = Set;
if( !cnt ) {
/* after all the work we've done, the set turned out
out to be empty!
*/
free((char *) set);
set = (arith *) 0;
}
expp->nd_set = set;
FreeNode(expp->nd_right);
expp->nd_right = NULLNODE;
}
return 1;
}
char *
ChkAllowedVar(nd, reading) /* reading indicates read or readln */
register struct node *nd;
{
char *message = 0;
switch( nd->nd_class ) {
case Def:
if( nd->nd_def->df_flags & D_INLOOP ) {
message = "control variable";
break;
}
if( nd->nd_def->df_kind != D_FIELD ) break;
/* FALL THROUGH */
case LinkDef:
assert(nd->nd_def->df_kind == D_FIELD);
if( nd->nd_def->fld_flags & F_PACKED )
message = "field of packed record";
else if( nd->nd_def->fld_flags & F_SELECTOR )
message = "variant selector";
break;
case Arrsel:
if( IsPacked(nd->nd_left->nd_type) )
if( !reading ) message = "component of packed array";
break;
case Arrow:
if( nd->nd_right->nd_type->tp_fund == T_FILE )
message = "filebuffer variable";
break;
default:
crash("(ChkAllowedVar)");
/*NOTREACHED*/
}
MarkDef(nd, D_SET, 1);
return message;
}
int
ChkVarPar(nd, name)
register struct node *nd, *name;
{
/* ISO 6.6.3.3 :
An actual variable parameter shall not denote a field
that is the selector of a variant-part or a component
of a variable where that variable possesses a type
that is designated packed.
*/
static char err_mes[80];
char *message = (char *) 0;
extern char *sprint();
if( !ChkVariable(nd) ) return 0;
message = ChkAllowedVar(nd, 0);
if( message ) {
sprint(err_mes, "%s can't be a variable parameter", message);
Xerror(name, err_mes);
return 0;
}
return 1;
}
STATIC struct node *
getarg(argp, bases, varaccess, name, paramtp)
struct node **argp, *name;
struct type *paramtp;
{
/* This routine is used to fetch the next argument from an
argument list. The argument list is indicated by "argp".
The parameter "bases" is a bitset indicating which types are
allowed at this point, and "varaccess" is a flag indicating
that the address from this argument is taken, so that it
must be a varaccess and may not be a register variable.
*/
register struct node *arg = (*argp)->nd_right;
register struct node *left;
if( !arg ) {
Xerror(name, "too few arguments supplied");
return 0;
}
left = arg->nd_left;
*argp = arg;
if( paramtp && paramtp->tp_fund & T_ROUTINE ) {
/* From the context it appears that the occurrence of the
procedure/function-identifier is not a call.
*/
if( left->nd_class != NameOrCall ) {
Xerror(name, "illegal proc/func parameter");
return 0;
}
else if( ChkLinkOrName(left->nd_left) ) {
left->nd_type = left->nd_left->nd_type;
MarkUsed(left->nd_left);
}
else return 0;
}
else if( varaccess ) {
if( !ChkVarPar(left, name) ) {
MarkUsed(left);
return 0;
}
}
else if( !ChkExpression(left) ) {
MarkUsed(left);
return 0;
}
MarkUsed(left);
if( !varaccess && bases == T_INTEGER &&
BaseType(left->nd_type)->tp_fund == T_LONG) {
arg->nd_left = MkNode(IntReduc, NULLNODE, left, &dot);
arg->nd_left->nd_type = int_type;
left = arg->nd_left;
}
if( bases && !(BaseType(left->nd_type)->tp_fund & bases) ) {
Xerror(name, "unexpected parameter type");
return 0;
}
return left;
}
STATIC int
ChkProcCall(expp)
struct node *expp;
{
/* Check a procedure call
*/
register struct node *left;
struct node *name;
register struct paramlist *param;
char ebuf[80];
int retval = 1;
int cnt = 0;
int new_par_section;
struct type *lasttp = NULLTYPE;
name = left = expp->nd_left;
if( left->nd_type == error_type ) {
/* Just check parameters as if they were value parameters
*/
expp->nd_type = error_type;
while( expp->nd_right )
(void) getarg(&expp, 0, 0, name, NULLTYPE);
return 0;
}
expp->nd_type = ResultType(left->nd_type);
/* Check parameter list
*/
for( param = ParamList(left->nd_type); param; param = param->next ) {
if( !(left = getarg(&expp, 0, (int) IsVarParam(param), name,
TypeOfParam(param))) )
return 0;
cnt++;
new_par_section = lasttp != TypeOfParam(param);
if( !TstParCompat(TypeOfParam(param), left->nd_type,
(int) IsVarParam(param), left, new_par_section) ) {
sprint(ebuf, "type incompatibility in parameter %d",
cnt);
Xerror(name, ebuf);
retval = 0;
}
/* Convert between integers and longs.
*/
if( !IsVarParam(param) && options['d'] ) {
if( left->nd_type->tp_fund == T_INTEGER &&
TypeOfParam(param)->tp_fund == T_LONG) {
expp->nd_left =
MkNode(IntCoerc, NULLNODE, left, &dot);
expp->nd_left->nd_type = long_type;
left = expp->nd_left;
}
else if( left->nd_type->tp_fund == T_LONG &&
TypeOfParam(param)->tp_fund == T_INTEGER) {
expp->nd_left =
MkNode(IntReduc, NULLNODE, left, &dot);
expp->nd_left->nd_type = int_type;
left = expp->nd_left;
}
}
if( left->nd_type == emptyset_type )
/* type of emptyset determined by the context */
left->nd_type = TypeOfParam(param);
lasttp = TypeOfParam(param);
}
if( expp->nd_right ) {
Xerror(name, "too many arguments supplied");
while( expp->nd_right )
(void) getarg(&expp, 0, 0, name, NULLTYPE);
return 0;
}
return retval;
}
STATIC int ChkStandard();
int
ChkCall(expp)
register struct node *expp;
{
/* Check something that looks like a procedure or function call.
Of course this does not have to be a call at all,
it may also be a standard procedure call.
*/
/* First, get the name of the function or procedure
*/
register struct node *left = expp->nd_left;
expp->nd_type = error_type;
if( ChkLinkOrName(left) ) {
MarkUsed(left);
if( IsProcCall(left) || left->nd_type == error_type ) {
/* A call.
It may also be a call to a standard procedure
*/
if( left->nd_type == std_type )
/* A standard procedure
*/
return ChkStandard(expp, left);
/* Here, we have found a real procedure call.
*/
}
else {
node_error(left, "procedure or function expected");
return 0;
}
}
return ChkProcCall(expp);
}
STATIC int
ChkExCall(expp)
register struct node *expp;
{
if( !ChkCall(expp) ) return 0;
if( !expp->nd_type ) {
node_error(expp, "function call expected");
return 0;
}
return 1;
}
STATIC int
ChkNameOrCall(expp)
register struct node *expp;
{
/* From the context it appears that the occurrence of the function-
identifier is a call to that function
*/
assert(expp->nd_class == NameOrCall);
expp->nd_class = Call;
return ChkExCall(expp);
}
STATIC int
ChkStandard(expp,left)
register struct node *expp, *left;
{
/* Check a call of a standard procedure or function
*/
struct node *arg = expp;
struct node *name = left;
int req;
assert(left->nd_class == Def);
req = left->nd_def->df_value.df_reqname;
switch( req ) {
case R_ABS:
case R_SQR:
if( !(left = getarg(&arg, T_NUMERIC, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = left->nd_type;
if( left->nd_class == Value &&
expp->nd_type->tp_fund != T_REAL )
cstcall(expp, req);
break;
case R_SIN:
case R_COS:
case R_EXP:
case R_LN:
case R_SQRT:
case R_ARCTAN:
if( !(left = getarg(&arg, T_NUMERIC, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = real_type;
if( BaseType(left->nd_type)->tp_fund == T_INTEGER ||
BaseType(left->nd_type)->tp_fund == T_LONG) {
arg->nd_left = MkNode(Cast,NULLNODE, arg->nd_left,&dot);
arg->nd_left->nd_type = real_type;
}
break;
case R_TRUNC:
case R_ROUND:
if( !(left = getarg(&arg, T_REAL, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = int_type;
break;
case R_ORD:
if( !(left = getarg(&arg, T_ORDINAL, 0, name, NULLTYPE)) )
return 0;
if( BaseType(left->nd_type)->tp_fund == T_LONG ) {
arg->nd_left = MkNode(IntReduc, NULLNODE, arg->nd_left, &dot);
arg->nd_left->nd_type = int_type;
}
expp->nd_type = int_type;
if( left->nd_class == Value )
cstcall(expp, R_ORD);
break;
case R_CHR:
if( !(left = getarg(&arg, T_INTEGER, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = char_type;
if( left->nd_class == Value )
cstcall(expp, R_CHR);
break;
case R_SUCC:
case R_PRED:
if( !(left = getarg(&arg, T_ORDINAL, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = left->nd_type;
if( left->nd_class == Value && options['R'] )
cstcall(expp, req);
break;
case R_ODD:
if( !(left = getarg(&arg, T_INTEGER | T_LONG , 0, name, NULLTYPE)) )
return 0;
expp->nd_type = bool_type;
if( left->nd_class == Value )
cstcall(expp, R_ODD);
break;
case R_EOF:
case R_EOLN:
case R_PAGE: {
int st_out;
if( req == R_PAGE ) {
expp->nd_type = NULLTYPE;
st_out = 1;
}
else {
expp->nd_type = bool_type;
st_out = 0;
}
if( !arg->nd_right ) {
struct node *nd;
if( !(nd = ChkStdInOut(name->nd_IDF->id_text, st_out)) )
return 0;
expp->nd_right = MkNode(Link, nd, NULLNODE, &dot);
expp->nd_right->nd_symb = ',';
arg = arg->nd_right;
}
else {
if( !(left = getarg(&arg, T_FILE, 1, name, NULLTYPE)) )
return 0;
if( req != R_EOF && left->nd_type != text_type ) {
Xerror(name, "textfile expected");
return 0;
}
}
break;
}
case R_REWRITE:
case R_PUT:
case R_RESET:
case R_GET:
if( !(left = getarg(&arg, T_FILE, 1, name, NULLTYPE)) )
return 0;
expp->nd_type = NULLTYPE;
break;
case R_PACK:
case R_UNPACK: {
struct type *tp1, *tp2, *tp3;
if( req == R_PACK ) {
/* pack(a, i, z) */
if( !(left = getarg(&arg, T_ARRAY, 1, name, NULLTYPE)) )
return 0;
tp1 = left->nd_type; /* (a) */
if( !(left = getarg(&arg, 0, 0, name, NULLTYPE)) )
return 0;
tp2 = left->nd_type; /* (i) */
if( !(left = getarg(&arg, T_ARRAY, 1, name, NULLTYPE)) )
return 0;
tp3 = left->nd_type; /* (z) */
}
else {
/* unpack(z, a, i) */
if( !(left = getarg(&arg, T_ARRAY, 1, name, NULLTYPE)) )
return 0;
tp3 = left->nd_type; /* (z) */
if( !(left = getarg(&arg, T_ARRAY, 1, name, NULLTYPE)) )
return 0;
tp1 = left->nd_type; /* (a) */
if( !(left = getarg(&arg, 0, 0, name, NULLTYPE)) )
return 0;
tp2 = left->nd_type; /* (i) */
}
if( IsConformantArray(tp1) || IsPacked(tp1) ) {
Xerror(name, "unpacked array expected");
return 0;
}
if( !TstAssCompat(IndexType(tp1), tp2) ) {
Xerror(name, "ordinal constant expected");
return 0;
}
if( IsConformantArray(tp3) || !IsPacked(tp3) ) {
Xerror(name, "packed array expected");
return 0;
}
if( !TstTypeEquiv(tp1->arr_elem, tp3->arr_elem) ) {
Xerror(name, "component types of arrays not equal");
return 0;
}
expp->nd_type = NULLTYPE;
break;
}
case R_NEW:
case R_DISPOSE:
if( !(left = getarg(&arg, T_POINTER, 1, name, NULLTYPE)) )
return 0;
if( arg->nd_right ) {
/* varargs new/dispose(p,c1,.....) */
register struct selector *sel;
register arith i;
if( PointedtoType(left->nd_type)->tp_fund != T_RECORD )
break;
sel = PointedtoType(left->nd_type)->rec_sel;
do {
if( !sel ) break;
arg = arg->nd_right;
left = arg->nd_left;
/* ISO : COMPILETIME CONSTANTS NOT PERMITTED */
if( !ChkConstant(left) ) return 0;
if( !TstCompat(left->nd_type, sel->sel_type) ) {
node_error(left,
"type incompatibility in caselabel");
return 0;
}
i = left->nd_INT - sel->sel_lb;
if( i < 0 || i >= sel->sel_ncst ) {
node_error(left,
"case constant: out of bounds");
return 0;
}
sel = sel->sel_ptrs[i];
} while( arg->nd_right );
FreeNode(expp->nd_right->nd_right);
expp->nd_right->nd_right = NULLNODE;
}
expp->nd_type = NULLTYPE;
break;
case R_MARK:
case R_RELEASE:
if( !(left = getarg(&arg, T_POINTER, 1, name, NULLTYPE)) )
return 0;
expp->nd_type = NULLTYPE;
break;
case R_HALT:
if( !arg->nd_right ) /* insert 0 parameter */
arg->nd_right = ZeroParam();
if( !(left = getarg(&arg, T_INTEGER, 0, name, NULLTYPE)) )
return 0;
expp->nd_type = NULLTYPE;
break;
default:
crash("(ChkStandard)");
}
if( arg->nd_right ) {
Xerror(name, "too many arguments supplied");
return 0;
}
return 1;
}
STATIC int
ChkArrow(expp)
register struct node *expp;
{
/* Check an application of the '^' operator.
The operand must be a variable of a pointer-type or a
variable of a file-type.
*/
register struct type *tp;
assert(expp->nd_class == Arrow);
assert(expp->nd_symb == '^');
expp->nd_type = error_type;
if( !ChkVariable(expp->nd_right) ) return 0;
MarkUsed(expp->nd_right);
tp = expp->nd_right->nd_type;
if( !(tp->tp_fund & (T_POINTER | T_FILE)) ) {
node_error(expp, "\"^\": illegal operand");
return 0;
}
expp->nd_type = PointedtoType(tp);
return 1;
}
STATIC int
ChkArr(expp)
register struct node *expp;
{
/* Check an array selection.
The left hand side must be a variable of an array type,
and the right hand side must be an expression that is
assignment compatible with the array-index.
*/
register struct type *tpl, *tpr;
int retval;
assert(expp->nd_class == Arrsel);
assert(expp->nd_symb == '[');
expp->nd_type = error_type;
/* Check the index first, so a[a[j]] is checked in order of
* evaluation. This to make sure that warnings are generated
* in the right order.
*/
retval = ChkExpression(expp->nd_right);
MarkUsed(expp->nd_right);
retval &= ChkVariable(expp->nd_left);
tpl = expp->nd_left->nd_type;
tpr = expp->nd_right->nd_type;
if( tpl == error_type || tpr == error_type ) return 0;
if( tpl->tp_fund != T_ARRAY ) {
node_error(expp, "not indexing an ARRAY type");
return 0;
}
/* Type of the index must be assignment compatible with
the index type of the array.
*/
if( !TstCompat(IndexType(tpl), tpr) ) {
node_error(expp, "incompatible index type");
return 0;
}
if( tpr == long_type ) {
expp->nd_right = MkNode(IntReduc, NULLNODE, expp->nd_right, &dot);
expp->nd_right->nd_type = int_type;
}
expp->nd_type = tpl->arr_elem;
return retval;
}
STATIC int
done_before()
{
return 1;
}
STATIC int
no_var_access(expp)
struct node *expp;
{
node_error(expp, "variable-access expected");
return 0;
}
extern int NodeCrash();
int (*ExprChkTable[])() = {
#ifdef DEBUG
ChkValue,
#else
done_before,
#endif
ChkExLinkOrName,
ChkUnOper,
ChkBinOper,
ChkSet,
NodeCrash,
ChkExCall,
ChkNameOrCall,
ChkArrow,
ChkArr,
NodeCrash,
ChkExLinkOrName,
NodeCrash,
NodeCrash,
NodeCrash,
NodeCrash
};
int (*VarAccChkTable[])() = {
no_var_access,
ChkLinkOrName,
no_var_access,
no_var_access,
no_var_access,
NodeCrash,
no_var_access,
no_var_access,
ChkArrow,
ChkArr,
done_before,
ChkLinkOrName,
done_before,
no_var_access,
no_var_access,
no_var_access
};