1989-02-07 11:04:05 +00:00
|
|
|
/*
|
|
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
|
|
*/
|
1994-06-27 08:03:14 +00:00
|
|
|
/* $Id$ */
|
1989-02-07 11:04:05 +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 */
|
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
#include <assert.h>
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "cstoper.h"
|
2013-05-12 19:45:55 +00:00
|
|
|
#include "parameters.h"
|
1989-09-19 16:13:23 +00:00
|
|
|
#include <flt_arith.h>
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "arith.h"
|
|
|
|
#include "type.h"
|
|
|
|
#include "label.h"
|
|
|
|
#include "expr.h"
|
|
|
|
#include "sizes.h"
|
|
|
|
#include "Lpars.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "error.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
|
2012-09-07 19:53:13 +00:00
|
|
|
/* full_mask[1] == 0XFF, full_mask[2] == 0XFFFF, .. */
|
|
|
|
arith full_mask[MAXSIZE + 1];
|
1989-11-08 16:52:34 +00:00
|
|
|
#ifndef NOCROSS
|
1989-02-07 11:04:05 +00:00
|
|
|
arith max_int; /* maximum integer on target machine */
|
|
|
|
arith max_unsigned; /* maximum unsigned on target machine */
|
1989-11-08 16:52:34 +00:00
|
|
|
#endif /* NOCROSS */
|
1989-09-19 16:13:23 +00:00
|
|
|
extern int ResultKnown;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
void cstbin(register struct expr **expp, int oper, register struct expr *expr)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
/* The operation oper is performed on the constant
|
|
|
|
expressions *expp(ld) and expr(ct), and the result restored in
|
|
|
|
*expp.
|
|
|
|
*/
|
|
|
|
register arith o1 = (*expp)->VL_VALUE;
|
|
|
|
register arith o2 = expr->VL_VALUE;
|
|
|
|
int uns = (*expp)->ex_type->tp_unsigned;
|
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(is_ld_cst(*expp) && is_cp_cst(expr));
|
1989-02-07 11:04:05 +00:00
|
|
|
switch (oper) {
|
|
|
|
case '*':
|
|
|
|
o1 *= o2;
|
|
|
|
break;
|
|
|
|
case '/':
|
|
|
|
if (o2 == 0) {
|
1989-09-19 16:13:23 +00:00
|
|
|
if (!ResultKnown)
|
|
|
|
expr_error(expr, "division by 0");
|
|
|
|
else
|
|
|
|
expr_warning(expr, "division by 0");
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-10-29 21:01:29 +00:00
|
|
|
if (uns)
|
|
|
|
o1 /= (unsigned arith) o2;
|
1989-02-07 11:04:05 +00:00
|
|
|
else
|
|
|
|
o1 /= o2;
|
|
|
|
break;
|
|
|
|
case '%':
|
|
|
|
if (o2 == 0) {
|
1989-09-19 16:13:23 +00:00
|
|
|
if (!ResultKnown)
|
|
|
|
expr_error(expr, "modulo by 0");
|
|
|
|
else
|
|
|
|
expr_warning(expr, "modulo by 0");
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-10-29 21:01:29 +00:00
|
|
|
if (uns)
|
|
|
|
o1 %= (unsigned arith) o2;
|
1989-02-07 11:04:05 +00:00
|
|
|
else
|
|
|
|
o1 %= o2;
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
o1 += o2;
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
o1 -= o2;
|
|
|
|
break;
|
|
|
|
case LEFT:
|
|
|
|
o1 <<= o2;
|
|
|
|
break;
|
|
|
|
case RIGHT:
|
|
|
|
if (o2 == 0)
|
|
|
|
break;
|
|
|
|
if (uns) {
|
1989-10-19 14:53:25 +00:00
|
|
|
o1 = (o1 >> 1) & ~arith_sign;
|
|
|
|
o1 >>= (o2 - 1);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1989-10-19 14:53:25 +00:00
|
|
|
else o1 >>= o2;
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
{
|
|
|
|
arith tmp = o1;
|
|
|
|
|
|
|
|
o1 = o2;
|
|
|
|
o2 = tmp;
|
|
|
|
}
|
|
|
|
/* Fall through */
|
|
|
|
case '>':
|
2017-10-29 21:01:29 +00:00
|
|
|
if (uns)
|
|
|
|
o1 = (unsigned arith) o1 > (unsigned arith) o2;
|
1989-02-07 11:04:05 +00:00
|
|
|
else
|
|
|
|
o1 = o1 > o2;
|
|
|
|
break;
|
|
|
|
case LESSEQ:
|
|
|
|
{
|
|
|
|
arith tmp = o1;
|
|
|
|
|
|
|
|
o1 = o2;
|
|
|
|
o2 = tmp;
|
|
|
|
}
|
|
|
|
/* Fall through */
|
|
|
|
case GREATEREQ:
|
2017-10-29 21:01:29 +00:00
|
|
|
if (uns)
|
|
|
|
o1 = (unsigned arith) o1 >= (unsigned arith) o2;
|
1989-02-07 11:04:05 +00:00
|
|
|
else
|
|
|
|
o1 = o1 >= o2;
|
|
|
|
break;
|
|
|
|
case EQUAL:
|
|
|
|
o1 = o1 == o2;
|
|
|
|
break;
|
|
|
|
case NOTEQUAL:
|
|
|
|
o1 = o1 != o2;
|
|
|
|
break;
|
|
|
|
case '&':
|
|
|
|
o1 &= o2;
|
|
|
|
break;
|
|
|
|
case '|':
|
|
|
|
o1 |= o2;
|
|
|
|
break;
|
|
|
|
case '^':
|
|
|
|
o1 ^= o2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
(*expp)->VL_VALUE = o1;
|
|
|
|
cut_size(*expp);
|
|
|
|
(*expp)->ex_flags |= expr->ex_flags;
|
|
|
|
(*expp)->ex_flags &= ~EX_PARENS;
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
void cut_size(register struct expr *expr)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
/* The constant value of the expression expr is made to
|
|
|
|
conform to the size of the type of the expression.
|
|
|
|
*/
|
|
|
|
register arith o1 = expr->VL_VALUE;
|
|
|
|
int uns = expr->ex_type->tp_unsigned;
|
|
|
|
int size = (int) expr->ex_type->tp_size;
|
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(expr->ex_class == Value);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (expr->ex_type->tp_fund == POINTER) {
|
|
|
|
/* why warn on "ptr-3" ?
|
|
|
|
This quick hack fixes it
|
|
|
|
*/
|
|
|
|
uns = 0;
|
|
|
|
}
|
|
|
|
if (uns) {
|
|
|
|
if (o1 & ~full_mask[size])
|
1989-09-19 16:13:23 +00:00
|
|
|
if (!ResultKnown)
|
1989-02-07 11:04:05 +00:00
|
|
|
expr_warning(expr,
|
|
|
|
"overflow in unsigned constant expression");
|
|
|
|
o1 &= full_mask[size];
|
|
|
|
}
|
|
|
|
else {
|
1989-09-19 16:13:23 +00:00
|
|
|
int nbits = (int) (arith_size - size) * 8;
|
|
|
|
arith remainder = o1 & ~full_mask[size];
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
if (remainder != 0 && remainder != ~full_mask[size])
|
1989-09-19 16:13:23 +00:00
|
|
|
if (!ResultKnown)
|
|
|
|
expr_warning(expr,"overflow in constant expression");
|
1989-10-19 14:53:25 +00:00
|
|
|
o1 = (o1 << nbits) >> nbits; /* ??? */
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
expr->VL_VALUE = o1;
|
|
|
|
}
|
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
void init_cst(void)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
register int i = 0;
|
|
|
|
register arith bt = (arith)0;
|
|
|
|
|
Begin to add `long long` to C compiler for linux386.
Add long long type, but without literals; you can't say '123LL' yet.
You can try constant operations, like `(long long)123 + 1`, but the
compiler's `arith` type might not be wide enough. Conversions,
shifts, and some other operations don't work in i386 ncg; I am using a
union instead of conversions:
union q {
long long ll;
unsigned long long ull;
int i[2];
};
Hack plat/linux386/descr to enable long long (size 8, alignment 4)
only for this platform. The default for other platforms is to disable
long long (size -1).
In lang/cem/cemcom.ansi,
- BigPars, SmallPars: Add default size, alignment of long long.
- align.h: Add lnglng_align.
- arith.c: Convert arithmetic operands to long long or unsigned long
long when necessary; avoid conversion from long long to long.
Allow long long as an arithmetic, integral, or logical operand.
- ch3.c: Handle long long like int and long when erroneously applying
a selector, like `long long ll; ll.member` or `ll->member`. Add
long long to integral and arithmetic types.
- code.c: Add long long to type stabs for debugging.
- conversion.c: Add long long to integral conversions.
- cstoper.c: Write masks up to full_mask[8]. Add FIXME comment.
- declar.g: Parse `long long` in code.
- decspecs.c: Understand long long in type declarations.
- eval.c: Add long long to operations, to generate code like `adi 8`.
Don't use `ldc` with constant over 4 bytes.
- ival.g: Allow long long in initializations.
- main.c: Set lnglng_type and related values.
- options.c: Add option like `-Vq8.4` to set long long to size 8,
alignment 4. I chose 'q', because Perl's pack and Ruby's
Array#pack use 'q' for 64-bit or long long values; it might be a
reference to BSD's old quad_t alias for long long.
- sizes.h: Add lnglng_size.
- stab.c: Allow long long when writing the type stab for debugging.
Switch from calculating the ranges to hardcoding them in strings;
add 8-byte ranges as a special case. This also hardcodes the
unsigned 4-byte range as "0;-1". Before it was either "0;-1" or
"0;4294967295", depending on sizeof(long) in the compiler.
- struct.c: Try long long bitfield, but it will probably give the
error, "bit field type long long does not fit in a word".
- switch.c: Update comment.
- tokenname.c: Define LNGLNG (long long) like LNGDBL (long double).
- type.c, type.str: Add lnglng_type and ulnglng_type. Add function
no_long_long() to check if long long is disabled.
2019-09-02 15:24:44 +00:00
|
|
|
/* FIXME arith is insufficient for long long. We ignore
|
|
|
|
this problem and write masks up to full_mask[8], but
|
|
|
|
masks are wrong after bt < 0.
|
|
|
|
*/
|
|
|
|
while (!(bt < 0) || i < 8) {
|
1989-02-07 11:04:05 +00:00
|
|
|
bt = (bt << 8) + 0377, i++;
|
2012-09-07 19:53:13 +00:00
|
|
|
if (i > MAXSIZE)
|
1989-02-07 11:04:05 +00:00
|
|
|
fatal("array full_mask too small for this machine");
|
|
|
|
full_mask[i] = bt;
|
|
|
|
}
|
1989-09-19 16:13:23 +00:00
|
|
|
if ((int)long_size > arith_size)
|
|
|
|
fatal("sizeof (arith) insufficient on this machine");
|
1989-11-08 16:52:34 +00:00
|
|
|
#ifndef NOCROSS
|
1989-02-07 11:04:05 +00:00
|
|
|
max_int = full_mask[(int)int_size] & ~(1L << ((int)int_size * 8 - 1));
|
|
|
|
max_unsigned = full_mask[(int)int_size];
|
1989-11-08 16:52:34 +00:00
|
|
|
#endif /* NOCROSS */
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|