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
|
|
|
/* EXPRESSION-CODE GENERATOR */
|
|
|
|
|
2013-05-12 19:45:55 +00:00
|
|
|
#include "parameters.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#ifndef LINT
|
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
#include <assert.h>
|
2017-11-10 22:31:11 +00:00
|
|
|
#include <ack_string.h>
|
1989-02-07 11:04:05 +00:00
|
|
|
#include <em.h>
|
|
|
|
#include <em_reg.h>
|
1989-09-19 16:13:23 +00:00
|
|
|
#include <alloc.h>
|
|
|
|
#include <flt_arith.h>
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "interface.h"
|
|
|
|
#include "eval.h"
|
2013-05-12 19:45:55 +00:00
|
|
|
#include "idf.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "arith.h"
|
|
|
|
#include "type.h"
|
|
|
|
#include "label.h"
|
|
|
|
#include "code.h"
|
|
|
|
#include "def.h"
|
|
|
|
#include "expr.h"
|
|
|
|
#include "sizes.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "field.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "Lpars.h"
|
|
|
|
#include "level.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "conversion.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "stack.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "struct.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "align.h"
|
|
|
|
#include "mes.h"
|
|
|
|
#include "atw.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "ch3.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "blocks.h"
|
|
|
|
#include "dataflow.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
#include "specials.h"
|
2019-02-18 16:42:15 +00:00
|
|
|
#include "error.h"
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
#define CRASH() crash("EVAL: CRASH at line %u", __LINE__)
|
|
|
|
|
|
|
|
char *symbol2str();
|
|
|
|
arith NewLocal(); /* util.c */
|
|
|
|
#define LocalPtrVar() NewLocal(pointer_size, pointer_align, reg_pointer, REGISTER)
|
1991-01-15 12:00:24 +00:00
|
|
|
extern int err_occurred; /* error.c */
|
1989-02-07 11:04:05 +00:00
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
|
|
|
|
/* Forward internal declarations */
|
|
|
|
static void operands(register struct expr *, int);
|
|
|
|
static void ptr_add(arith size);
|
|
|
|
static void truthvalue(int relop);
|
|
|
|
static void compare(int relop, label lbl);
|
2016-11-10 21:04:18 +00:00
|
|
|
|
1989-02-07 11:04:05 +00:00
|
|
|
/* EVAL() is the main expression-tree evaluator, which turns
|
2013-05-12 19:45:55 +00:00
|
|
|
any legal expression tree into EM code. parameters.h:
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
struct expr *expr
|
|
|
|
pointer to root of the expression tree to be evaluated
|
|
|
|
|
|
|
|
int val
|
|
|
|
indicates whether the resulting expression is to be
|
|
|
|
dereferenced (if val == RVAL and expr->ex_lvalue == 1)
|
|
|
|
or not (val == LVAL). The latter case indicates that
|
|
|
|
the resulting expression is an lvalue expression which
|
|
|
|
should not be dereferenced by EVAL
|
|
|
|
|
|
|
|
int code
|
|
|
|
indicates whether the expression tree must be turned
|
|
|
|
into EM code or not. E.g. the expression statement "12;"
|
|
|
|
delivers the expression "12" to EVAL while this should
|
|
|
|
not result in any EM code
|
|
|
|
|
|
|
|
label false_label, label true_label
|
|
|
|
if the expression is a logical or relational expression
|
|
|
|
and if the loop of the program depends on the resulting
|
|
|
|
value then EVAL generates jumps to the specified program
|
|
|
|
labels, in case they are specified (i.e. are non-zero)
|
|
|
|
*/
|
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
void EVAL(register struct expr *expr, int val, int code, label true_label, label false_label)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
1990-04-02 15:57:51 +00:00
|
|
|
int vol = (code != TRUE && recurqual(expr->ex_type, TQ_VOLATILE));
|
1991-01-15 12:00:24 +00:00
|
|
|
register int gencode = code == TRUE;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
1991-01-15 12:00:24 +00:00
|
|
|
if (err_occurred) return;
|
1989-02-07 11:04:05 +00:00
|
|
|
switch (expr->ex_class) {
|
|
|
|
case Value: /* just a simple value */
|
1990-03-29 10:41:46 +00:00
|
|
|
if (gencode) {
|
|
|
|
if (true_label) {
|
|
|
|
/* can only result from ','-expressions with
|
|
|
|
constant right-hand sides ???
|
|
|
|
*/
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(is_cp_cst(expr));
|
1990-03-29 10:41:46 +00:00
|
|
|
C_bra(expr->VL_VALUE == 0 ? false_label : true_label);
|
|
|
|
}
|
|
|
|
else load_val(expr, val);
|
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
else if (vol) {
|
|
|
|
load_val(expr, val);
|
|
|
|
C_asp(ATW(expr->ex_type->tp_size));
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case String: /* a string constant */
|
|
|
|
if (gencode) {
|
|
|
|
string2pointer(expr);
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
C_lae_dlb(expr->VL_LBL, (arith)expr->VL_VALUE);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Float: /* a floating constant */
|
|
|
|
if (gencode) {
|
|
|
|
label datlab = data_label();
|
1991-01-15 18:21:16 +00:00
|
|
|
char buf[FLT_STRLEN];
|
1989-02-07 11:04:05 +00:00
|
|
|
|
1990-02-28 14:51:35 +00:00
|
|
|
fp_used = 1;
|
1991-01-15 18:21:16 +00:00
|
|
|
flt_flt2str(&(expr->FL_ARITH), buf, FLT_STRLEN);
|
1989-02-07 11:04:05 +00:00
|
|
|
C_df_dlb(datlab);
|
1991-01-15 18:21:16 +00:00
|
|
|
C_rom_fcon(buf, expr->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
C_lae_dlb(datlab, (arith)0);
|
|
|
|
C_loi(expr->ex_type->tp_size);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Oper: /* compound expression */
|
|
|
|
{
|
|
|
|
int oper = expr->OP_OPER;
|
|
|
|
register struct expr *left = expr->OP_LEFT;
|
|
|
|
register struct expr *right = expr->OP_RIGHT;
|
|
|
|
register struct type *tp = expr->OP_TYPE;
|
|
|
|
|
|
|
|
switch (oper) {
|
|
|
|
case '+':
|
|
|
|
/* We have the following possibilities :
|
|
|
|
int + int, pointer + int, pointer + long,
|
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
|
|
|
long + long, long long + long long,
|
|
|
|
double + double
|
1989-02-07 11:04:05 +00:00
|
|
|
*/
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode) {
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_adu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_adi(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case POINTER:
|
1991-02-01 09:54:21 +00:00
|
|
|
ptr_add(right->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
C_adf(tp->tp_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("bad type +");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
if (left == 0) { /* unary */
|
|
|
|
EVAL(right, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode) {
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
case POINTER:
|
|
|
|
C_ngi(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
C_ngf(tp->tp_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CRASH();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* else binary; we have the following flavours:
|
|
|
|
int - int, pointer - int, pointer - long,
|
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
|
|
|
pointer - pointer, long - long,
|
|
|
|
long long - long long, double - double
|
1989-02-07 11:04:05 +00:00
|
|
|
*/
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (!gencode)
|
|
|
|
break;
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_sbu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_sbi(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case POINTER:
|
|
|
|
if (right->ex_type->tp_fund == POINTER)
|
|
|
|
C_sbs(pointer_size);
|
|
|
|
else {
|
|
|
|
C_ngi(right->ex_type->tp_size);
|
1991-02-01 09:54:21 +00:00
|
|
|
ptr_add(right->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
C_sbf(tp->tp_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("bad type -");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '*':
|
|
|
|
if (left == 0) { /* unary */
|
|
|
|
EVAL(right, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode && right->ex_class == String) {
|
|
|
|
C_loi((arith)1);
|
|
|
|
}
|
1989-10-20 17:08:48 +00:00
|
|
|
break;
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
|
|
|
if (gencode) {
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-10-20 17:08:48 +00:00
|
|
|
case POINTER:
|
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_mlu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_mli(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
C_mlf(tp->tp_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("bad type *");
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '/':
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode)
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
case POINTER:
|
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_dvu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_dvi(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
/*C_dvf(double_size);*/
|
|
|
|
C_dvf(tp->tp_size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("bad type /");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '%':
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
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
|
|
|
assert(tp->tp_fund==INT || tp->tp_fund==LONG ||
|
|
|
|
tp->tp_fund==LNGLNG);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode)
|
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_rmu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_rmi(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case LEFT:
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode)
|
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_slu(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_sli(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case RIGHT:
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode)
|
|
|
|
if (tp->tp_unsigned)
|
|
|
|
C_sru(tp->tp_size);
|
|
|
|
else
|
|
|
|
C_sri(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
case LESSEQ:
|
|
|
|
case '>':
|
|
|
|
case GREATEREQ:
|
|
|
|
case EQUAL:
|
|
|
|
case NOTEQUAL:
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode) {
|
|
|
|
/* The operands have the same type */
|
|
|
|
arith size = left->ex_type->tp_size;
|
|
|
|
|
|
|
|
switch (tp->tp_fund) {
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
if (left->ex_type->tp_unsigned)
|
|
|
|
C_cmu(size);
|
|
|
|
else
|
|
|
|
C_cmi(size);
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
C_cmf(size);
|
|
|
|
break;
|
|
|
|
case POINTER:
|
|
|
|
C_cmp();
|
|
|
|
break;
|
|
|
|
case ENUM:
|
|
|
|
C_cmi(size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CRASH();
|
|
|
|
}
|
|
|
|
if (true_label != 0) {
|
|
|
|
compare(oper, true_label);
|
|
|
|
C_bra(false_label);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
truthvalue(oper);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '&':
|
|
|
|
case '|':
|
|
|
|
case '^':
|
|
|
|
/* both operands should have type int */
|
1989-10-20 17:08:48 +00:00
|
|
|
operands(expr, gencode);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (gencode) {
|
|
|
|
arith size = tp->tp_size;
|
|
|
|
|
|
|
|
if ((int)size < (int)word_size)
|
|
|
|
size = word_size;
|
|
|
|
switch (oper) {
|
|
|
|
case '&':
|
|
|
|
C_and(size);
|
|
|
|
break;
|
|
|
|
case '|':
|
|
|
|
C_ior(size);
|
|
|
|
break;
|
|
|
|
case '^':
|
|
|
|
C_xor(size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
1991-01-15 12:00:24 +00:00
|
|
|
case '=':
|
1989-02-07 11:04:05 +00:00
|
|
|
#ifndef NOBITFIELD
|
|
|
|
if (left->ex_type->tp_fund == FIELD) {
|
|
|
|
eval_field(expr, gencode);
|
|
|
|
break;
|
|
|
|
}
|
1991-12-17 13:12:22 +00:00
|
|
|
#endif /* NOBITFIELD */
|
1991-08-26 14:26:07 +00:00
|
|
|
if (is_struct_or_union(tp->tp_fund) && ! gencode) {
|
|
|
|
EVAL(right, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
copy_block(tp->tp_size, tp->tp_align);
|
|
|
|
break;
|
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(right, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
1989-02-07 13:16:02 +00:00
|
|
|
if (gencode && val == RVAL)
|
1989-02-07 11:04:05 +00:00
|
|
|
C_dup(ATW(tp->tp_size));
|
|
|
|
if (left->ex_class != Value) {
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode && val == LVAL) {
|
1989-02-07 13:16:02 +00:00
|
|
|
arith tmp = LocalPtrVar();
|
|
|
|
C_dup(pointer_size);
|
|
|
|
StoreLocal(tmp, pointer_size);
|
|
|
|
store_block(tp->tp_size, tp->tp_align);
|
|
|
|
LoadLocal(tmp, pointer_size);
|
|
|
|
FreeLocal(tmp);
|
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
else store_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
else {
|
1989-02-07 11:04:05 +00:00
|
|
|
store_val(&(left->EX_VALUE), left->ex_type);
|
1989-02-07 13:16:02 +00:00
|
|
|
if (gencode && val == LVAL) {
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
1989-02-07 13:16:02 +00:00
|
|
|
}
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case PLUSAB:
|
|
|
|
case MINAB:
|
|
|
|
case TIMESAB:
|
|
|
|
case DIVAB:
|
|
|
|
case MODAB:
|
|
|
|
case LEFTAB:
|
|
|
|
case RIGHTAB:
|
|
|
|
case ANDAB:
|
|
|
|
case XORAB:
|
|
|
|
case ORAB:
|
|
|
|
case POSTINCR:
|
|
|
|
case POSTDECR:
|
|
|
|
case PLUSPLUS:
|
|
|
|
case MINMIN:
|
|
|
|
{
|
1989-11-08 16:52:34 +00:00
|
|
|
arith tmp = 0;
|
1989-02-07 11:04:05 +00:00
|
|
|
int compl; /* Complexity of left operand */
|
1990-02-28 12:18:36 +00:00
|
|
|
int right_done = 0;
|
1991-01-15 12:00:24 +00:00
|
|
|
int dupval;
|
1989-02-07 11:04:05 +00:00
|
|
|
#ifndef NOBITFIELD
|
|
|
|
if (left->ex_type->tp_fund == FIELD) {
|
|
|
|
eval_field(expr, gencode);
|
|
|
|
break;
|
|
|
|
}
|
1991-12-17 13:12:22 +00:00
|
|
|
#endif /* NOBITFIELD */
|
1991-01-15 12:00:24 +00:00
|
|
|
if (left->ex_class == Value) {
|
1989-02-07 11:04:05 +00:00
|
|
|
compl = 0; /* Value */
|
|
|
|
}
|
1990-02-28 12:18:36 +00:00
|
|
|
else if (left->ex_depth == 1 &&
|
1989-02-07 11:04:05 +00:00
|
|
|
!(left->ex_flags & EX_SIDEEFFECTS)) {
|
|
|
|
compl = 1;
|
1990-02-28 12:18:36 +00:00
|
|
|
}
|
|
|
|
else compl = 2;
|
|
|
|
|
|
|
|
/* evaluate right-hand side first when possible,
|
|
|
|
but not for POSTINCR or PLUSPLUS, because then
|
|
|
|
we might miss a chance for increment instructions.
|
|
|
|
*/
|
|
|
|
if (compl != 2 &&
|
|
|
|
tp->tp_fund != POINTER &&
|
|
|
|
(oper == PLUSAB || oper == TIMESAB ||
|
|
|
|
oper == ANDAB || oper == XORAB || oper == ORAB)) {
|
|
|
|
right_done = 1;
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(right, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
1990-02-28 12:18:36 +00:00
|
|
|
}
|
|
|
|
if (compl == 0) {
|
|
|
|
load_val(left, RVAL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (compl == 1) {
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(left, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
else {
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
tmp = LocalPtrVar();
|
|
|
|
C_dup(pointer_size);
|
|
|
|
StoreLocal(tmp, pointer_size);
|
|
|
|
C_loi(left->ex_type->tp_size);
|
|
|
|
}
|
|
|
|
if (gencode && (oper == POSTINCR ||
|
|
|
|
oper == POSTDECR))
|
|
|
|
C_dup(ATW(left->ex_type->tp_size));
|
|
|
|
conversion(left->ex_type, tp);
|
|
|
|
if (! right_done) {
|
|
|
|
EVAL(right, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
dupval = gencode && oper != POSTINCR &&
|
|
|
|
oper != POSTDECR;
|
|
|
|
assop(tp, oper);
|
|
|
|
conversion(tp, left->ex_type);
|
|
|
|
if (compl == 0) {
|
|
|
|
store_val(&(left->EX_VALUE),
|
|
|
|
left->ex_type);
|
|
|
|
if (dupval) load_val(left, RVAL);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
else if (compl == 1) {
|
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
C_sti(left->ex_type->tp_size);
|
|
|
|
if (dupval) {
|
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
C_loi(left->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
LoadLocal(tmp, pointer_size);
|
|
|
|
C_sti(left->ex_type->tp_size);
|
|
|
|
if (dupval) {
|
1989-02-07 11:04:05 +00:00
|
|
|
LoadLocal(tmp, pointer_size);
|
1991-01-15 12:00:24 +00:00
|
|
|
C_loi(left->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
FreeLocal(tmp);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '(':
|
|
|
|
{
|
|
|
|
register struct expr *ex;
|
|
|
|
arith ParSize = (arith)0;
|
|
|
|
label setjmp_label = 0;
|
1993-01-28 10:09:51 +00:00
|
|
|
arith retspace = 0;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
if (ISNAME(left)) {
|
|
|
|
if (left->VL_IDF->id_special == SP_SETJMP) {
|
|
|
|
label addr_label = data_label();
|
|
|
|
|
|
|
|
setjmp_label = text_label();
|
|
|
|
C_df_dlb(addr_label);
|
|
|
|
C_rom_ilb(setjmp_label);
|
|
|
|
C_lae_dlb(addr_label, (arith) 0);
|
|
|
|
C_loi(pointer_size);
|
|
|
|
ParSize += pointer_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((ex = right) != NILEXPR) {
|
|
|
|
/* function call with parameters*/
|
|
|
|
while ( ex->ex_class == Oper &&
|
|
|
|
ex->OP_OPER == PARCOMMA
|
|
|
|
) {
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(ex->OP_RIGHT, RVAL, TRUE,
|
1989-02-07 11:04:05 +00:00
|
|
|
NO_LABEL, NO_LABEL);
|
1991-01-15 12:00:24 +00:00
|
|
|
ParSize += ATW(ex->OP_RIGHT->ex_type->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
ex = ex->OP_LEFT;
|
|
|
|
}
|
1991-01-15 12:00:24 +00:00
|
|
|
EVAL(ex, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
1989-02-07 11:04:05 +00:00
|
|
|
ParSize += ATW(ex->ex_type->tp_size);
|
|
|
|
}
|
1993-01-28 11:45:38 +00:00
|
|
|
if (is_struct_or_union(tp->tp_fund)) {
|
|
|
|
retspace = NewLocal(tp->tp_size, tp->tp_align,
|
|
|
|
-1, 0);
|
|
|
|
C_lal(retspace);
|
|
|
|
ParSize += pointer_size;
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
if (ISNAME(left)) {
|
|
|
|
/* e.g., main() { (*((int (*)())0))(); } */
|
|
|
|
C_cal(left->VL_IDF->id_text);
|
|
|
|
if (setjmp_label) {
|
|
|
|
C_df_ilb(setjmp_label);
|
|
|
|
}
|
|
|
|
#ifdef DATAFLOW
|
|
|
|
{ extern char options[];
|
|
|
|
if (options['d'])
|
|
|
|
DfaCallFunction(
|
|
|
|
left->VL_IDF->id_text);
|
|
|
|
}
|
1991-12-17 13:12:22 +00:00
|
|
|
#endif /* DATAFLOW */
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
EVAL(left, LVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
|
|
C_cai();
|
|
|
|
}
|
|
|
|
/* remove parameters from stack */
|
|
|
|
if (ParSize > (arith)0)
|
|
|
|
C_asp(ParSize);
|
|
|
|
if (gencode) {
|
|
|
|
if (is_struct_or_union(tp->tp_fund)) {
|
|
|
|
C_lfr(pointer_size);
|
1993-01-28 10:09:51 +00:00
|
|
|
if (val == RVAL) {
|
1989-02-07 13:16:02 +00:00
|
|
|
load_block(tp->tp_size, (int) word_size);
|
1993-01-28 10:09:51 +00:00
|
|
|
FreeLocal(retspace);
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
C_lfr(ATW(tp->tp_size));
|
|
|
|
}
|
1990-02-28 15:32:07 +00:00
|
|
|
if (tp->tp_fund == FLOAT || tp->tp_fund == DOUBLE
|
|
|
|
|| tp->tp_fund == LNGDBL)
|
|
|
|
fp_used = 1;
|
1989-11-09 12:33:08 +00:00
|
|
|
/* ??? set filename and line number ??? */
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '.':
|
1989-10-20 17:08:48 +00:00
|
|
|
case ARROW:
|
1990-03-01 13:02:39 +00:00
|
|
|
if (tp->tp_fund == FLOAT || tp->tp_fund == DOUBLE
|
|
|
|
|| tp->tp_fund == LNGDBL)
|
|
|
|
fp_used = 1;
|
1989-10-20 17:08:48 +00:00
|
|
|
EVAL(left, oper == '.' ? LVAL : RVAL, gencode,
|
|
|
|
NO_LABEL, NO_LABEL);
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(is_cp_cst(right));
|
1989-02-07 13:16:02 +00:00
|
|
|
if (gencode) {
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
C_adp((arith)right->VL_VALUE);
|
1989-02-07 13:16:02 +00:00
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case ',':
|
|
|
|
EVAL(left, RVAL, FALSE, NO_LABEL, NO_LABEL);
|
1993-01-28 10:09:51 +00:00
|
|
|
EVAL(right, val, gencode, true_label, false_label);
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
case '~':
|
|
|
|
EVAL(right, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode)
|
|
|
|
C_com(tp->tp_size);
|
|
|
|
break;
|
|
|
|
case '?': /* must be followed by ':' */
|
|
|
|
{
|
|
|
|
label l_true = text_label();
|
|
|
|
label l_false = text_label();
|
|
|
|
label l_end = text_label();
|
|
|
|
|
|
|
|
EVAL(left, RVAL, TRUE, l_true, l_false);
|
|
|
|
C_df_ilb(l_true);
|
1989-02-07 13:16:02 +00:00
|
|
|
EVAL(right->OP_LEFT, val, gencode, NO_LABEL, NO_LABEL);
|
1989-02-07 11:04:05 +00:00
|
|
|
C_bra(l_end);
|
|
|
|
C_df_ilb(l_false);
|
1989-02-07 13:16:02 +00:00
|
|
|
EVAL(right->OP_RIGHT, val, gencode, NO_LABEL, NO_LABEL);
|
1989-02-07 11:04:05 +00:00
|
|
|
C_df_ilb(l_end);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case OR:
|
|
|
|
case AND: {
|
|
|
|
label l_false, l_true, l_maybe;
|
|
|
|
|
|
|
|
l_maybe = text_label();
|
|
|
|
if (true_label) {
|
|
|
|
l_false = false_label;
|
|
|
|
l_true = true_label;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
l_false = text_label();
|
|
|
|
l_true = gencode ? text_label(): l_false;
|
|
|
|
}
|
|
|
|
|
|
|
|
EVAL(left, RVAL, TRUE, oper == AND ? l_maybe : l_true,
|
|
|
|
oper == AND ? l_false : l_maybe);
|
|
|
|
C_df_ilb(l_maybe);
|
|
|
|
EVAL(right, RVAL, gencode, l_true, l_false);
|
|
|
|
if (gencode && !true_label) {
|
|
|
|
label l_end = text_label();
|
|
|
|
|
|
|
|
C_df_ilb(l_true);
|
|
|
|
C_loc((arith)1);
|
|
|
|
C_bra(l_end);
|
|
|
|
C_df_ilb(l_false);
|
|
|
|
C_loc((arith)0);
|
|
|
|
C_df_ilb(l_end);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (! true_label) C_df_ilb(l_false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '!':
|
|
|
|
if (true_label == 0) {
|
|
|
|
EVAL(right, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode) {
|
|
|
|
C_teq();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
EVAL(right, RVAL, gencode, false_label,
|
|
|
|
true_label);
|
|
|
|
break;
|
|
|
|
case INT2INT:
|
|
|
|
case INT2FLOAT:
|
|
|
|
case FLOAT2INT:
|
|
|
|
case FLOAT2FLOAT:
|
|
|
|
EVAL(right, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
if (gencode)
|
|
|
|
conversion(right->ex_type, left->ex_type);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("(EVAL) bad operator %s\n", symbol2str(oper));
|
|
|
|
}
|
|
|
|
/* If the rvalue of the expression is required but
|
|
|
|
only its lvalue is evaluated, its rvalue is
|
|
|
|
loaded by the following statements:
|
|
|
|
*/
|
1989-09-19 16:13:23 +00:00
|
|
|
if (gencode && val == RVAL && expr->ex_lvalue == 1) {
|
1990-03-01 13:02:39 +00:00
|
|
|
if (expr->ex_type->tp_fund == FLOAT
|
|
|
|
|| expr->ex_type->tp_fund == DOUBLE
|
|
|
|
|| expr->ex_type->tp_fund == LNGDBL)
|
|
|
|
fp_used = 1;
|
1989-02-07 11:04:05 +00:00
|
|
|
load_block(expr->ex_type->tp_size,
|
|
|
|
expr->ex_type->tp_align);
|
1989-09-19 16:13:23 +00:00
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
crash("(EVAL) bad expression class");
|
|
|
|
}
|
1991-06-21 10:31:20 +00:00
|
|
|
if (val == RVAL && ((expr->ex_flags & EX_VOLATILE) || vol)) C_nop();
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* compare() serves as an auxiliary function of EVAL */
|
2019-02-18 16:42:15 +00:00
|
|
|
static void compare(int relop, label lbl)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
switch (relop) {
|
|
|
|
case '<':
|
|
|
|
C_zlt(lbl);
|
|
|
|
break;
|
|
|
|
case LESSEQ:
|
|
|
|
C_zle(lbl);
|
|
|
|
break;
|
|
|
|
case '>':
|
|
|
|
C_zgt(lbl);
|
|
|
|
break;
|
|
|
|
case GREATEREQ:
|
|
|
|
C_zge(lbl);
|
|
|
|
break;
|
|
|
|
case EQUAL:
|
|
|
|
C_zeq(lbl);
|
|
|
|
break;
|
|
|
|
case NOTEQUAL:
|
|
|
|
C_zne(lbl);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CRASH();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* truthvalue() serves as an auxiliary function of EVAL */
|
2019-02-18 16:42:15 +00:00
|
|
|
static void truthvalue(int relop)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
switch (relop) {
|
|
|
|
case '<':
|
|
|
|
C_tlt();
|
|
|
|
break;
|
|
|
|
case LESSEQ:
|
|
|
|
C_tle();
|
|
|
|
break;
|
|
|
|
case '>':
|
|
|
|
C_tgt();
|
|
|
|
break;
|
|
|
|
case GREATEREQ:
|
|
|
|
C_tge();
|
|
|
|
break;
|
|
|
|
case EQUAL:
|
|
|
|
C_teq();
|
|
|
|
break;
|
|
|
|
case NOTEQUAL:
|
|
|
|
C_tne();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CRASH();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* assop() generates the opcode of an assignment operators op= */
|
2019-02-18 16:42:15 +00:00
|
|
|
void assop(register struct type *type, int oper)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
register arith size;
|
2019-02-18 16:42:15 +00:00
|
|
|
register int uns = type->tp_unsigned;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
if ((int)(size = type->tp_size) < (int)word_size)
|
|
|
|
size = word_size;
|
|
|
|
switch (type->tp_fund) {
|
|
|
|
case CHAR:
|
|
|
|
case SHORT:
|
|
|
|
case INT:
|
|
|
|
case LONG:
|
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
|
|
|
case LNGLNG:
|
1989-02-07 11:04:05 +00:00
|
|
|
case ENUM:
|
|
|
|
switch (oper) {
|
|
|
|
case PLUSAB:
|
|
|
|
case PLUSPLUS:
|
|
|
|
case POSTINCR:
|
|
|
|
if (uns)
|
|
|
|
C_adu(size);
|
|
|
|
else
|
|
|
|
C_adi(size);
|
|
|
|
break;
|
|
|
|
case MINAB:
|
|
|
|
case MINMIN:
|
|
|
|
case POSTDECR:
|
|
|
|
if (uns)
|
|
|
|
C_sbu(size);
|
|
|
|
else
|
|
|
|
C_sbi(size);
|
|
|
|
break;
|
|
|
|
case TIMESAB:
|
|
|
|
if (uns)
|
|
|
|
C_mlu(size);
|
|
|
|
else
|
|
|
|
C_mli(size);
|
|
|
|
break;
|
|
|
|
case DIVAB:
|
|
|
|
if (uns)
|
|
|
|
C_dvu(size);
|
|
|
|
else
|
|
|
|
C_dvi(size);
|
|
|
|
break;
|
|
|
|
case MODAB:
|
|
|
|
if (uns)
|
|
|
|
C_rmu(size);
|
|
|
|
else
|
|
|
|
C_rmi(size);
|
|
|
|
break;
|
|
|
|
case LEFTAB:
|
|
|
|
if (uns)
|
|
|
|
C_slu(size);
|
|
|
|
else
|
|
|
|
C_sli(size);
|
|
|
|
break;
|
|
|
|
case RIGHTAB:
|
|
|
|
if (uns)
|
|
|
|
C_sru(size);
|
|
|
|
else
|
|
|
|
C_sri(size);
|
|
|
|
break;
|
|
|
|
case ANDAB:
|
|
|
|
C_and(size);
|
|
|
|
break;
|
|
|
|
case XORAB:
|
|
|
|
C_xor(size);
|
|
|
|
break;
|
|
|
|
case ORAB:
|
|
|
|
C_ior(size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case FLOAT:
|
|
|
|
case DOUBLE:
|
|
|
|
case LNGDBL:
|
|
|
|
switch (oper) {
|
|
|
|
case PLUSAB:
|
1989-09-19 16:13:23 +00:00
|
|
|
case PLUSPLUS: /* ??? etc... */
|
1989-02-07 11:04:05 +00:00
|
|
|
case POSTINCR:
|
|
|
|
C_adf(size);
|
|
|
|
break;
|
|
|
|
case MINAB:
|
|
|
|
case MINMIN:
|
|
|
|
case POSTDECR:
|
|
|
|
C_sbf(size);
|
|
|
|
break;
|
|
|
|
case TIMESAB:
|
|
|
|
C_mlf(size);
|
|
|
|
break;
|
|
|
|
case DIVAB:
|
|
|
|
C_dvf(size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case POINTER:
|
|
|
|
if (oper == MINAB || oper == MINMIN || oper == POSTDECR)
|
|
|
|
C_ngi(size);
|
1991-02-01 09:54:21 +00:00
|
|
|
ptr_add(size);
|
1989-02-07 11:04:05 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crash("(assop) bad type %s\n", symbol2str(type->tp_fund));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
static void ptr_add(arith size)
|
1991-02-01 09:54:21 +00:00
|
|
|
{
|
|
|
|
if (size != pointer_size) {
|
|
|
|
C_loc(size);
|
|
|
|
C_loc(pointer_size);
|
|
|
|
C_cuu();
|
|
|
|
}
|
|
|
|
C_ads(pointer_size);
|
|
|
|
}
|
|
|
|
|
1989-02-07 11:04:05 +00:00
|
|
|
/* store_val() generates code for a store operation.
|
|
|
|
There are four ways of storing data:
|
|
|
|
- into a global variable
|
|
|
|
- into an automatic local variable
|
|
|
|
- into a local static variable
|
|
|
|
- absolute addressing
|
|
|
|
*/
|
2019-02-18 16:42:15 +00:00
|
|
|
void store_val(register struct value *vl, register struct type *tp)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
1989-11-08 16:52:34 +00:00
|
|
|
register int inword = 0;
|
|
|
|
register int indword = 0;
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
writh wval = vl->vl_value;
|
|
|
|
arith val = (arith)wval;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
|
|
|
if (vl->vl_class == Const) { /* absolute addressing */
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
load_cst(wval, pointer_size);
|
1989-10-20 17:08:48 +00:00
|
|
|
store_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
return;
|
|
|
|
}
|
1989-11-08 16:52:34 +00:00
|
|
|
if (tp->tp_align % word_align == 0) {
|
|
|
|
if (tp->tp_size == word_size) inword = 1;
|
|
|
|
else if (tp->tp_size == dword_size) indword = 1;
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
if (vl->vl_class == Name) {
|
|
|
|
register struct idf *id = vl->vl_data.vl_idf;
|
|
|
|
register struct def *df = id->id_def;
|
|
|
|
|
2007-02-20 00:35:37 +00:00
|
|
|
/* if (df->df_level == L_GLOBAL) { // } ??? re-examine */
|
1989-09-19 16:13:23 +00:00
|
|
|
if (df->df_sc == GLOBAL
|
|
|
|
|| df->df_sc == EXTERN
|
1990-11-02 09:23:27 +00:00
|
|
|
|| df->df_sc == STATIC) {
|
1989-02-07 11:04:05 +00:00
|
|
|
if (inword)
|
|
|
|
C_ste_dnam(id->id_text, val);
|
|
|
|
else
|
|
|
|
if (indword)
|
|
|
|
C_sde_dnam(id->id_text, val);
|
|
|
|
else {
|
|
|
|
C_lae_dnam(id->id_text, val);
|
1989-10-20 17:08:48 +00:00
|
|
|
store_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(df->df_sc != STATIC);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (inword || indword)
|
1989-10-20 17:08:48 +00:00
|
|
|
StoreLocal(df->df_address + val, tp->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
else {
|
|
|
|
AddrLocal(df->df_address + val);
|
1989-10-20 17:08:48 +00:00
|
|
|
store_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
label dlb = vl->vl_data.vl_lbl;
|
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(vl->vl_class == Label);
|
1989-02-07 11:04:05 +00:00
|
|
|
if (inword)
|
|
|
|
C_ste_dlb(dlb, val);
|
|
|
|
else
|
|
|
|
if (indword)
|
|
|
|
C_sde_dlb(dlb, val);
|
|
|
|
else {
|
|
|
|
C_lae_dlb(dlb, val);
|
1989-10-20 17:08:48 +00:00
|
|
|
store_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* load_val() generates code for stacking a certain value (from ex),
|
|
|
|
which can be obtained in one of the following ways:
|
|
|
|
- value from absolute addressed memory
|
|
|
|
- constant value
|
|
|
|
- function result
|
|
|
|
- global variable
|
|
|
|
- static variable
|
|
|
|
- local variable
|
2019-02-18 16:42:15 +00:00
|
|
|
|
|
|
|
rlval generate rlval or lval
|
1989-02-07 11:04:05 +00:00
|
|
|
*/
|
2019-02-18 16:42:15 +00:00
|
|
|
void load_val(register struct expr *expr, int rlval)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
|
|
|
register struct type *tp = expr->ex_type;
|
|
|
|
int rvalue = (rlval == RVAL && expr->ex_lvalue != 0);
|
1989-11-08 16:52:34 +00:00
|
|
|
register int inword = 0, indword = 0;
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
writh wval = expr->VL_VALUE;
|
|
|
|
arith val = (arith)wval;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
1990-03-01 13:02:39 +00:00
|
|
|
if (expr->ex_type->tp_fund == FLOAT
|
|
|
|
|| expr->ex_type->tp_fund == DOUBLE
|
|
|
|
|| expr->ex_type->tp_fund == LNGDBL)
|
|
|
|
fp_used = 1;
|
1989-02-07 11:04:05 +00:00
|
|
|
if (expr->VL_CLASS == Const) {
|
|
|
|
if (rvalue) { /* absolute addressing */
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
load_cst(wval, pointer_size);
|
1989-10-20 17:08:48 +00:00
|
|
|
load_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
else /* integer, unsigned, long, enum etc */
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
load_cst(wval, tp->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (rvalue) {
|
1989-11-08 16:52:34 +00:00
|
|
|
if (tp->tp_align % word_align == 0) {
|
|
|
|
if (tp->tp_size == word_size) inword = 1;
|
|
|
|
else if (tp->tp_size == dword_size) indword = 1;
|
|
|
|
}
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
if (expr->VL_CLASS == Label) {
|
|
|
|
if (rvalue) {
|
|
|
|
if (inword)
|
|
|
|
C_loe_dlb(expr->VL_LBL, val);
|
|
|
|
else
|
|
|
|
if (indword)
|
|
|
|
C_lde_dlb(expr->VL_LBL, val);
|
|
|
|
else {
|
|
|
|
C_lae_dlb(expr->VL_LBL, val);
|
1989-10-20 17:08:48 +00:00
|
|
|
load_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
C_lae_dlb(expr->VL_LBL, (arith)0);
|
|
|
|
C_adp(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
register struct idf *id = expr->VL_IDF;
|
|
|
|
register struct def *df = id->id_def;
|
1990-02-28 14:51:35 +00:00
|
|
|
int fund = df->df_type->tp_fund;
|
1989-02-07 11:04:05 +00:00
|
|
|
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(ISNAME(expr));
|
1990-02-28 14:51:35 +00:00
|
|
|
if (fund == FUNCTION) {
|
1989-02-07 11:04:05 +00:00
|
|
|
/* the previous statement tried to catch a function
|
|
|
|
identifier, which may be cast to a pointer to a
|
|
|
|
function.
|
2017-11-10 03:22:13 +00:00
|
|
|
assert(!(rvalue)); ???
|
1989-02-07 11:04:05 +00:00
|
|
|
*/
|
|
|
|
C_lpi(id->id_text);
|
|
|
|
}
|
|
|
|
else
|
2007-02-20 00:35:37 +00:00
|
|
|
/* if (df->df_level == L_GLOBAL) { // } ??? re-examine */
|
1989-09-19 16:13:23 +00:00
|
|
|
if ( df->df_sc == GLOBAL
|
|
|
|
|| df->df_sc == STATIC
|
1990-11-02 09:23:27 +00:00
|
|
|
|| df->df_sc == EXTERN) {
|
1989-02-07 11:04:05 +00:00
|
|
|
if (rvalue) {
|
|
|
|
if (inword)
|
|
|
|
C_loe_dnam(id->id_text, val);
|
|
|
|
else
|
|
|
|
if (indword)
|
|
|
|
C_lde_dnam(id->id_text, val);
|
|
|
|
else {
|
|
|
|
C_lae_dnam(id->id_text, val);
|
1989-10-20 17:08:48 +00:00
|
|
|
load_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
C_lae_dnam(id->id_text, (arith)0);
|
|
|
|
C_adp(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2017-11-10 03:22:13 +00:00
|
|
|
/* assert(df->df_sc != STATIC); */
|
1989-02-07 11:04:05 +00:00
|
|
|
if (rvalue) {
|
|
|
|
if (inword || indword)
|
1989-10-20 17:08:48 +00:00
|
|
|
LoadLocal(df->df_address + val, tp->tp_size);
|
1989-02-07 11:04:05 +00:00
|
|
|
else {
|
|
|
|
AddrLocal(df->df_address + val);
|
1989-10-20 17:08:48 +00:00
|
|
|
load_block(tp->tp_size, tp->tp_align);
|
1989-02-07 11:04:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
AddrLocal(df->df_address);
|
|
|
|
C_adp(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
void load_cst(writh val, arith siz)
|
1989-02-07 11:04:05 +00:00
|
|
|
{
|
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
|
|
|
/* EM can't encode ldc with constant over 4 bytes.
|
|
|
|
Such a constant must go into rom.
|
|
|
|
*/
|
1990-07-19 17:16:36 +00:00
|
|
|
if ((int)siz <= (int)word_size)
|
1989-02-07 11:04:05 +00:00
|
|
|
C_loc(val);
|
|
|
|
else
|
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
|
|
|
if ((int)siz == (int)dword_size && (int)dword_size <= 4)
|
1989-02-07 11:04:05 +00:00
|
|
|
C_ldc(val);
|
|
|
|
else {
|
|
|
|
label datlab;
|
|
|
|
|
|
|
|
C_df_dlb(datlab = data_label());
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
C_rom_icon(writh2str(val, 0), siz);
|
1989-02-07 11:04:05 +00:00
|
|
|
C_lae_dlb(datlab, (arith)0);
|
|
|
|
C_loi(siz);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-18 16:42:15 +00:00
|
|
|
static void operands(register struct expr *expr, int gencode)
|
1989-10-20 17:08:48 +00:00
|
|
|
{
|
|
|
|
EVAL(expr->OP_LEFT, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
EVAL(expr->OP_RIGHT, RVAL, gencode, NO_LABEL, NO_LABEL);
|
|
|
|
}
|
1991-12-17 13:12:22 +00:00
|
|
|
#endif /* LINT */
|
1989-02-07 11:04:05 +00:00
|
|
|
|