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
|
|
|
|
|
|
|
static char *RcsId = "$Header$";
|
|
|
|
|
|
|
|
#include <em_arith.h>
|
|
|
|
#include <em_label.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "def_sizes.h"
|
|
|
|
#include "idf.h"
|
|
|
|
#include "type.h"
|
|
|
|
#include "LLlex.h"
|
|
|
|
#include "node.h"
|
|
|
|
#include "Lpars.h"
|
|
|
|
|
|
|
|
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-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;
|
|
|
|
break;
|
|
|
|
case NOT:
|
|
|
|
o1 = !o1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
}
|
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;
|
|
|
|
cut_size(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);
|
|
|
|
if (expp->nd_right->nd_type->tp_fund == SET) {
|
|
|
|
cstset(expp);
|
1986-04-07 22:15:08 +00:00
|
|
|
return;
|
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-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-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;
|
|
|
|
case '+':
|
|
|
|
o1 += o2;
|
|
|
|
break;
|
|
|
|
case '-':
|
1986-04-07 22:15:08 +00:00
|
|
|
o1 -= o2;
|
1986-04-07 17:40:38 +00:00
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
if (uns) {
|
|
|
|
o1 = (o1 & mach_long_sign ?
|
|
|
|
(o2 & mach_long_sign ? o1 < o2 : 0) :
|
|
|
|
(o2 & mach_long_sign ? 1 : o1 < o2)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
o1 = o1 < o2;
|
|
|
|
break;
|
|
|
|
case '>':
|
|
|
|
if (uns) {
|
|
|
|
o1 = (o1 & mach_long_sign ?
|
|
|
|
(o2 & mach_long_sign ? o1 > o2 : 1) :
|
|
|
|
(o2 & mach_long_sign ? 0 : o1 > o2)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
o1 = o1 > o2;
|
|
|
|
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
|
|
|
|
o1 = o1 <= o2;
|
|
|
|
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
|
|
|
|
o1 = o1 >= o2;
|
|
|
|
break;
|
1986-04-07 22:15:08 +00:00
|
|
|
case '=':
|
1986-04-07 17:40:38 +00:00
|
|
|
o1 = o1 == o2;
|
|
|
|
break;
|
1986-04-07 22:15:08 +00:00
|
|
|
case '#':
|
1986-04-07 17:40:38 +00:00
|
|
|
o1 = o1 != o2;
|
|
|
|
break;
|
1986-04-07 22:15:08 +00:00
|
|
|
case AND:
|
|
|
|
o1 = o1 && o2;
|
1986-04-07 17:40:38 +00:00
|
|
|
break;
|
1986-04-07 22:15:08 +00:00
|
|
|
case OR:
|
|
|
|
o1 = o1 || o2;
|
1986-04-07 17:40:38 +00:00
|
|
|
break;
|
1986-04-07 22:15:08 +00:00
|
|
|
default:
|
|
|
|
assert(0);
|
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-04-07 22:15:08 +00:00
|
|
|
expp->nd_INT = o1;
|
|
|
|
cut_size(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;
|
|
|
|
{
|
|
|
|
switch(expp->nd_symb) {
|
|
|
|
case IN:
|
|
|
|
case '+':
|
|
|
|
case '-':
|
|
|
|
case '*':
|
|
|
|
case '/':
|
|
|
|
case GREATEREQUAL:
|
|
|
|
case LESSEQUAL:
|
|
|
|
case '=':
|
|
|
|
case '#':
|
|
|
|
/* ??? */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
}
|
1986-04-07 17:40:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cut_size(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;
|
|
|
|
int uns = expr->nd_type == card_type || expr->nd_type == intorcard_type;
|
|
|
|
int size = expr->nd_type->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-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
|
|
|
}
|
|
|
|
|
|
|
|
init_cst()
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
if (sizeof(long) < mach_long_size)
|
|
|
|
fatal("sizeof (long) insufficient on this machine");
|
|
|
|
|
|
|
|
max_int = full_mask[int_size] & ~(1 << (int_size * 8 - 1));
|
|
|
|
max_longint = full_mask[lint_size] & ~(1 << (lint_size * 8 - 1));
|
|
|
|
max_unsigned = full_mask[int_size];
|
|
|
|
}
|