ack/lang/m2/comp/cstoper.c

519 lines
11 KiB
C
Raw Normal View History

1986-04-07 22:15:08 +00:00
/* C O N S T A N T E X P R E S S I O N H A N D L I N G */
1986-04-07 17:40:38 +00:00
1986-05-01 19:06:53 +00:00
#ifndef NORCSID
1986-04-07 17:40:38 +00:00
static char *RcsId = "$Header$";
1986-05-01 19:06:53 +00:00
#endif
#include "debug.h"
#include "target_sizes.h"
1986-04-07 17:40:38 +00:00
#include <em_arith.h>
#include <em_label.h>
#include <assert.h>
1986-04-17 09:28:09 +00:00
1986-04-07 17:40:38 +00:00
#include "idf.h"
#include "type.h"
#include "LLlex.h"
#include "node.h"
#include "Lpars.h"
1986-04-11 11:57:19 +00:00
#include "standards.h"
1986-04-07 17:40:38 +00:00
long mach_long_sign; /* sign bit of the machine long */
int mach_long_size; /* size of long on this machine == sizeof(long) */
1986-04-07 22:15:08 +00:00
long full_mask[MAXSIZE];/* full_mask[1] == 0xFF, full_mask[2] == 0xFFFF, .. */
1986-04-07 17:40:38 +00:00
arith max_int; /* maximum integer on target machine */
arith max_unsigned; /* maximum unsigned on target machine */
arith max_longint; /* maximum longint on target machine */
1986-04-09 18:14:49 +00:00
arith wrd_bits; /* number of bits in a word */
1986-04-07 17:40:38 +00:00
1986-04-08 18:15:46 +00:00
cstunary(expp)
1986-04-07 22:15:08 +00:00
register struct node *expp;
{
1986-04-08 18:15:46 +00:00
/* The unary operation in "expp" is performed on the constant
expression below it, and the result restored in expp.
1986-04-07 22:15:08 +00:00
*/
1986-04-08 18:15:46 +00:00
arith o1 = expp->nd_right->nd_INT;
1986-04-07 22:15:08 +00:00
1986-04-08 18:15:46 +00:00
switch(expp->nd_symb) {
1986-04-07 22:15:08 +00:00
case '+':
1986-04-08 18:15:46 +00:00
break;
1986-04-07 22:15:08 +00:00
case '-':
o1 = -o1;
1986-06-06 02:22:09 +00:00
if (expp->nd_type->tp_fund == T_INTORCARD) {
expp->nd_type = int_type;
}
1986-04-07 22:15:08 +00:00
break;
case NOT:
1986-04-22 22:36:16 +00:00
case '~':
1986-04-07 22:15:08 +00:00
o1 = !o1;
break;
default:
1986-05-28 18:36:51 +00:00
crash("(cstunary)");
1986-04-07 22:15:08 +00:00
}
1986-04-08 18:15:46 +00:00
expp->nd_class = Value;
expp->nd_token = expp->nd_right->nd_token;
1986-04-07 22:15:08 +00:00
expp->nd_INT = o1;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-08 18:15:46 +00:00
FreeNode(expp->nd_right);
expp->nd_right = 0;
1986-04-07 22:15:08 +00:00
}
1986-04-07 17:40:38 +00:00
1986-04-08 18:15:46 +00:00
cstbin(expp)
register struct node *expp;
1986-04-07 17:40:38 +00:00
{
1986-04-08 18:15:46 +00:00
/* The binary operation in "expp" is performed on the constant
expressions below it, and the result restored in
1986-04-07 22:15:08 +00:00
expp.
1986-04-07 17:40:38 +00:00
*/
1986-04-08 18:15:46 +00:00
arith o1 = expp->nd_left->nd_INT;
arith o2 = expp->nd_right->nd_INT;
1986-04-07 22:15:08 +00:00
int uns = expp->nd_type != int_type;
1986-04-07 17:40:38 +00:00
1986-04-08 18:15:46 +00:00
assert(expp->nd_class == Oper);
1986-05-01 19:06:53 +00:00
assert(expp->nd_left->nd_class == Value);
assert(expp->nd_right->nd_class == Value);
1986-04-08 18:15:46 +00:00
switch (expp->nd_symb) {
1986-04-07 17:40:38 +00:00
case '*':
o1 *= o2;
break;
1986-05-01 19:06:53 +00:00
1986-04-07 22:15:08 +00:00
case DIV:
1986-04-07 17:40:38 +00:00
if (o2 == 0) {
1986-04-08 18:15:46 +00:00
node_error(expp, "division by 0");
1986-04-07 22:15:08 +00:00
return;
1986-04-07 17:40:38 +00:00
}
if (uns) {
/* this is more of a problem than you might
think on C compilers which do not have
unsigned long.
*/
if (o2 & mach_long_sign) {/* o2 > max_long */
o1 = ! (o1 >= 0 || o1 < o2);
/* this is the unsigned test
o1 < o2 for o2 > max_long
*/
}
else { /* o2 <= max_long */
long half, bit, hdiv, hrem, rem;
half = (o1 >> 1) & ~mach_long_sign;
bit = o1 & 01;
/* now o1 == 2 * half + bit
and half <= max_long
and bit <= max_long
*/
hdiv = half / o2;
hrem = half % o2;
rem = 2 * hrem + bit;
o1 = 2 * hdiv + (rem < 0 || rem >= o2);
/* that is the unsigned compare
rem >= o2 for o2 <= max_long
*/
}
}
else
o1 /= o2;
break;
1986-05-01 19:06:53 +00:00
1986-04-07 22:15:08 +00:00
case MOD:
1986-04-07 17:40:38 +00:00
if (o2 == 0) {
1986-04-08 18:15:46 +00:00
node_error(expp, "modulo by 0");
1986-04-07 22:15:08 +00:00
return;
1986-04-07 17:40:38 +00:00
}
if (uns) {
if (o2 & mach_long_sign) {/* o2 > max_long */
o1 = (o1 >= 0 || o1 < o2) ? o1 : o1 - o2;
/* this is the unsigned test
o1 < o2 for o2 > max_long
*/
}
else { /* o2 <= max_long */
long half, bit, hrem, rem;
half = (o1 >> 1) & ~mach_long_sign;
bit = o1 & 01;
/* now o1 == 2 * half + bit
and half <= max_long
and bit <= max_long
*/
hrem = half % o2;
rem = 2 * hrem + bit;
o1 = (rem < 0 || rem >= o2) ? rem - o2 : rem;
}
}
else
o1 %= o2;
break;
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '+':
o1 += o2;
break;
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '-':
1986-04-07 22:15:08 +00:00
o1 -= o2;
1986-06-06 02:22:09 +00:00
if (expp->nd_type->tp_fund == T_INTORCARD) {
if (o1 < 0) expp->nd_type = int_type;
}
1986-04-07 17:40:38 +00:00
break;
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '<':
if (uns) {
o1 = (o1 & mach_long_sign ?
(o2 & mach_long_sign ? o1 < o2 : 0) :
(o2 & mach_long_sign ? 1 : o1 < o2)
);
}
else
1986-05-28 18:36:51 +00:00
o1 = (o1 < o2);
1986-04-07 17:40:38 +00:00
break;
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '>':
if (uns) {
o1 = (o1 & mach_long_sign ?
(o2 & mach_long_sign ? o1 > o2 : 1) :
(o2 & mach_long_sign ? 0 : o1 > o2)
);
}
else
1986-05-28 18:36:51 +00:00
o1 = (o1 > o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case LESSEQUAL:
1986-04-07 17:40:38 +00:00
if (uns) {
o1 = (o1 & mach_long_sign ?
(o2 & mach_long_sign ? o1 <= o2 : 0) :
(o2 & mach_long_sign ? 1 : o1 <= o2)
);
}
else
1986-05-28 18:36:51 +00:00
o1 = (o1 <= o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case GREATEREQUAL:
1986-04-07 17:40:38 +00:00
if (uns) {
o1 = (o1 & mach_long_sign ?
(o2 & mach_long_sign ? o1 >= o2 : 1) :
(o2 & mach_long_sign ? 0 : o1 >= o2)
);
}
else
1986-05-28 18:36:51 +00:00
o1 = (o1 >= o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case '=':
1986-05-28 18:36:51 +00:00
o1 = (o1 == o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case '#':
1986-05-28 18:36:51 +00:00
o1 = (o1 != o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case AND:
1986-04-22 22:36:16 +00:00
case '&':
1986-05-28 18:36:51 +00:00
o1 = (o1 && o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
case OR:
1986-05-28 18:36:51 +00:00
o1 = (o1 || o2);
1986-04-07 17:40:38 +00:00
break;
1986-04-07 22:15:08 +00:00
default:
1986-05-28 18:36:51 +00:00
crash("(cstbin)");
1986-04-07 17:40:38 +00:00
}
1986-04-08 18:15:46 +00:00
expp->nd_class = Value;
expp->nd_token = expp->nd_right->nd_token;
1986-05-28 18:36:51 +00:00
if (expp->nd_type == bool_type) expp->nd_symb = INTEGER;
1986-04-07 22:15:08 +00:00
expp->nd_INT = o1;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-08 18:15:46 +00:00
FreeNode(expp->nd_left);
FreeNode(expp->nd_right);
expp->nd_left = expp->nd_right = 0;
}
cstset(expp)
register struct node *expp;
{
1986-04-09 18:14:49 +00:00
register arith *set1 = 0, *set2;
1986-05-28 18:36:51 +00:00
arith *resultset = 0;
1986-04-09 18:14:49 +00:00
register int setsize, j;
assert(expp->nd_right->nd_class == Set);
assert(expp->nd_symb == IN || expp->nd_left->nd_class == Set);
set2 = expp->nd_right->nd_set;
1986-04-17 09:28:09 +00:00
setsize = expp->nd_right->nd_type->tp_size / word_size;
1986-04-09 18:14:49 +00:00
if (expp->nd_symb == IN) {
arith i;
assert(expp->nd_left->nd_class == Value);
1986-05-28 18:36:51 +00:00
1986-04-09 18:14:49 +00:00
i = expp->nd_left->nd_INT;
1986-05-28 18:36:51 +00:00
expp->nd_INT = (i >= 0 && set2 != 0 &&
1986-04-09 18:14:49 +00:00
i < setsize * wrd_bits &&
(set2[i / wrd_bits] & (1 << (i % wrd_bits))));
1986-05-28 18:36:51 +00:00
if (set2) free((char *) set2);
1986-04-09 18:14:49 +00:00
}
else {
set1 = expp->nd_left->nd_set;
1986-05-28 18:36:51 +00:00
resultset = set1;
expp->nd_left->nd_set = 0;
1986-04-09 18:14:49 +00:00
switch(expp->nd_symb) {
case '+':
1986-05-28 18:36:51 +00:00
if (!set1) {
resultset = set2;
expp->nd_right->nd_set = 0;
break;
}
if (set2) for (j = 0; j < setsize; j++) {
1986-04-09 18:14:49 +00:00
*set1++ |= *set2++;
}
break;
case '-':
1986-05-28 18:36:51 +00:00
if (!set1 || !set2) {
/* The set from which something is substracted
is already empty, or the set that is
substracted is empty
*/
break;
}
1986-04-09 18:14:49 +00:00
for (j = 0; j < setsize; j++) {
*set1++ &= ~*set2++;
}
break;
case '*':
1986-05-28 18:36:51 +00:00
if (!set1) break;
if (!set2) {
resultset = set2;
expp->nd_right->nd_set = 0;
break;
}
1986-04-09 18:14:49 +00:00
for (j = 0; j < setsize; j++) {
*set1++ &= *set2++;
}
break;
case '/':
1986-05-28 18:36:51 +00:00
if (!set1) {
resultset = set2;
expp->nd_right->nd_set = 0;
break;
}
if (set2) for (j = 0; j < setsize; j++) {
1986-04-09 18:14:49 +00:00
*set1++ ^= *set2++;
}
break;
case GREATEREQUAL:
case LESSEQUAL:
case '=':
case '#':
/* Clumsy, but who cares? Nobody writes these things! */
1986-05-28 18:36:51 +00:00
expp->nd_left->nd_set = set1;
1986-04-09 18:14:49 +00:00
for (j = 0; j < setsize; j++) {
switch(expp->nd_symb) {
case GREATEREQUAL:
1986-05-28 18:36:51 +00:00
if (!set2) {j = setsize; break; }
if (!set1) break;
1986-04-09 18:14:49 +00:00
if ((*set1 | *set2++) != *set1) break;
set1++;
continue;
case LESSEQUAL:
1986-05-28 18:36:51 +00:00
if (!set1) {j = setsize; break; }
if (!set2) break;
1986-04-09 18:14:49 +00:00
if ((*set2 | *set1++) != *set2) break;
set2++;
continue;
case '=':
case '#':
1986-05-28 18:36:51 +00:00
if (!set1 && !set2) {
j = setsize; break;
}
if (!set1 || !set2) break;
1986-04-09 18:14:49 +00:00
if (*set1++ != *set2++) break;
continue;
}
1986-05-28 18:36:51 +00:00
if (j < setsize) {
expp->nd_INT = expp->nd_symb == '#';
}
else {
expp->nd_INT = expp->nd_symb != '#';
}
1986-04-09 18:14:49 +00:00
break;
}
expp->nd_class = Value;
1986-05-23 09:46:31 +00:00
expp->nd_symb = INTEGER;
1986-05-28 18:36:51 +00:00
if (expp->nd_left->nd_set) {
free((char *) expp->nd_left->nd_set);
}
if (expp->nd_right->nd_set) {
free((char *) expp->nd_right->nd_set);
}
1986-05-23 09:46:31 +00:00
FreeNode(expp->nd_left);
FreeNode(expp->nd_right);
expp->nd_left = expp->nd_right = 0;
return;
1986-04-09 18:14:49 +00:00
default:
1986-05-28 18:36:51 +00:00
crash("(cstset)");
}
if (expp->nd_right->nd_set) {
free((char *) expp->nd_right->nd_set);
}
if (expp->nd_left->nd_set) {
free((char *) expp->nd_left->nd_set);
1986-04-09 18:14:49 +00:00
}
expp->nd_class = Set;
1986-05-28 18:36:51 +00:00
expp->nd_set = resultset;
1986-04-08 18:15:46 +00:00
}
1986-04-09 18:14:49 +00:00
FreeNode(expp->nd_left);
FreeNode(expp->nd_right);
expp->nd_left = expp->nd_right = 0;
1986-04-07 17:40:38 +00:00
}
1986-04-11 11:57:19 +00:00
cstcall(expp, call)
register struct node *expp;
{
/* a standard procedure call is found that can be evaluated
compile time, so do so.
*/
register struct node *expr = 0;
assert(expp->nd_class == Call);
if (expp->nd_right) {
expr = expp->nd_right->nd_left;
expp->nd_right->nd_left = 0;
FreeNode(expp->nd_right);
}
expp->nd_class = Value;
1986-05-23 09:46:31 +00:00
expp->nd_symb = INTEGER;
1986-04-11 11:57:19 +00:00
switch(call) {
case S_ABS:
if (expr->nd_INT < 0) expp->nd_INT = - expr->nd_INT;
else expp->nd_INT = expr->nd_INT;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-11 11:57:19 +00:00
break;
case S_CAP:
if (expr->nd_INT >= 'a' && expr->nd_INT <= 'z') {
expp->nd_INT = expr->nd_INT + ('A' - 'a');
}
else expp->nd_INT = expr->nd_INT;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-11 11:57:19 +00:00
break;
case S_CHR:
expp->nd_INT = expr->nd_INT;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-11 11:57:19 +00:00
break;
case S_MAX:
if (expp->nd_type == int_type) {
expp->nd_INT = max_int;
}
else if (expp->nd_type == longint_type) {
expp->nd_INT = max_longint;
}
else if (expp->nd_type == card_type) {
expp->nd_INT = max_unsigned;
}
else if (expp->nd_type->tp_fund == T_SUBRANGE) {
expp->nd_INT = expp->nd_type->sub_ub;
}
else expp->nd_INT = expp->nd_type->enm_ncst - 1;
break;
case S_MIN:
if (expp->nd_type == int_type) {
expp->nd_INT = (-max_int) - 1;
}
else if (expp->nd_type == longint_type) {
expp->nd_INT = (-max_longint) - 1;
}
else if (expp->nd_type->tp_fund == T_SUBRANGE) {
expp->nd_INT = expp->nd_type->sub_lb;
}
else expp->nd_INT = 0;
break;
case S_ODD:
expp->nd_INT = (expr->nd_INT & 1);
break;
case S_ORD:
expp->nd_INT = expr->nd_INT;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1986-04-11 11:57:19 +00:00
break;
case S_SIZE:
1986-04-17 09:28:09 +00:00
expp->nd_INT = align(expr->nd_type->tp_size, (int) word_size) /
word_size;
1986-04-11 11:57:19 +00:00
break;
case S_VAL:
expp->nd_INT = expr->nd_INT;
if ( /* Check overflow of subranges or enumerations */
( expp->nd_type->tp_fund == T_SUBRANGE
&&
( expp->nd_INT < expp->nd_type->sub_lb
|| expp->nd_INT > expp->nd_type->sub_ub
)
)
||
( expp->nd_type->tp_fund == T_ENUMERATION
&&
( expp->nd_INT < 0
|| expp->nd_INT >= expp->nd_type->enm_ncst
)
)
) node_warning(expp,"overflow in constant expression");
1986-05-01 19:06:53 +00:00
else CutSize(expp);
1986-04-11 11:57:19 +00:00
break;
default:
1986-05-28 18:36:51 +00:00
crash("(cstcall)");
1986-04-11 11:57:19 +00:00
}
FreeNode(expr);
FreeNode(expp->nd_left);
expp->nd_right = expp->nd_left = 0;
}
1986-05-01 19:06:53 +00:00
CutSize(expr)
1986-04-07 22:15:08 +00:00
register struct node *expr;
1986-04-07 17:40:38 +00:00
{
/* The constant value of the expression expr is made to
conform to the size of the type of the expression.
*/
1986-04-07 22:15:08 +00:00
arith o1 = expr->nd_INT;
1986-04-11 11:57:19 +00:00
struct type *tp = expr->nd_type;
int uns;
int size = tp->tp_size;
1986-04-07 17:40:38 +00:00
1986-04-07 22:15:08 +00:00
assert(expr->nd_class == Value);
1986-04-11 11:57:19 +00:00
if (tp->tp_fund == T_SUBRANGE) tp = tp->next;
uns = (tp->tp_fund & (T_CARDINAL|T_CHAR));
1986-04-07 17:40:38 +00:00
if (uns) {
1986-04-07 22:15:08 +00:00
if (o1 & ~full_mask[size]) {
node_warning(expr,
"overflow in constant expression");
}
1986-04-07 17:40:38 +00:00
o1 &= full_mask[size];
}
else {
int nbits = (int) (mach_long_size - size) * 8;
long remainder = o1 & ~full_mask[size];
1986-04-07 22:15:08 +00:00
if (remainder != 0 && remainder != ~full_mask[size]) {
node_warning(expr, "overflow in constant expression");
}
o1 <<= nbits;
1986-04-07 17:40:38 +00:00
o1 >>= nbits;
}
1986-04-07 22:15:08 +00:00
expr->nd_INT = o1;
1986-04-07 17:40:38 +00:00
}
1986-05-01 19:06:53 +00:00
InitCst()
1986-04-07 17:40:38 +00:00
{
int i = 0;
arith bt = (arith)0;
while (!(bt < 0)) {
bt = (bt << 8) + 0377, i++;
if (i == MAXSIZE)
fatal("array full_mask too small for this machine");
full_mask[i] = bt;
}
mach_long_size = i;
mach_long_sign = 1 << (mach_long_size * 8 - 1);
1986-04-17 09:28:09 +00:00
if (long_size > mach_long_size) {
1986-04-07 17:40:38 +00:00
fatal("sizeof (long) insufficient on this machine");
1986-04-08 23:34:10 +00:00
}
1986-04-07 17:40:38 +00:00
max_int = full_mask[int_size] & ~(1 << (int_size * 8 - 1));
max_unsigned = full_mask[int_size];
1986-04-17 09:28:09 +00:00
max_longint = full_mask[long_size] & ~(1 << (long_size * 8 - 1));
wrd_bits = 8 * word_size;
1986-04-07 17:40:38 +00:00
}