ack/lang/m2/comp/cstoper.c

595 lines
11 KiB
C
Raw Normal View History

/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*
* Author: Ceriel J.H. Jacobs
*/
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
/* $Header$ */
1986-05-01 19:06:53 +00:00
#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-11-05 14:33:00 +00:00
#include "warning.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, .. */
1987-11-26 14:15:24 +00:00
long max_int[MAXSIZE]; /* max_int[1] == 0x7F, max_int[2] == 0x7FFF, .. */
long min_int[MAXSIZE]; /* min_int[1] == 0xFFFFFF80, min_int[2] = 0xFFFF8000,
...
*/
1987-07-21 13:54:33 +00:00
unsigned int wrd_bits; /* number of bits in a word */
1986-04-07 17:40:38 +00:00
1987-06-23 17:12:25 +00:00
extern char options[];
1987-11-26 14:15:24 +00:00
overflow(expp)
t_node *expp;
{
node_warning(expp, W_ORDINARY, "overflow in constant expression");
}
arith
ar_abs(i)
arith i;
{
return i < 0 ? -i : i;
}
1986-11-05 14:33:00 +00:00
1986-04-08 18:15:46 +00:00
cstunary(expp)
register t_node *expp;
1986-04-07 22:15:08 +00:00
{
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
*/
register t_node *right = expp->nd_right;
1986-04-07 22:15:08 +00:00
1986-04-08 18:15:46 +00:00
switch(expp->nd_symb) {
1986-10-06 20:36:30 +00:00
/* Should not get here
1986-04-07 22:15:08 +00:00
case '+':
1986-04-08 18:15:46 +00:00
break;
1986-10-06 20:36:30 +00:00
*/
1986-07-08 14:59:02 +00:00
1986-04-07 22:15:08 +00:00
case '-':
1987-11-26 14:15:24 +00:00
if (right->nd_INT == min_int[(int)(right->nd_type->tp_size)])
overflow(expp);
1987-07-21 13:54:33 +00:00
expp->nd_INT = -right->nd_INT;
1986-04-07 22:15:08 +00:00
break;
1986-07-08 14:59:02 +00:00
1986-04-07 22:15:08 +00:00
case NOT:
1986-04-22 22:36:16 +00:00
case '~':
1987-07-21 13:54:33 +00:00
expp->nd_INT = !right->nd_INT;
1986-04-07 22:15:08 +00:00
break;
1986-07-08 14:59:02 +00:00
1986-04-07 22:15:08 +00:00
default:
1986-05-28 18:36:51 +00:00
crash("(cstunary)");
1986-04-07 22:15:08 +00:00
}
1986-07-08 14:59:02 +00:00
1986-04-08 18:15:46 +00:00
expp->nd_class = Value;
1987-07-21 13:54:33 +00:00
expp->nd_symb = right->nd_symb;
1986-05-01 19:06:53 +00:00
CutSize(expp);
1987-07-21 13:54:33 +00:00
FreeNode(right);
1986-04-08 18:15:46 +00:00
expp->nd_right = 0;
1986-04-07 22:15:08 +00:00
}
1986-04-07 17:40:38 +00:00
STATIC
1987-11-26 14:15:24 +00:00
divide(pdiv, prem)
arith *pdiv, *prem;
{
1987-11-26 14:15:24 +00:00
/* Unsigned divide *pdiv by *prem, and store result in *pdiv,
remainder in *prem
*/
register arith o1 = *pdiv;
register arith o2 = *prem;
1987-11-26 14:15:24 +00:00
/* 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 */
if (! (o1 >= 0 || o1 < o2)) {
/* this is the unsigned test
o1 < o2 for o2 > max_long
*/
*prem = o2 - o1;
*pdiv = 1;
}
1987-11-26 14:15:24 +00:00
else {
*pdiv = 0;
}
}
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;
*pdiv = 2*hdiv;
*prem = rem;
if (rem < 0 || rem >= o2) {
/* that is the unsigned compare
rem >= o2 for o2 <= max_long
*/
1987-11-26 14:15:24 +00:00
*pdiv += 1;
*prem -= o2;
}
}
1987-11-26 14:15:24 +00:00
}
cstibin(expp)
register t_node *expp;
{
/* The binary operation in "expp" is performed on the constant
expressions below it, and the result restored in expp.
This version is for INTEGER expressions.
*/
arith o1 = expp->nd_left->nd_INT;
arith o2 = expp->nd_right->nd_INT;
register int sz = expp->nd_type->tp_size;
assert(expp->nd_class == Oper);
assert(expp->nd_left->nd_class == Value);
assert(expp->nd_right->nd_class == Value);
switch (expp->nd_symb) {
case '*':
if (o1 == 0 || o2 == 0) {
o1 = 0;
break;
}
if ((o1 > 0 && o2 > 0) || (o1 < 0 && o2 < 0)) {
if (o1 == min_int[sz] ||
o2 == min_int[sz] ||
max_int[sz] / ar_abs(o1) < ar_abs(o2)) overflow(expp);
}
else if (o1 > 0) {
if (min_int[sz] / o1 > o2) overflow(expp);
}
else if (min_int[sz] / o2 > o1) overflow(expp);
o1 *= o2;
break;
case DIV:
if (o2 == 0) {
node_error(expp, "division by 0");
return;
}
o1 /= o2; /* ??? */
break;
case MOD:
if (o2 == 0) {
node_error(expp, "modulo by 0");
return;
}
o1 %= o2; /* ??? */
break;
case '+':
if (o1 > 0 && o2 > 0) {
if (max_int[sz] - o1 < o2) overflow(expp);
}
else if (o1 < 0 && o2 < 0) {
if (min_int[sz] - o1 > o2) overflow(expp);
}
o1 += o2;
break;
case '-':
if (o1 >= 0 && o2 < 0) {
if (max_int[sz] + o2 < o1) overflow(expp);
}
else if (o1 < 0 && o2 >= 0) {
if (min_int[sz] + o2 > o1) overflow(expp);
}
o1 -= o2;
break;
case '<':
{ arith tmp = o1;
o1 = o2;
o2 = tmp;
}
/* Fall through */
case '>':
o1 = (o1 > o2);
break;
case LESSEQUAL:
{ arith tmp = o1;
o1 = o2;
o2 = tmp;
}
/* Fall through */
case GREATEREQUAL:
o1 = chk_bounds(o2, o1, T_INTEGER);
break;
case '=':
o1 = (o1 == o2);
break;
case '#':
o1 = (o1 != o2);
break;
default:
crash("(cstibin)");
}
1987-11-26 14:15:24 +00:00
expp->nd_class = Value;
expp->nd_token = expp->nd_right->nd_token;
expp->nd_INT = o1;
CutSize(expp);
FreeNode(expp->nd_left);
FreeNode(expp->nd_right);
expp->nd_left = expp->nd_right = 0;
}
1987-11-26 14:15:24 +00:00
cstubin(expp)
register t_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
*/
arith o1 = expp->nd_left->nd_INT;
arith o2 = expp->nd_right->nd_INT;
1987-11-26 14:15:24 +00:00
register int sz = expp->nd_type->tp_size;
arith tmp1, tmp2;
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 '*':
1987-11-26 14:15:24 +00:00
if (o1 == 0 || o2 == 0) {
o1 = 0;
break;
}
tmp1 = full_mask[sz];
tmp2 = o2;
divide(&tmp1, &tmp2);
if (! chk_bounds(o1, tmp1, T_CARDINAL)) overflow(expp);
1986-04-07 17:40:38 +00:00
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
}
1987-11-26 14:15:24 +00:00
divide(&o1, &o2);
1986-04-07 17:40:38 +00:00
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
}
1987-11-26 14:15:24 +00:00
divide(&o1, &o2);
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 '+':
1987-11-26 14:15:24 +00:00
if (! chk_bounds(o2, full_mask[sz] - o1, T_CARDINAL)) {
overflow(expp);
}
1986-04-07 17:40:38 +00:00
o1 += o2;
break;
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '-':
1987-11-26 14:15:24 +00:00
if (! chk_bounds(o2, o1, T_CARDINAL)) {
if (expp->nd_type->tp_fund == T_INTORCARD) {
expp->nd_type = int_type;
if (! chk_bounds(min_int[sz], o1 - o2, T_CARDINAL)) {
overflow();
}
}
else overflow();
1986-06-06 02:22:09 +00:00
}
1987-11-26 14:15:24 +00:00
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 '<':
1986-07-08 14:59:02 +00:00
{ arith tmp = o1;
o1 = o2;
o2 = tmp;
1986-04-07 17:40:38 +00:00
}
1986-07-08 14:59:02 +00:00
/* Fall through */
1986-05-01 19:06:53 +00:00
1986-04-07 17:40:38 +00:00
case '>':
1987-11-26 14:15:24 +00:00
o1 = ! chk_bounds(o1, o2, T_CARDINAL);
1986-04-07 17:40:38 +00:00
break;
1986-07-08 14:59:02 +00:00
1986-04-07 22:15:08 +00:00
case LESSEQUAL:
1986-07-08 14:59:02 +00:00
{ arith tmp = o1;
o1 = o2;
o2 = tmp;
1986-04-07 17:40:38 +00:00
}
1986-07-08 14:59:02 +00:00
/* Fall through */
1986-04-07 22:15:08 +00:00
case GREATEREQUAL:
1987-11-26 14:15:24 +00:00
o1 = chk_bounds(o2, o1, T_CARDINAL);
1986-04-07 17:40:38 +00:00
break;
1986-07-08 14:59:02 +00:00
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-07-08 14:59:02 +00:00
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-07-08 14:59:02 +00:00
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-07-08 14:59:02 +00:00
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-07-08 14:59:02 +00:00
1986-04-07 22:15:08 +00:00
default:
1987-11-26 14:15:24 +00:00
crash("(cstubin)");
1986-04-07 17:40:38 +00:00
}
1986-07-08 14:59:02 +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 t_node *expp;
1986-04-08 18:15:46 +00:00
{
extern arith *MkSet();
1986-07-08 14:59:02 +00:00
register arith *set1, *set2;
register arith *resultset;
1987-07-21 13:54:33 +00:00
register unsigned int setsize;
register int j;
1986-04-09 18:14:49 +00:00
assert(expp->nd_right->nd_class == Set);
assert(expp->nd_symb == IN || expp->nd_left->nd_class == Set);
1986-04-09 18:14:49 +00:00
set2 = expp->nd_right->nd_set;
1987-10-20 13:32:18 +00:00
setsize = (unsigned) (expp->nd_right->nd_type->tp_size) / (unsigned) word_size;
1986-04-09 18:14:49 +00:00
if (expp->nd_symb == IN) {
1987-07-21 13:54:33 +00:00
unsigned i;
1986-04-09 18:14:49 +00:00
assert(expp->nd_left->nd_class == Value);
1986-05-28 18:36:51 +00:00
1987-10-28 16:03:56 +00:00
expp->nd_left->nd_INT -= expp->nd_right->nd_type->set_low;
1986-04-09 18:14:49 +00:00
i = expp->nd_left->nd_INT;
1986-08-26 14:33:24 +00:00
expp->nd_class = Value;
1987-07-21 13:54:33 +00:00
expp->nd_INT = (expp->nd_left->nd_INT >= 0 &&
expp->nd_left->nd_INT < setsize * wrd_bits &&
1986-04-09 18:14:49 +00:00
(set2[i / wrd_bits] & (1 << (i % wrd_bits))));
FreeSet(set2);
expp->nd_symb = INTEGER;
FreeNode(expp->nd_left);
FreeNode(expp->nd_right);
expp->nd_left = expp->nd_right = 0;
return;
1986-04-09 18:14:49 +00:00
}
1986-07-08 14:59:02 +00:00
set1 = expp->nd_left->nd_set;
switch(expp->nd_symb) {
case '+': /* Set union */
case '-': /* Set difference */
case '*': /* Set intersection */
case '/': /* Symmetric set difference */
expp->nd_set = resultset = MkSet(setsize * (unsigned) word_size);
for (j = 0; j < setsize; j++) {
switch(expp->nd_symb) {
case '+':
*resultset = *set1++ | *set2++;
break;
case '-':
*resultset = *set1++ & ~*set2++;
break;
case '*':
*resultset = *set1++ & *set2++;
break;
case '/':
*resultset = *set1++ ^ *set2++;
1986-04-09 18:14:49 +00:00
break;
}
resultset++;
1986-05-28 18:36:51 +00:00
}
1986-04-09 18:14:49 +00:00
expp->nd_class = Set;
break;
1987-05-21 09:37:28 +00:00
case GREATEREQUAL:
case LESSEQUAL:
case '=':
case '#':
/* Constant set comparisons
*/
for (j = 0; j < setsize; j++) {
switch(expp->nd_symb) {
case GREATEREQUAL:
if ((*set1 | *set2++) != *set1) break;
set1++;
continue;
case LESSEQUAL:
if ((*set2 | *set1++) != *set2) break;
set2++;
continue;
case '=':
case '#':
if (*set1++ != *set2++) break;
continue;
}
break;
}
if (j < setsize) {
expp->nd_INT = expp->nd_symb == '#';
}
else {
expp->nd_INT = expp->nd_symb != '#';
}
expp->nd_class = Value;
expp->nd_symb = INTEGER;
break;
default:
crash("(cstset)");
1986-04-08 18:15:46 +00:00
}
FreeSet(expp->nd_left->nd_set);
FreeSet(expp->nd_right->nd_set);
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 t_node *expp;
1986-04-11 11:57:19 +00:00
{
/* a standard procedure call is found that can be evaluated
compile time, so do so.
*/
register t_node *expr;
register t_type *tp;
1986-04-11 11:57:19 +00:00
assert(expp->nd_class == Call);
1986-07-08 14:59:02 +00:00
expr = expp->nd_right->nd_left;
expp->nd_right->nd_left = 0;
FreeNode(expp->nd_right);
tp = expr->nd_type;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
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:
1987-11-26 14:15:24 +00:00
if (expr->nd_INT < 0) {
if (expr->nd_INT <= min_int[(int)(tp->tp_size)]) {
overflow(expr);
}
expp->nd_INT = - expr->nd_INT;
}
1986-04-11 11:57:19 +00:00
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;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
case S_CAP:
if (expr->nd_INT >= 'a' && expr->nd_INT <= 'z') {
expr->nd_INT = expr->nd_INT + ('A' - 'a');
1986-04-11 11:57:19 +00:00
}
expp->nd_INT = expr->nd_INT;
break;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
case S_MAX:
if (tp->tp_fund == T_INTEGER) {
1987-11-26 14:15:24 +00:00
expp->nd_INT = max_int[(int)(tp->tp_size)];
1986-04-11 11:57:19 +00:00
}
else if (tp == card_type) {
expp->nd_INT = full_mask[(int)(int_size)];
1986-04-11 11:57:19 +00:00
}
else if (tp->tp_fund == T_SUBRANGE) {
expp->nd_INT = tp->sub_ub;
1986-04-11 11:57:19 +00:00
}
else expp->nd_INT = tp->enm_ncst - 1;
1986-04-11 11:57:19 +00:00
break;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
case S_MIN:
if (tp->tp_fund == T_INTEGER) {
1987-11-26 14:15:24 +00:00
expp->nd_INT = min_int[(int)(tp->tp_size)];
1986-04-11 11:57:19 +00:00
}
else if (tp->tp_fund == T_SUBRANGE) {
expp->nd_INT = tp->sub_lb;
1986-04-11 11:57:19 +00:00
}
else expp->nd_INT = 0;
break;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
case S_ODD:
expp->nd_INT = (expr->nd_INT & 1);
break;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
case S_SIZE:
expp->nd_INT = tp->tp_size;
1986-04-11 11:57:19 +00:00
break;
1986-07-08 14:59:02 +00:00
1986-04-11 11:57:19 +00:00
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)
register t_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.
*/
register t_type *tp = BaseType(expr->nd_type);
1986-04-07 17:40:38 +00:00
1986-04-07 22:15:08 +00:00
assert(expr->nd_class == Value);
1987-11-26 14:15:24 +00:00
if (tp->tp_fund != T_INTEGER) {
expr->nd_INT &= full_mask[tp->tp_size];
1986-04-07 17:40:38 +00:00
}
else {
1987-11-26 14:15:24 +00:00
int nbits = (int) (mach_long_size - tp->tp_size) * 8;
1986-04-07 17:40:38 +00:00
1987-11-26 14:15:24 +00:00
expr->nd_INT <<= nbits;
expr->nd_INT >>= nbits;
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
{
1986-07-08 14:59:02 +00:00
register int i = 0;
register arith bt = (arith)0;
1986-04-07 17:40:38 +00:00
while (!(bt < 0)) {
bt = (bt << 8) + 0377, i++;
if (i == MAXSIZE)
fatal("array full_mask too small for this machine");
full_mask[i] = bt;
1987-11-26 14:15:24 +00:00
max_int[i] = bt & ~(1L << ((i << 3) - 1));
min_int[i] = - max_int[i];
if (! options['s']) min_int[i]--;
1986-04-07 17:40:38 +00:00
}
mach_long_size = i;
mach_long_sign = 1L << (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
1987-07-21 13:54:33 +00:00
wrd_bits = 8 * (unsigned) word_size;
1986-04-07 17:40:38 +00:00
}