ack/lang/m2/comp/code.c

1138 lines
20 KiB
C
Raw Normal View History

/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*
* Author: Ceriel J.H. Jacobs
*/
1986-05-21 18:32:20 +00:00
/* C O D E G E N E R A T I O N R O U T I N E S */
/* $Header$ */
1986-05-21 18:32:20 +00:00
/* Code generation for expressions and coercions
*/
#include "debug.h"
#include <em_arith.h>
#include <em_label.h>
1986-07-08 14:59:02 +00:00
#include <em_code.h>
#include <em_abs.h>
1986-05-21 18:32:20 +00:00
#include <assert.h>
#include <alloc.h>
1986-05-21 18:32:20 +00:00
#include "type.h"
#include "LLlex.h"
1986-05-21 18:32:20 +00:00
#include "def.h"
#include "scope.h"
#include "desig.h"
#include "node.h"
#include "Lpars.h"
1986-05-30 18:48:00 +00:00
#include "standards.h"
1986-06-20 14:36:49 +00:00
#include "walk.h"
1986-05-21 18:32:20 +00:00
extern char *long2str();
extern char *symbol2str();
extern int proclevel;
extern char options[];
1986-06-06 02:22:09 +00:00
int fp_used;
1986-05-21 18:32:20 +00:00
STATIC char *
NameOfProc(df)
register t_def *df;
{
assert(df->df_kind & (D_PROCHEAD|D_PROCEDURE));
if (df->df_kind == D_PROCEDURE) {
return df->prc_vis->sc_scope->sc_name;
}
return df->for_name;
}
1986-05-21 18:32:20 +00:00
CodeConst(cst, size)
1987-07-21 13:54:33 +00:00
arith cst;
int size;
1986-05-21 18:32:20 +00:00
{
/* Generate code to push constant "cst" with size "size"
*/
1987-07-21 13:54:33 +00:00
if (size <= (int) word_size) {
1986-05-21 18:32:20 +00:00
C_loc(cst);
}
1987-07-21 13:54:33 +00:00
else if (size == (int) dword_size) {
1986-05-21 18:32:20 +00:00
C_ldc(cst);
}
else {
1986-10-06 20:36:30 +00:00
crash("(CodeConst)");
/*
1987-05-18 15:57:33 +00:00
C_df_dlb(++data_label);
1987-07-21 13:54:33 +00:00
C_rom_icon(long2str((long) cst), (arith) size);
c_lae_dlb(data_label);
1987-07-21 13:54:33 +00:00
C_loi((arith) size);
1986-10-06 20:36:30 +00:00
*/
1986-05-21 18:32:20 +00:00
}
}
CodeString(nd)
register t_node *nd;
1986-05-21 18:32:20 +00:00
{
1986-09-25 19:39:06 +00:00
if (nd->nd_type->tp_fund != T_STRING) {
1987-05-18 15:57:33 +00:00
/* Character constant */
1986-05-23 09:46:31 +00:00
C_loc(nd->nd_INT);
1987-07-21 13:54:33 +00:00
return;
1986-05-23 09:46:31 +00:00
}
1987-07-21 13:54:33 +00:00
C_df_dlb(++data_label);
C_rom_scon(nd->nd_STR, WA((arith)(nd->nd_SLE + 1)));
c_lae_dlb(data_label);
1986-06-04 09:01:48 +00:00
}
1986-05-21 18:32:20 +00:00
CodeExpr(nd, ds, true_label, false_label)
register t_node *nd;
register t_desig *ds;
1986-05-21 18:32:20 +00:00
label true_label, false_label;
{
register t_type *tp = nd->nd_type;
1986-05-21 18:32:20 +00:00
DoLineno(nd);
1986-06-06 02:22:09 +00:00
if (tp->tp_fund == T_REAL) fp_used = 1;
1986-05-21 18:32:20 +00:00
switch(nd->nd_class) {
case Def:
if (nd->nd_def->df_kind & (D_PROCEDURE|D_PROCHEAD)) {
1986-06-04 09:01:48 +00:00
C_lpi(NameOfProc(nd->nd_def));
1986-05-30 18:48:00 +00:00
ds->dsg_kind = DSG_LOADED;
break;
}
1986-06-04 09:01:48 +00:00
/* Fall through */
case Link:
1986-06-10 13:18:52 +00:00
case Arrsel:
case Arrow:
1986-05-21 18:32:20 +00:00
CodeDesig(nd, ds);
break;
case Oper:
CodeOper(nd, true_label, false_label);
1987-05-18 15:57:33 +00:00
ds->dsg_kind = DSG_LOADED;
true_label = NO_LABEL;
1986-05-21 18:32:20 +00:00
break;
case Uoper:
CodeUoper(nd);
ds->dsg_kind = DSG_LOADED;
break;
case Value:
switch(nd->nd_symb) {
1987-05-18 15:57:33 +00:00
case REAL:
C_df_dlb(++data_label);
1987-07-21 13:54:33 +00:00
C_rom_fcon(nd->nd_REL, tp->tp_size);
c_lae_dlb(data_label);
1987-07-21 13:54:33 +00:00
C_loi(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case STRING:
CodeString(nd);
break;
case INTEGER:
1987-07-21 13:54:33 +00:00
CodeConst(nd->nd_INT, (int) (tp->tp_size));
1986-05-21 18:32:20 +00:00
break;
default:
crash("Value error");
}
ds->dsg_kind = DSG_LOADED;
break;
case Call:
CodeCall(nd);
ds->dsg_kind = DSG_LOADED;
break;
1986-05-23 19:25:21 +00:00
case Set: {
1987-07-21 13:54:33 +00:00
register unsigned i = (unsigned) (tp->tp_size) / (int) word_size;
1987-05-18 15:57:33 +00:00
register arith *st = nd->nd_set + i;
1986-05-23 19:25:21 +00:00
1986-05-28 18:36:51 +00:00
ds->dsg_kind = DSG_LOADED;
1987-07-21 13:54:33 +00:00
for (; i; i--) {
1986-05-23 19:25:21 +00:00
C_loc(*--st);
}
FreeSet(nd->nd_set);
CodeSet(nd);
1986-05-23 19:25:21 +00:00
}
break;
1986-05-21 18:32:20 +00:00
default:
crash("(CodeExpr) bad node type");
}
1987-05-18 15:57:33 +00:00
if (true_label != NO_LABEL) {
1986-10-06 20:36:30 +00:00
/* Only for boolean expressions
*/
1987-06-23 17:12:25 +00:00
CodeValue(ds, tp);
1986-05-21 18:32:20 +00:00
C_zne(true_label);
C_bra(false_label);
}
}
CodeCoercion(t1, t2)
register t_type *t1, *t2;
1986-05-21 18:32:20 +00:00
{
1986-06-04 09:01:48 +00:00
register int fund1, fund2;
1987-06-23 17:12:25 +00:00
arith sz1 = t1->tp_size;
arith sz2;
1986-05-28 18:36:51 +00:00
1986-06-26 09:39:36 +00:00
t1 = BaseType(t1);
t2 = BaseType(t2);
sz2 = t2->tp_size;
1987-06-23 17:12:25 +00:00
switch(fund1 = t1->tp_fund) {
case T_WORD:
fund1 = T_INTEGER;
break;
case T_CHAR:
case T_ENUMERATION:
case T_CARDINAL:
case T_INTORCARD:
if (sz1 < word_size) sz1 = word_size;
/* fall through */
case T_EQUAL:
1987-06-23 17:12:25 +00:00
case T_POINTER:
fund1 = T_CARDINAL;
break;
}
switch(fund2 = t2->tp_fund) {
1987-06-23 17:12:25 +00:00
case T_WORD:
fund2 = T_INTEGER;
break;
case T_CHAR:
case T_ENUMERATION:
sz2 = word_size;
/* fall through */
case T_EQUAL:
1987-06-23 17:12:25 +00:00
case T_POINTER:
fund2 = T_CARDINAL;
break;
}
1986-05-28 18:36:51 +00:00
switch(fund1) {
case T_INTEGER:
1987-06-23 17:12:25 +00:00
if (sz1 < word_size) {
1987-11-09 10:17:20 +00:00
c_loc((int)sz1);
c_loc((int) word_size);
1987-06-23 17:12:25 +00:00
C_cii();
sz1 = word_size;
1987-06-23 17:12:25 +00:00
}
if (fund2 == T_REAL) {
c_loc((int)sz1);
c_loc((int)sz2);
C_cif();
1986-05-28 18:36:51 +00:00
break;
}
if (sz2 != sz1) {
c_loc((int)sz1);
c_loc((int)sz2);
switch(fund2) {
case T_INTEGER:
C_cii();
break;
case T_CARDINAL:
1986-05-28 18:36:51 +00:00
C_ciu();
break;
default:
crash("Funny integer conversion");
1986-05-28 18:36:51 +00:00
}
}
break;
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
if (fund2 == T_REAL) {
c_loc((int)sz1);
c_loc((int)sz2);
C_cuf();
1986-05-28 18:36:51 +00:00
break;
}
if (sz1 != sz2) {
c_loc((int)sz1);
c_loc((int)sz2);
switch(fund2) {
case T_CARDINAL:
case T_INTORCARD:
C_cuu();
break;
case T_INTEGER:
1987-06-23 17:12:25 +00:00
C_cui();
break;
default:
crash("Funny cardinal conversion");
1987-06-23 17:12:25 +00:00
}
1986-05-28 18:36:51 +00:00
}
break;
case T_REAL:
switch(fund2) {
case T_REAL:
if (sz1 != sz2) {
c_loc((int)sz1);
c_loc((int)sz2);
1986-05-28 18:36:51 +00:00
C_cff();
}
break;
case T_INTEGER:
c_loc((int)sz1);
c_loc((int)sz2);
1986-05-28 18:36:51 +00:00
C_cfi();
break;
case T_CARDINAL:
if (! options['R']) {
label lb = ++text_label;
C_dup(sz1);
C_zrf(sz1);
C_cmf(sz1);
C_zge(lb);
c_loc(ECONV);
C_trp();
def_ilb(lb);
}
c_loc((int)sz1);
c_loc((int)sz2);
1986-05-28 18:36:51 +00:00
C_cfu();
break;
default:
crash("Funny REAL conversion");
}
break;
}
1986-05-21 18:32:20 +00:00
}
CodeCall(nd)
register t_node *nd;
1986-05-21 18:32:20 +00:00
{
/* Generate code for a procedure call. Checking of parameters
and result is already done.
*/
register t_node *left = nd->nd_left;
t_type *result_tp;
1988-10-25 17:43:19 +00:00
int needs_fn;
1986-05-21 18:32:20 +00:00
if (left->nd_type == std_type) {
CodeStd(nd);
return;
}
assert(IsProc(left));
1986-05-21 18:32:20 +00:00
if (result_tp = ResultType(left->nd_type)) {
if (TooBigForReturnArea(result_tp)) {
C_asp(-WA(result_tp->tp_size));
}
}
1988-10-25 17:43:19 +00:00
if (nd->nd_right) {
CodeParameters(ParamList(left->nd_type), nd->nd_right);
1986-05-21 18:32:20 +00:00
}
1986-08-26 14:33:24 +00:00
switch(left->nd_class) {
case Def: {
register t_def *df = left->nd_def;
if (df->df_kind == D_CONST) {
df = df->con_const.tk_data.tk_def;
}
if (df->df_kind & (D_PROCEDURE|D_PROCHEAD)) {
int level = df->df_scope->sc_level;
1986-08-26 14:33:24 +00:00
if (level > 0) {
1987-05-18 15:57:33 +00:00
C_lxl((arith) (proclevel - level));
1986-08-26 14:33:24 +00:00
}
needs_fn = df->df_scope->sc_defmodule;
C_cal(NameOfProc(df));
1986-08-26 14:33:24 +00:00
break;
}}
/* Fall through */
default:
1988-10-25 17:43:19 +00:00
needs_fn = 1;
1986-05-30 18:48:00 +00:00
CodePExpr(left);
1986-05-21 18:32:20 +00:00
C_cai();
}
1987-05-18 15:57:33 +00:00
C_asp(left->nd_type->prc_nbpar);
if (result_tp) {
if (TooBigForReturnArea(result_tp)) {
1986-06-20 14:36:49 +00:00
}
else C_lfr(WA(result_tp->tp_size));
1986-05-21 18:32:20 +00:00
}
1988-10-25 17:43:19 +00:00
DoFilename(needs_fn);
1988-03-23 17:44:25 +00:00
DoLineno(nd);
1986-05-21 18:32:20 +00:00
}
1986-06-17 12:04:05 +00:00
CodeParameters(param, arg)
t_param *param;
t_node *arg;
1986-06-17 12:04:05 +00:00
{
register t_type *tp;
register t_node *left;
register t_type *left_type;
1986-09-25 19:39:06 +00:00
1986-06-17 12:04:05 +00:00
assert(param != 0 && arg != 0);
1987-07-16 19:51:40 +00:00
if (param->par_next) {
CodeParameters(param->par_next, arg->nd_right);
1986-06-17 12:04:05 +00:00
}
tp = TypeOfParam(param);
left = arg->nd_left;
1986-08-26 14:33:24 +00:00
left_type = left->nd_type;
1986-06-17 12:04:05 +00:00
if (IsConformantArray(tp)) {
register t_type *elem = tp->arr_elem;
1986-11-26 16:40:45 +00:00
1986-06-17 12:04:05 +00:00
C_loc(tp->arr_elsize);
1986-08-26 14:33:24 +00:00
if (IsConformantArray(left_type)) {
1987-05-18 15:57:33 +00:00
DoHIGH(left->nd_def);
1986-11-26 16:40:45 +00:00
if (elem->tp_size != left_type->arr_elem->tp_size) {
1986-06-17 12:04:05 +00:00
/* This can only happen if the formal type is
1986-11-26 16:40:45 +00:00
ARRAY OF (WORD|BYTE)
1986-06-17 12:04:05 +00:00
*/
1986-08-26 14:33:24 +00:00
C_loc(left_type->arr_elem->tp_size);
1986-11-26 16:40:45 +00:00
C_mli(word_size);
if (elem == word_type) {
c_loc((int) word_size - 1);
1986-11-26 16:40:45 +00:00
C_adi(word_size);
c_loc((int) word_size);
1986-11-26 16:40:45 +00:00
C_dvi(word_size);
}
else {
assert(elem == byte_type);
}
1986-06-17 12:04:05 +00:00
}
}
else if (left->nd_symb == STRING) {
C_loc((arith)(left->nd_SLE - 1));
1986-06-17 12:04:05 +00:00
}
1986-11-26 16:40:45 +00:00
else if (elem == word_type) {
1986-08-26 14:33:24 +00:00
C_loc((left_type->tp_size+word_size-1) / word_size - 1);
1986-06-17 12:04:05 +00:00
}
1986-11-26 16:40:45 +00:00
else if (elem == byte_type) {
C_loc(left_type->tp_size - 1);
}
1986-06-17 12:04:05 +00:00
else {
C_loc(left_type->arr_high - left_type->arr_low);
1986-06-17 12:04:05 +00:00
}
c_loc(0);
1986-06-17 12:04:05 +00:00
if (left->nd_symb == STRING) {
CodeString(left);
}
1987-07-22 10:59:24 +00:00
else switch(left->nd_class) {
case Arrsel:
case Arrow:
case Def:
CodeDAddress(left, IsVarParam(param));
1987-07-22 10:59:24 +00:00
break;
default:{
1986-11-26 16:40:45 +00:00
arith tmp, TmpSpace();
CodePExpr(left);
tmp = TmpSpace(left->nd_type->tp_size, left->nd_type->tp_align);
1988-03-22 17:54:01 +00:00
STL(tmp, WA(left->nd_type->tp_size));
1986-11-26 16:40:45 +00:00
C_lal(tmp);
1987-07-22 10:59:24 +00:00
}
break;
1986-11-26 16:40:45 +00:00
}
return;
1986-06-17 12:04:05 +00:00
}
1986-11-26 16:40:45 +00:00
if (IsVarParam(param)) {
CodeDAddress(left, 1);
1986-11-26 16:40:45 +00:00
return;
1986-06-17 12:04:05 +00:00
}
1986-11-26 16:40:45 +00:00
if (left_type->tp_fund == T_STRING) {
1986-12-01 10:06:53 +00:00
CodePString(left, tp);
1986-11-26 16:40:45 +00:00
return;
1986-06-17 12:04:05 +00:00
}
1986-11-26 16:40:45 +00:00
CodePExpr(left);
1986-06-17 12:04:05 +00:00
}
1986-12-01 10:06:53 +00:00
CodePString(nd, tp)
t_node *nd;
t_type *tp;
1986-12-01 10:06:53 +00:00
{
arith szarg = WA(nd->nd_type->tp_size);
register arith zersz = WA(tp->tp_size) - szarg;
if (zersz) {
/* null padding required */
assert(zersz > 0);
C_zer(zersz);
}
CodeString(nd); /* push address of string */
C_loi(szarg);
}
1987-11-26 14:15:24 +00:00
static
subu(sz)
arith sz;
{
if (options['R']) C_sbu(sz);
else {
CAL((int) sz == (int) word_size ? "subu" : "subul", (int) sz);
1987-11-26 14:15:24 +00:00
}
}
static
addu(sz)
arith sz;
{
if (options['R']) C_adu(sz);
else {
CAL((int) sz == (int) word_size ? "addu" : "addul", (int) sz);
1987-11-26 14:15:24 +00:00
}
}
1986-05-21 18:32:20 +00:00
CodeStd(nd)
t_node *nd;
1986-05-21 18:32:20 +00:00
{
register t_node *arg = nd->nd_right;
register t_node *left = 0;
1988-10-13 15:43:23 +00:00
register t_type *tp = 0;
1986-10-06 20:36:30 +00:00
int std = nd->nd_left->nd_def->df_value.df_stdname;
1986-05-30 18:48:00 +00:00
if (arg) {
left = arg->nd_left;
1986-06-26 09:39:36 +00:00
tp = BaseType(left->nd_type);
1986-05-30 18:48:00 +00:00
arg = arg->nd_right;
}
1986-10-06 20:36:30 +00:00
switch(std) {
case S_ORD:
case S_VAL:
CodePExpr(left);
break;
1986-05-30 18:48:00 +00:00
case S_ABS:
CodePExpr(left);
if (tp->tp_fund == T_INTEGER) {
CAL((int)(tp->tp_size) == (int)int_size ? "absi" : "absl", (int)(tp->tp_size));
1986-05-30 18:48:00 +00:00
}
else if (tp->tp_fund == T_REAL) {
CAL((int)(tp->tp_size) == (int)float_size ? "absf" : "absd", (int)(tp->tp_size));
1986-05-30 18:48:00 +00:00
}
C_lfr(tp->tp_size);
break;
case S_CAP:
CodePExpr(left);
C_cal("cap");
1986-05-30 18:48:00 +00:00
break;
case S_HIGH:
assert(IsConformantArray(tp));
1987-05-18 15:57:33 +00:00
DoHIGH(left->nd_def);
1986-05-30 18:48:00 +00:00
break;
case S_SIZE:
case S_TSIZE:
assert(IsConformantArray(tp));
1987-05-18 15:57:33 +00:00
DoHIGH(left->nd_def);
C_inc();
C_loc(tp->arr_elem->tp_size);
C_mlu(word_size);
break;
1986-05-30 18:48:00 +00:00
case S_ODD:
1986-08-26 14:33:24 +00:00
CodePExpr(left);
1986-05-30 18:48:00 +00:00
if (tp->tp_size == word_size) {
c_loc(1);
1986-05-30 18:48:00 +00:00
C_and(word_size);
}
else {
assert(tp->tp_size == dword_size);
C_ldc((arith) 1);
C_and(dword_size);
C_ior(word_size);
}
break;
case S_ADR:
CodeDAddress(left, 1);
1986-05-30 18:48:00 +00:00
break;
case S_DEC:
1986-10-06 20:36:30 +00:00
case S_INC: {
register arith size;
1986-10-06 20:36:30 +00:00
size = left->nd_type->tp_size;
1986-10-06 20:36:30 +00:00
if (size < word_size) size = word_size;
1986-05-30 18:48:00 +00:00
CodePExpr(left);
CodeCoercion(left->nd_type, tp);
1987-06-23 17:12:25 +00:00
if (arg) {
CodePExpr(arg->nd_left);
CodeCoercion(arg->nd_left->nd_type, tp);
}
else {
c_loc(1);
1987-06-23 17:12:25 +00:00
CodeCoercion(intorcard_type, tp);
}
1986-10-06 20:36:30 +00:00
if (std == S_DEC) {
if (tp->tp_fund == T_INTEGER) C_sbi(size);
1987-11-26 14:15:24 +00:00
else subu(size);
1986-05-30 18:48:00 +00:00
}
else {
1986-10-06 20:36:30 +00:00
if (tp->tp_fund == T_INTEGER) C_adi(size);
1987-11-26 14:15:24 +00:00
else addu(size);
1986-05-30 18:48:00 +00:00
}
if (size == word_size) {
RangeCheck(left->nd_type, tp->tp_fund == T_INTEGER ?
int_type : card_type);
}
1986-05-30 18:48:00 +00:00
CodeDStore(left);
break;
1986-10-06 20:36:30 +00:00
}
1986-05-30 18:48:00 +00:00
case S_HALT:
1988-03-23 17:44:25 +00:00
C_cal("halt");
1986-05-30 18:48:00 +00:00
break;
case S_INCL:
case S_EXCL:
CodePExpr(left);
CodePExpr(arg->nd_left);
1987-10-28 16:03:56 +00:00
C_loc(tp->set_low);
C_sbi(word_size);
1986-05-30 18:48:00 +00:00
C_set(tp->tp_size);
if (std == S_INCL) {
C_ior(tp->tp_size);
}
else {
C_com(tp->tp_size);
C_and(tp->tp_size);
}
CodeDStore(left);
break;
default:
crash("(CodeStd)");
}
1986-05-21 18:32:20 +00:00
}
1986-09-25 19:39:06 +00:00
RangeCheck(tpl, tpr)
register t_type *tpl, *tpr;
1986-05-28 18:36:51 +00:00
{
/* Generate a range check if neccessary
*/
arith rlo, rhi;
1986-05-28 18:36:51 +00:00
if (options['R']) return;
1986-05-28 18:36:51 +00:00
if (bounded(tpl)) {
/* In this case we might need a range check.
If both types are restricted. check the bounds
to see wether we need a range check.
We don't need one if the range of values of the
right hand side is a subset of the range of values
of the left hand side.
*/
if (bounded(tpr)) {
getbounds(tpr, &rlo, &rhi);
if (in_range(rlo, tpl) && in_range(rhi, tpl)) {
return;
}
1986-05-28 18:36:51 +00:00
}
genrck(tpl);
return;
1986-05-28 18:36:51 +00:00
}
if (tpl->tp_size <= tpr->tp_size &&
((tpl->tp_fund == T_INTEGER && tpr == card_type) ||
(tpr->tp_fund == T_INTEGER && tpl == card_type))) {
label lb = ++text_label;
C_dup(word_size);
C_zge(lb);
c_loc(ECONV);
C_trp();
def_ilb(lb);
}
1986-05-21 18:32:20 +00:00
}
Operands(nd)
register t_node *nd;
1986-05-21 18:32:20 +00:00
{
CodePExpr(nd->nd_left);
CodePExpr(nd->nd_right);
DoLineno(nd);
1986-05-21 18:32:20 +00:00
}
CodeOper(expr, true_label, false_label)
register t_node *expr; /* the expression tree itself */
1986-05-21 18:32:20 +00:00
label true_label;
label false_label; /* labels to jump to in logical expr's */
{
register t_node *leftop = expr->nd_left;
register t_node *rightop = expr->nd_right;
register t_type *tp = expr->nd_type;
1986-05-21 18:32:20 +00:00
1986-08-26 14:33:24 +00:00
switch (expr->nd_symb) {
1986-05-21 18:32:20 +00:00
case '+':
Operands(expr);
1986-05-21 18:32:20 +00:00
switch (tp->tp_fund) {
case T_INTEGER:
C_adi(tp->tp_size);
break;
case T_REAL:
C_adf(tp->tp_size);
break;
1986-08-26 14:33:24 +00:00
case T_POINTER:
1986-09-25 19:39:06 +00:00
case T_EQUAL:
C_ads(rightop->nd_type->tp_size);
break;
1986-05-21 18:32:20 +00:00
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1987-11-26 14:15:24 +00:00
addu(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case T_SET:
C_ior(tp->tp_size);
break;
default:
crash("bad type +");
}
break;
case '-':
Operands(expr);
1986-05-21 18:32:20 +00:00
switch (tp->tp_fund) {
case T_INTEGER:
C_sbi(tp->tp_size);
break;
case T_REAL:
C_sbf(tp->tp_size);
break;
case T_POINTER:
case T_EQUAL:
if (rightop->nd_type == address_type) {
C_sbs(tp->tp_size);
break;
}
C_ngi(rightop->nd_type->tp_size);
C_ads(rightop->nd_type->tp_size);
break;
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
case T_CARDINAL:
1987-11-26 14:15:24 +00:00
subu(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case T_SET:
C_com(tp->tp_size);
C_and(tp->tp_size);
break;
default:
crash("bad type -");
}
break;
case '*':
Operands(expr);
1986-05-21 18:32:20 +00:00
switch (tp->tp_fund) {
case T_INTEGER:
C_mli(tp->tp_size);
break;
case T_POINTER:
1986-09-25 19:39:06 +00:00
case T_EQUAL:
1986-05-21 18:32:20 +00:00
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1987-11-26 14:15:24 +00:00
if (options['R']) {
C_mlu(tp->tp_size);
}
else {
CAL((int)(tp->tp_size) <= (int)word_size ? "mulu" : "mulul",
(int)(tp->tp_size));
1987-11-26 14:15:24 +00:00
}
1986-05-21 18:32:20 +00:00
break;
case T_REAL:
C_mlf(tp->tp_size);
break;
case T_SET:
C_and(tp->tp_size);
break;
default:
crash("bad type *");
}
break;
case '/':
Operands(expr);
1986-05-21 18:32:20 +00:00
switch (tp->tp_fund) {
case T_REAL:
C_dvf(tp->tp_size);
break;
case T_SET:
C_xor(tp->tp_size);
break;
default:
crash("bad type /");
}
break;
case DIV:
Operands(expr);
1986-05-21 18:32:20 +00:00
switch(tp->tp_fund) {
case T_INTEGER:
C_dvi(tp->tp_size);
break;
case T_POINTER:
1986-09-25 19:39:06 +00:00
case T_EQUAL:
1986-05-21 18:32:20 +00:00
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1986-05-21 18:32:20 +00:00
C_dvu(tp->tp_size);
break;
default:
crash("bad type DIV");
}
break;
case MOD:
Operands(expr);
1986-05-21 18:32:20 +00:00
switch(tp->tp_fund) {
case T_INTEGER:
C_rmi(tp->tp_size);
break;
case T_POINTER:
1986-09-25 19:39:06 +00:00
case T_EQUAL:
1986-05-21 18:32:20 +00:00
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1986-05-21 18:32:20 +00:00
C_rmu(tp->tp_size);
break;
default:
crash("bad type MOD");
}
break;
case '<':
case LESSEQUAL:
case '>':
case GREATEREQUAL:
case '=':
case '#':
Operands(expr);
1986-08-26 14:33:24 +00:00
tp = BaseType(leftop->nd_type);
if (tp == intorcard_type) tp = BaseType(rightop->nd_type);
1986-05-21 18:32:20 +00:00
switch (tp->tp_fund) {
case T_INTEGER:
1986-05-28 18:36:51 +00:00
C_cmi(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case T_POINTER:
1986-09-25 19:39:06 +00:00
case T_HIDDEN:
1986-12-01 10:06:53 +00:00
case T_EQUAL:
1988-02-01 10:17:51 +00:00
C_cmp();
break;
1986-05-21 18:32:20 +00:00
case T_CARDINAL:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1986-05-28 18:36:51 +00:00
C_cmu(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case T_ENUMERATION:
case T_CHAR:
C_cmu(word_size);
break;
case T_REAL:
1986-05-28 18:36:51 +00:00
C_cmf(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
case T_SET:
1986-08-26 14:33:24 +00:00
if (expr->nd_symb == GREATEREQUAL) {
1986-05-28 18:36:51 +00:00
/* A >= B is the same as A equals A + B
*/
C_dup(tp->tp_size << 1);
1986-05-28 18:36:51 +00:00
C_asp(tp->tp_size);
1986-08-26 14:33:24 +00:00
C_ior(tp->tp_size);
expr->nd_symb = '=';
1986-05-28 18:36:51 +00:00
}
1986-08-26 14:33:24 +00:00
else if (expr->nd_symb == LESSEQUAL) {
1986-05-28 18:36:51 +00:00
/* A <= B is the same as A - B = {}
*/
C_com(tp->tp_size);
C_and(tp->tp_size);
1986-06-20 14:36:49 +00:00
C_zer(tp->tp_size);
expr->nd_symb = '=';
1986-05-28 18:36:51 +00:00
}
C_cms(tp->tp_size);
1986-05-21 18:32:20 +00:00
break;
default:
crash("bad type COMPARE");
}
1987-05-18 15:57:33 +00:00
if (true_label != NO_LABEL) {
1986-08-26 14:33:24 +00:00
compare(expr->nd_symb, true_label);
1986-05-21 18:32:20 +00:00
C_bra(false_label);
1987-07-21 13:54:33 +00:00
break;
1986-05-21 18:32:20 +00:00
}
1987-07-21 13:54:33 +00:00
truthvalue(expr->nd_symb);
1986-05-21 18:32:20 +00:00
break;
1987-07-21 13:54:33 +00:00
1986-05-21 18:32:20 +00:00
case IN:
1986-05-28 18:36:51 +00:00
/* In this case, evaluate right hand side first! The
INN instruction expects the bit number on top of the
stack
*/
1986-08-26 14:33:24 +00:00
CodePExpr(rightop);
CodePExpr(leftop);
1987-10-28 16:03:56 +00:00
C_loc(rightop->nd_type->set_low);
1987-12-02 10:41:38 +00:00
C_sbu(word_size);
1986-05-28 18:36:51 +00:00
C_inn(rightop->nd_type->tp_size);
1987-05-18 15:57:33 +00:00
if (true_label != NO_LABEL) {
1986-06-10 13:18:52 +00:00
C_zne(true_label);
C_bra(false_label);
}
1986-05-21 18:32:20 +00:00
break;
1986-10-06 20:36:30 +00:00
case OR:
case AND: {
1988-10-13 15:43:23 +00:00
label l_maybe = ++text_label, l_end = NO_LABEL;
t_desig *Des = new_desig();
1986-08-26 14:33:24 +00:00
1987-05-18 15:57:33 +00:00
if (true_label == NO_LABEL) {
1986-11-26 16:40:45 +00:00
true_label = ++text_label;
false_label = ++text_label;
1986-08-26 14:33:24 +00:00
l_end = ++text_label;
}
1986-10-06 20:36:30 +00:00
if (expr->nd_symb == OR) {
CodeExpr(leftop, Des, true_label, l_maybe);
1986-10-06 20:36:30 +00:00
}
else CodeExpr(leftop, Des, l_maybe, false_label);
def_ilb(l_maybe);
1988-03-22 17:54:01 +00:00
clear((char *) Des, sizeof(t_desig));
CodeExpr(rightop, Des, true_label, false_label);
1988-10-13 15:43:23 +00:00
if (l_end != NO_LABEL) {
def_ilb(true_label);
c_loc(1);
1986-05-21 18:32:20 +00:00
C_bra(l_end);
def_ilb(false_label);
c_loc(0);
def_ilb(l_end);
1986-05-21 18:32:20 +00:00
}
free_desig(Des);
1986-05-21 18:32:20 +00:00
break;
1986-08-26 14:33:24 +00:00
}
1986-05-21 18:32:20 +00:00
default:
1986-08-26 14:33:24 +00:00
crash("(CodeOper) Bad operator %s\n",symbol2str(expr->nd_symb));
1986-05-21 18:32:20 +00:00
}
}
/* compare() serves as an auxiliary function of CodeOper */
compare(relop, lbl)
int relop;
1986-05-23 19:25:21 +00:00
register label lbl;
1986-05-21 18:32:20 +00:00
{
switch (relop) {
case '<':
C_zlt(lbl);
break;
case LESSEQUAL:
C_zle(lbl);
break;
case '>':
C_zgt(lbl);
break;
case GREATEREQUAL:
C_zge(lbl);
break;
case '=':
C_zeq(lbl);
break;
case '#':
C_zne(lbl);
break;
default:
crash("(compare)");
}
}
/* truthvalue() serves as an auxiliary function of CodeOper */
truthvalue(relop)
int relop;
{
switch (relop) {
case '<':
C_tlt();
break;
case LESSEQUAL:
C_tle();
break;
case '>':
C_tgt();
break;
case GREATEREQUAL:
C_tge();
break;
case '=':
C_teq();
break;
case '#':
C_tne();
break;
default:
crash("(truthvalue)");
}
}
CodeUoper(nd)
register t_node *nd;
1986-05-21 18:32:20 +00:00
{
register t_type *tp = nd->nd_type;
1986-05-21 18:32:20 +00:00
1986-06-17 12:04:05 +00:00
CodePExpr(nd->nd_right);
1986-05-21 18:32:20 +00:00
switch(nd->nd_symb) {
case NOT:
C_teq();
break;
case '-':
switch(tp->tp_fund) {
case T_INTEGER:
1986-08-26 14:33:24 +00:00
case T_INTORCARD:
1986-05-21 18:32:20 +00:00
C_ngi(tp->tp_size);
break;
case T_REAL:
C_ngf(tp->tp_size);
break;
default:
crash("Bad operand to unary -");
}
break;
case COERCION:
CodeCoercion(nd->nd_right->nd_type, tp);
RangeCheck(tp, nd->nd_right->nd_type);
break;
case CAST:
break;
1986-05-21 18:32:20 +00:00
default:
crash("Bad unary operator");
}
}
1986-05-23 19:25:21 +00:00
CodeSet(nd)
register t_node *nd;
1986-05-23 19:25:21 +00:00
{
register t_type *tp = nd->nd_type;
1986-05-23 19:25:21 +00:00
nd = nd->nd_right;
while (nd) {
assert(nd->nd_class == Link && nd->nd_symb == ',');
if (nd->nd_left) CodeEl(nd->nd_left, tp);
1986-05-23 19:25:21 +00:00
nd = nd->nd_right;
}
}
CodeEl(nd, tp)
register t_node *nd;
register t_type *tp;
1986-05-23 19:25:21 +00:00
{
register t_type *eltype = ElementType(tp);
1986-05-23 19:25:21 +00:00
if (nd->nd_class == Link && nd->nd_symb == UPTO) {
1987-10-28 16:03:56 +00:00
C_loc(tp->set_low);
1986-06-06 02:22:09 +00:00
C_loc(tp->tp_size); /* push size */
1986-06-26 09:39:36 +00:00
if (eltype->tp_fund == T_SUBRANGE) {
C_loc(eltype->sub_ub);
1986-06-04 09:01:48 +00:00
}
1986-06-26 09:39:36 +00:00
else C_loc((arith) (eltype->enm_ncst - 1));
Operands(nd);
CAL("LtoUset", 5 * (int) word_size);
/* library routine to fill set */
1986-05-23 19:25:21 +00:00
}
else {
1986-05-30 18:48:00 +00:00
CodePExpr(nd);
1987-10-28 16:03:56 +00:00
C_loc(tp->set_low);
C_sbi(word_size);
1986-05-23 19:25:21 +00:00
C_set(tp->tp_size);
1986-06-06 02:22:09 +00:00
C_ior(tp->tp_size);
1986-05-23 19:25:21 +00:00
}
}
1986-05-30 18:48:00 +00:00
CodePExpr(nd)
register t_node *nd;
1986-05-30 18:48:00 +00:00
{
/* Generate code to push the value of the expression "nd"
on the stack.
*/
register t_desig *designator = new_desig();
1986-05-30 18:48:00 +00:00
CodeExpr(nd, designator, NO_LABEL, NO_LABEL);
CodeValue(designator, nd->nd_type);
free_desig(designator);
1986-05-30 18:48:00 +00:00
}
CodeDAddress(nd, chk_controlvar)
t_node *nd;
1986-05-30 18:48:00 +00:00
{
/* Generate code to push the address of the designator "nd"
on the stack.
*/
register t_desig *designator = new_desig();
int chkptr;
1986-05-30 18:48:00 +00:00
if (chk_controlvar) ChkForFOR(nd);
CodeDesig(nd, designator);
chkptr = designator->dsg_kind==DSG_PLOADED ||
designator->dsg_kind==DSG_PFIXED;
CodeAddress(designator);
/* Generate dummy use of pointer, to get possible error message
as soon as possible
*/
if (chkptr && ! options['R']) {
C_dup(pointer_size);
C_loi((arith) 1);
C_asp(word_size);
}
free_desig(designator);
1986-05-30 18:48:00 +00:00
}
CodeDStore(nd)
register t_node *nd;
1986-05-30 18:48:00 +00:00
{
/* Generate code to store the expression on the stack into the
designator "nd".
*/
register t_desig *designator = new_desig();
1986-05-30 18:48:00 +00:00
1987-08-10 13:01:54 +00:00
ChkForFOR(nd);
CodeDesig(nd, designator);
CodeStore(designator, nd->nd_type);
free_desig(designator);
1986-05-30 18:48:00 +00:00
}
1986-06-04 09:01:48 +00:00
1987-05-18 15:57:33 +00:00
DoHIGH(df)
register t_def *df;
1986-06-04 09:01:48 +00:00
{
1986-08-26 14:33:24 +00:00
/* Get the high index of a conformant array, indicated by "nd".
The high index is the second field in the descriptor of
the array, so it is easily found.
*/
1986-06-20 14:36:49 +00:00
register arith highoff;
1986-06-04 09:01:48 +00:00
assert(df->df_kind == D_VARIABLE);
1986-08-26 14:33:24 +00:00
assert(IsConformantArray(df->df_type));
1986-06-04 09:01:48 +00:00
1986-08-26 14:33:24 +00:00
highoff = df->var_off /* base address and descriptor */
+ word_size + pointer_size;
/* skip base and first field of
descriptor
*/
1986-06-04 09:01:48 +00:00
if (df->df_scope->sc_level < proclevel) {
1986-06-10 13:18:52 +00:00
C_lxa((arith) (proclevel - df->df_scope->sc_level));
1986-06-04 09:01:48 +00:00
C_lof(highoff);
}
else C_lol(highoff);
}
#ifdef SQUEEZE
c_loc(n)
{
C_loc((arith) n);
}
c_lae_dlb(l)
label l;
{
C_lae_dlb(l, (arith) 0);
}
CAL(name, ssp)
char *name;
int ssp;
{
C_cal(name);
C_asp((arith) ssp);
}
#endif