15950f9c95
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.
799 lines
19 KiB
C
799 lines
19 KiB
C
/*
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
*/
|
|
/* $Id$ */
|
|
/* C O D E - G E N E R A T I N G R O U T I N E S */
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "parameters.h"
|
|
#ifndef LINT
|
|
#include <em.h>
|
|
#else
|
|
#include "l_em.h"
|
|
#include "l_lint.h"
|
|
#endif /* LINT */
|
|
#include <alloc.h>
|
|
#include <flt_arith.h>
|
|
#include "idf.h"
|
|
#include "arith.h"
|
|
#include "type.h"
|
|
#include "label.h"
|
|
#include "code.h"
|
|
#include "stmt.h"
|
|
#include "def.h"
|
|
#include "expr.h"
|
|
#include "sizes.h"
|
|
#include "stack.h"
|
|
#include "blocks.h"
|
|
#include "struct.h"
|
|
#include "level.h"
|
|
#include "dataflow.h"
|
|
#include "conversion.h"
|
|
#include "decspecs.h"
|
|
#include "declar.h"
|
|
#include "Lpars.h"
|
|
#include "specials.h"
|
|
#include "atw.h"
|
|
#include "ch3.h"
|
|
#include "eval.h"
|
|
#include "stab.h"
|
|
#include "LLlex.h"
|
|
#include "align.h"
|
|
#include "util.h"
|
|
#include "error.h"
|
|
|
|
#ifdef LINT
|
|
#include "l_lint.h"
|
|
#endif /* LINT */
|
|
#ifdef DBSYMTAB
|
|
#include <stb.h>
|
|
#endif /* DBSYMTAB */
|
|
|
|
label lab_count = 1;
|
|
label datlab_count = 1;
|
|
|
|
int fp_used;
|
|
|
|
extern void str_cst(register char *, register int, int); /* ival.c */
|
|
|
|
|
|
|
|
/* global function info */
|
|
char *func_name;
|
|
struct type *func_type;
|
|
#ifdef LINT
|
|
int func_notypegiven;
|
|
#endif
|
|
|
|
#ifdef USE_TMP
|
|
static int tmp_id;
|
|
static int pro_id;
|
|
#endif /* USE_TMP */
|
|
|
|
extern char options[];
|
|
extern char *symbol2str();
|
|
extern char *source;
|
|
|
|
void loc_init();
|
|
|
|
#ifndef LINT
|
|
void init_code(char *dst_file)
|
|
{
|
|
/* init_code() initialises the output file on which the
|
|
compact EM code is written
|
|
*/
|
|
C_init(word_size, pointer_size); /* initialise EM module */
|
|
if (C_open(dst_file) == 0)
|
|
fatal("cannot write to %s\n", dst_file);
|
|
C_magic();
|
|
C_ms_emx(word_size, pointer_size);
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
C_ms_std(source, N_SO, 0);
|
|
stb_typedef(int_type, "int");
|
|
stb_typedef(schar_type, "char");
|
|
stb_typedef(long_type, "long");
|
|
stb_typedef(short_type, "short");
|
|
stb_typedef(uchar_type, "unsigned char");
|
|
stb_typedef(ushort_type, "unsigned short");
|
|
stb_typedef(ulong_type, "unsigned long");
|
|
stb_typedef(uint_type, "unsigned int");
|
|
if (lnglng_size >= 0) {
|
|
stb_typedef(lnglng_type, "long long");
|
|
stb_typedef(ulnglng_type, "unsigned long long");
|
|
}
|
|
stb_typedef(float_type, "float");
|
|
stb_typedef(double_type, "double");
|
|
stb_typedef(lngdbl_type, "long double");
|
|
stb_typedef(void_type, "void");
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
#ifdef USE_TMP
|
|
#ifdef PREPEND_SCOPES
|
|
C_insertpart(tmp_id = C_getid());
|
|
#endif /* PREPEND_SCOPES */
|
|
#endif /* USE_TMP */
|
|
}
|
|
#endif /* LINT */
|
|
|
|
struct string_cst *str_list = 0;
|
|
|
|
label code_string(char* val, int len)
|
|
{
|
|
register struct string_cst *sc = new_string_cst();
|
|
label dlb = data_label();
|
|
|
|
C_ina_dlb(dlb);
|
|
sc->next = str_list;
|
|
str_list = sc;
|
|
sc->sc_value = val;
|
|
sc->sc_len = len;
|
|
sc->sc_dlb = dlb;
|
|
return dlb;
|
|
}
|
|
|
|
void def_strings(register struct string_cst *sc)
|
|
{
|
|
while (sc) {
|
|
struct string_cst *sc1 = sc;
|
|
|
|
C_df_dlb(sc->sc_dlb);
|
|
str_cst(sc->sc_value, sc->sc_len, 1); /* string in rom */
|
|
sc = sc->next;
|
|
free(sc1->sc_value);
|
|
free_string_cst(sc1);
|
|
}
|
|
}
|
|
|
|
/* flush_strings() is called from program.g after each external definition */
|
|
void flush_strings(void) {
|
|
if (str_list) {
|
|
def_strings(str_list);
|
|
str_list = 0;
|
|
}
|
|
}
|
|
|
|
#ifndef LINT
|
|
void end_code(void)
|
|
{
|
|
/* end_code() performs the actions to be taken when closing
|
|
the output stream.
|
|
*/
|
|
if (fp_used) {
|
|
/* floating point used */
|
|
C_ms_flt();
|
|
}
|
|
C_ms_src((int)(LineNumber - 2), source);
|
|
C_close();
|
|
}
|
|
#endif /* LINT */
|
|
|
|
#ifdef PREPEND_SCOPES
|
|
void prepend_scopes(void)
|
|
{
|
|
/* prepend_scopes() runs down the list of global idf's
|
|
and generates those exa's, exp's, ina's and inp's
|
|
that superior hindsight has provided.
|
|
*/
|
|
register struct stack_entry *se = local_level->sl_entry;
|
|
|
|
#ifdef USE_TMP
|
|
C_beginpart(tmp_id);
|
|
#endif /* USE_TMP */
|
|
while (se != 0) {
|
|
register struct def *df = se->se_idf->id_def;
|
|
|
|
if (df && (df->df_initialized || df->df_used || df->df_alloc)) {
|
|
code_scope(se->se_idf->id_text, df);
|
|
}
|
|
se = se->next;
|
|
}
|
|
#ifdef USE_TMP
|
|
C_endpart(tmp_id);
|
|
#endif /* USE_TMP */
|
|
}
|
|
#endif /* PREPEND_SCOPES */
|
|
|
|
void code_scope(char* text, register struct def *def)
|
|
{
|
|
/* generates code for one name, text, of the storage class
|
|
as given by def, if meaningful.
|
|
*/
|
|
int fund = def->df_type->tp_fund;
|
|
|
|
switch (def->df_sc) {
|
|
case EXTERN:
|
|
case GLOBAL:
|
|
if (fund == FUNCTION)
|
|
C_exp(text);
|
|
else
|
|
C_exa_dnam(text);
|
|
break;
|
|
case STATIC:
|
|
if (fund == FUNCTION)
|
|
C_inp(text);
|
|
else
|
|
C_ina_dnam(text);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static label return_label, return2_label;
|
|
static char return_expr_occurred;
|
|
static arith func_size;
|
|
static int struct_return;
|
|
static char *last_fn_given = (char *)0;
|
|
static label file_name_label;
|
|
|
|
void begin_proc(struct decspecs *ds, struct idf *idf) /* to be called when entering a procedure */
|
|
{
|
|
/* begin_proc() is called at the entrance of a new function
|
|
and performs the necessary code generation:
|
|
- a scope indicator (if needed) exp/inp
|
|
- the procedure entry pro $name
|
|
- reserves some space if the result of the function
|
|
does not fit in the return area
|
|
- a fil pseudo instruction
|
|
*/
|
|
register char *name = idf->id_text;
|
|
register struct def *def = idf->id_def;
|
|
|
|
/* idf->id_def does not indicate the right def structure
|
|
* when the function being defined has a parameter of the
|
|
* same name in an old-style function definition.
|
|
*/
|
|
while (def->df_level != L_GLOBAL) def = def->next;
|
|
|
|
/* When we have a new-style function definition, the parameters
|
|
* are defined first, which means that it may look as if they have
|
|
* the greatest scope. Correct this.
|
|
*/
|
|
if (idf->id_def == def && def->next && def->next->df_level == L_PROTO) {
|
|
struct def *tmpdef = def->next;
|
|
|
|
def->next = tmpdef->next;
|
|
tmpdef->next = def;
|
|
idf->id_def = tmpdef;
|
|
}
|
|
#ifndef PREPEND_SCOPES
|
|
code_scope(name, def);
|
|
#endif /* PREPEND_SCOPES */
|
|
#ifdef DATAFLOW
|
|
if (options['d'])
|
|
DfaStartFunction(name);
|
|
#endif /* DATAFLOW */
|
|
|
|
|
|
/* set global function info */
|
|
func_name = name;
|
|
if (def->df_type->tp_fund != FUNCTION) {
|
|
error("making function body for non-function");
|
|
def->df_type = error_type;
|
|
}
|
|
func_type = def->df_type->tp_up;
|
|
#ifdef LINT
|
|
func_notypegiven = ds->ds_notypegiven;
|
|
#endif
|
|
func_size = ATW(func_type->tp_size);
|
|
sp_occurred[SP_SETJMP] = 0;
|
|
|
|
#ifndef USE_TMP
|
|
C_pro_narg(name);
|
|
#else
|
|
C_insertpart(pro_id = C_getid());
|
|
#endif
|
|
if (is_struct_or_union(func_type->tp_fund)) {
|
|
if (func_size <= 0) {
|
|
error("unknown return type for function %s", name);
|
|
} else {
|
|
struct_return = 1;
|
|
}
|
|
}
|
|
else
|
|
struct_return = 0;
|
|
/* Special arrangements if the function result doesn't fit in
|
|
the function return area of the EM machine. The size of
|
|
the function return area is implementation dependent.
|
|
*/
|
|
lab_count = (label) 1;
|
|
return_label = text_label();
|
|
return2_label = text_label();
|
|
return_expr_occurred = 0;
|
|
LocalInit();
|
|
prc_entry(name);
|
|
if (! options['L']) { /* profiling */
|
|
if (!last_fn_given || strcmp(last_fn_given, FileName) != 0) {
|
|
/* previous function came from other file */
|
|
C_df_dlb(file_name_label = data_label());
|
|
C_con_scon(last_fn_given = FileName,
|
|
(arith)(strlen(FileName) + 1));
|
|
}
|
|
/* enable debug trace of EM source */
|
|
C_fil_dlb(file_name_label, (arith)0);
|
|
C_lin((arith)LineNumber);
|
|
}
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
stb_string(def, FUNCTION, name);
|
|
if (! strcmp(name, "main")) {
|
|
C_ms_stb_cst(name, N_MAIN, 0, (arith) 0);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void end_proc(arith fbytes)
|
|
{
|
|
/* end_proc() deals with the code to be generated at the end of
|
|
a function, as there is:
|
|
- the EM ret instruction: "ret 0"
|
|
- loading of the function result in the function
|
|
result area if there has been a return <expr>
|
|
in the function body (see do_return_expr())
|
|
- indication of the use of floating points
|
|
- indication of the number of bytes used for
|
|
formal parameters
|
|
- use of special identifiers such as "__setjmp"
|
|
- "end" + number of bytes used for local variables
|
|
*/
|
|
arith nbytes;
|
|
char optionsn = options['n'];
|
|
|
|
#ifdef DATAFLOW
|
|
if (options['d'])
|
|
DfaEndFunction();
|
|
#endif /* DATAFLOW */
|
|
C_df_ilb(return2_label);
|
|
if (return_expr_occurred && struct_return == 0) {
|
|
C_asp(-func_size);
|
|
}
|
|
C_df_ilb(return_label);
|
|
prc_exit();
|
|
if (return_expr_occurred) {
|
|
if (struct_return != 0) {
|
|
LoadLocal((arith) 0, pointer_size);
|
|
C_ret(pointer_size);
|
|
}
|
|
else
|
|
C_ret(func_size);
|
|
}
|
|
else C_ret((arith) 0);
|
|
|
|
/* getting the number of "local" bytes is posponed until here,
|
|
because copying the function result may need temporaries!
|
|
However, local_level is now L_FORMAL2, because
|
|
L_LOCAL is already unstacked. Therefore, "unstack_level" must
|
|
also pass "sl_max_block" to the level above L_LOCAL.
|
|
*/
|
|
nbytes = ATW(- local_level->sl_max_block);
|
|
#ifdef USE_TMP
|
|
C_beginpart(pro_id);
|
|
C_pro(func_name, nbytes);
|
|
#endif
|
|
if (fbytes > max_int) {
|
|
error("%s has more than %ld parameter bytes",
|
|
func_name, (long) max_int);
|
|
}
|
|
C_ms_par(fbytes); /* # bytes for formals */
|
|
if (sp_occurred[SP_SETJMP]) { /* indicate use of "__setjmp" */
|
|
options['n'] = 1;
|
|
C_ms_gto();
|
|
sp_occurred[SP_SETJMP] = 0;
|
|
}
|
|
#ifdef USE_TMP
|
|
C_endpart(pro_id);
|
|
#endif
|
|
LocalFinish();
|
|
C_end(nbytes);
|
|
if (nbytes > max_int) {
|
|
error("%s has more than %ld bytes of local variables",
|
|
func_name, (long) max_int);
|
|
}
|
|
options['n'] = optionsn;
|
|
}
|
|
|
|
void do_return(void)
|
|
{
|
|
/* do_return handles the case of a return without expression.
|
|
This version branches to the return label, which is
|
|
probably smarter than generating a direct return.
|
|
Return sequences may be expensive.
|
|
*/
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) db_line(dot.tk_file, dot.tk_line);
|
|
#endif /* DBSYMTAB */
|
|
C_bra(return2_label);
|
|
}
|
|
|
|
void do_return_expr(struct expr *expr)
|
|
{
|
|
/* do_return_expr() generates the expression and the jump for
|
|
a return statement with an expression.
|
|
*/
|
|
ch3cast(&expr, RETURN, func_type);
|
|
code_expr(expr, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
if (struct_return != 0) {
|
|
LoadLocal((arith) 0, pointer_size);
|
|
store_block(func_type->tp_size, func_type->tp_align);
|
|
}
|
|
C_bra(return_label);
|
|
return_expr_occurred = 1;
|
|
}
|
|
|
|
void
|
|
code_declaration(
|
|
register struct idf *idf, /* idf to be declared */
|
|
struct expr *expr, /* initialisation; NULL if absent */
|
|
int lvl, /* declaration level */
|
|
int sc) /* storage class, as in the declaration */
|
|
{
|
|
/* code_declaration() does the actual declaration of the
|
|
variable indicated by "idf" on declaration level "lvl".
|
|
If the variable is initialised, the expression is given
|
|
in "expr", but for global and static initialisations it
|
|
is just non-zero, as the expression is not parsed yet.
|
|
There are some cases to be considered:
|
|
- filter out typedefs, they don't correspond to code;
|
|
- global variables, coded only if initialized;
|
|
- local static variables;
|
|
- local automatic variables;
|
|
Since the expression may be modified in the process,
|
|
code_declaration() frees it after use, as the caller can
|
|
no longer do so.
|
|
|
|
If there is a storage class indication (EXTERN/STATIC),
|
|
code_declaration() will generate an exa or ina.
|
|
The sc is the actual storage class, as given in the
|
|
declaration.
|
|
*/
|
|
register struct def *def = idf->id_def;
|
|
register arith size = def->df_type->tp_size;
|
|
int fund = def->df_type->tp_fund;
|
|
int def_sc = def->df_sc;
|
|
|
|
if (def_sc == TYPEDEF) { /* no code for typedefs */
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
stb_typedef(def->df_type, idf->id_text);
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
return;
|
|
}
|
|
if (lvl == L_GLOBAL) { /* global variable */
|
|
/* is this an allocating declaration? */
|
|
if ( (sc == 0 || sc == STATIC)
|
|
&& fund != FUNCTION
|
|
)
|
|
def->df_alloc = ALLOC_SEEN;
|
|
if (expr && def_sc == STATIC && sc == EXTERN) {
|
|
warning("%s has internal linkage", idf->id_text);
|
|
}
|
|
if (expr) { /* code only if initialized */
|
|
#ifndef PREPEND_SCOPES
|
|
code_scope(idf->id_text, def);
|
|
#endif /* PREPEND_SCOPES */
|
|
def->df_alloc = ALLOC_DONE;
|
|
C_df_dnam(idf->id_text);
|
|
}
|
|
}
|
|
else
|
|
if (lvl >= L_LOCAL) { /* local variable */
|
|
/* STATIC, EXTERN, GLOBAL, AUTO or REGISTER */
|
|
switch (def_sc) {
|
|
case STATIC:
|
|
if (fund == FUNCTION) {
|
|
/* should produce "inp $function" ??? */
|
|
break;
|
|
}
|
|
/* they are handled on the spot and get an
|
|
integer label in EM.
|
|
*/
|
|
#ifdef DBSYMTAB
|
|
if (options['g'] && ! expr) {
|
|
stb_string(def, sc, idf->id_text);
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
C_df_dlb((label)def->df_address);
|
|
if (expr) { /* there is an initialisation */
|
|
}
|
|
else { /* produce blank space */
|
|
if (size <= 0) {
|
|
error("size of %s unknown", idf->id_text);
|
|
size = (arith)0;
|
|
}
|
|
C_bss_cst(ATW(size), (arith)0, 1);
|
|
}
|
|
break;
|
|
case EXTERN:
|
|
if (expr && !is_anon_idf(idf) && level != L_GLOBAL)
|
|
error("cannot initialize extern %s in block"
|
|
, idf->id_text);
|
|
case GLOBAL:
|
|
/* we are sure there is no expression */
|
|
break;
|
|
case AUTO:
|
|
case REGISTER:
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
stb_string(def, sc, idf->id_text);
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
if (expr)
|
|
loc_init(expr, idf);
|
|
else if ((fund == ARRAY)
|
|
&& (def->df_type->tp_size == (arith)-1)) {
|
|
error("size for local %s unknown"
|
|
, idf->id_text);
|
|
}
|
|
break;
|
|
default:
|
|
crash("bad local storage class");
|
|
/*NOTREACHED*/
|
|
}
|
|
}
|
|
}
|
|
|
|
void loc_init(struct expr *expr, struct idf *id)
|
|
{
|
|
/* loc_init() generates code for the assignment of
|
|
expression expr to the local variable described by id.
|
|
It frees the expression afterwards.
|
|
*/
|
|
register struct expr *e = expr;
|
|
register struct def *df = id->id_def;
|
|
register struct type *tp = df->df_type;
|
|
static arith tmpoffset = 0;
|
|
static arith unknownsize = 0;
|
|
|
|
assert(df->df_sc != STATIC);
|
|
switch (tp->tp_fund) {
|
|
case ARRAY:
|
|
if (tp->tp_size == (arith) -1)
|
|
unknownsize = 1;
|
|
case STRUCT:
|
|
case UNION:
|
|
if (e != (struct expr *) 0) {
|
|
break; /* switch */
|
|
} else if (!tmpoffset) {/* first time for this variable */
|
|
tmpoffset = df->df_address;
|
|
if (unknownsize) tmpoffset = -1;
|
|
df->df_address = data_label();
|
|
C_df_dlb((label)df->df_address);
|
|
} else {
|
|
C_lae_dlb((label)df->df_address, (arith)0);
|
|
load_block(tp->tp_size, word_align);
|
|
if (unknownsize) {
|
|
/* tmpoffset += tp->tp_size; */
|
|
unknownsize = 0;
|
|
|
|
tmpoffset = NewLocal(tp->tp_size
|
|
, tp->tp_align
|
|
, regtype(tp)
|
|
, df->df_sc);
|
|
}
|
|
C_lal(tmpoffset);
|
|
store_block(tp->tp_size, tmpoffset % word_align ? 1 : word_align);
|
|
df->df_address = tmpoffset;
|
|
tmpoffset = 0;
|
|
}
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
stb_string(df, AUTO, id->id_text);
|
|
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
return;
|
|
}
|
|
if (ISCOMMA(e)) { /* embraced: int i = {12}; */
|
|
while (e) {
|
|
loc_init(e->OP_LEFT, id);
|
|
e = e->OP_RIGHT;
|
|
}
|
|
}
|
|
else { /* not embraced */
|
|
ch3cast(&expr, '=', tp); /* may modify expr */
|
|
#ifndef LINT
|
|
{
|
|
struct value vl;
|
|
|
|
EVAL(expr, RVAL, TRUE, NO_LABEL, NO_LABEL);
|
|
vl.vl_class = Name;
|
|
vl.vl_data.vl_idf = id;
|
|
vl.vl_value = 0;
|
|
store_val(&vl, tp);
|
|
}
|
|
#else /* LINT */
|
|
id->id_def->df_set = 1;
|
|
#endif /* LINT */
|
|
free_expression(expr);
|
|
}
|
|
}
|
|
|
|
void bss(register struct idf *idf)
|
|
{
|
|
/* bss() allocates bss space for the global idf.
|
|
*/
|
|
register struct def *df = idf->id_def;
|
|
|
|
#ifndef PREPEND_SCOPES
|
|
code_scope(idf->id_text, df);
|
|
#endif /* PREPEND_SCOPES */
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) {
|
|
stb_string(df, df->df_sc, idf->id_text);
|
|
}
|
|
#endif /* DBSYMTAB */
|
|
if (df->df_type->tp_size <= 0) {
|
|
if (df->df_sc != STATIC &&
|
|
df->df_type->tp_fund == ARRAY &&
|
|
df->df_type->tp_up &&
|
|
df->df_type->tp_up->tp_size >= 0) {
|
|
C_df_dnam(idf->id_text);
|
|
C_bss_cst(ATW(df->df_type->tp_up->tp_size), (arith)0, 1);
|
|
}
|
|
else error("size of %s unknown (\"%s\", line %d)"
|
|
, idf->id_text, df->df_file, df->df_line);
|
|
} else {
|
|
C_df_dnam(idf->id_text);
|
|
C_bss_cst(ATW(df->df_type->tp_size), (arith)0, 1);
|
|
}
|
|
}
|
|
|
|
void formal_cvt(int hasproto, register struct def *df)
|
|
{
|
|
/* formal_cvt() converts a formal parameter of type char or
|
|
short from int to that type. It also converts a formal
|
|
parameter of type float from a double to a float.
|
|
*/
|
|
register struct type *tp = df->df_type;
|
|
|
|
if (tp->tp_size != int_size &&
|
|
(tp->tp_fund == CHAR || tp->tp_fund == SHORT)
|
|
) {
|
|
LoadLocal(df->df_address, int_size);
|
|
/* conversion(int_type, df->df_type); ???
|
|
No, you can't do this on the stack! (CJ)
|
|
*/
|
|
StoreLocal(df->df_address, tp->tp_size);
|
|
} else if (tp->tp_size != double_size
|
|
&& tp->tp_fund == FLOAT
|
|
&& !hasproto) {
|
|
LoadLocal(df->df_address, double_size);
|
|
#ifndef LINT
|
|
conversion(double_type, float_type);
|
|
#endif /* LINT */
|
|
StoreLocal(df->df_address, tp->tp_size);
|
|
}
|
|
}
|
|
|
|
#ifdef LINT
|
|
/*ARGSUSED*/
|
|
#endif /* LINT */
|
|
void code_expr(struct expr *expr, int val, int code, label tlbl, label flbl)
|
|
{
|
|
/* code_expr() is the parser's interface to the expression code
|
|
generator. If line number trace is wanted, it generates a
|
|
lin instruction. EVAL() is called directly.
|
|
*/
|
|
#ifndef LINT
|
|
if (! options['L']) /* profiling */
|
|
C_lin((arith)(expr->ex_line));
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) db_line(expr->ex_file, (unsigned int)expr->ex_line);
|
|
#endif
|
|
EVAL(expr, val, code, tlbl, flbl);
|
|
#else /* LINT */
|
|
lint_expr(expr, code ? USED : IGNORED);
|
|
#endif /* LINT */
|
|
}
|
|
|
|
/* The FOR/WHILE/DO/SWITCH stacking mechanism:
|
|
stack_stmt() has to be called at the entrance of a
|
|
for, while, do or switch statement to indicate the
|
|
EM labels where a subsequent break or continue causes
|
|
the program to jump to.
|
|
*/
|
|
static struct stmt_block *stmt_stack; /* top of statement stack */
|
|
|
|
/* code_break() generates EM code needed at the occurrence of "break":
|
|
it generates a branch instruction to the break label of the
|
|
innermost statement in which break has a meaning.
|
|
As "break" is legal in any of 'while', 'do', 'for' or 'switch',
|
|
which are the only ones that are stacked, only the top of
|
|
the stack is interesting.
|
|
*/
|
|
void code_break(void)
|
|
{
|
|
register struct stmt_block *stmt_block = stmt_stack;
|
|
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) db_line(dot.tk_file, dot.tk_line);
|
|
#endif /* DBSYMTAB */
|
|
if (stmt_block)
|
|
C_bra(stmt_block->st_break);
|
|
else
|
|
error("break not inside for, while, do or switch");
|
|
}
|
|
|
|
/* code_continue() generates EM code needed at the occurrence of
|
|
"continue":
|
|
it generates a branch instruction to the continue label of the
|
|
innermost statement in which continue has a meaning.
|
|
*/
|
|
void
|
|
code_continue(void)
|
|
{
|
|
register struct stmt_block *stmt_block = stmt_stack;
|
|
|
|
while (stmt_block) {
|
|
if (stmt_block->st_continue) {
|
|
#ifdef DBSYMTAB
|
|
if (options['g']) db_line(dot.tk_file, dot.tk_line);
|
|
#endif /* DBSYMTAB */
|
|
C_bra(stmt_block->st_continue);
|
|
return;
|
|
}
|
|
stmt_block = stmt_block->next;
|
|
}
|
|
error("continue not inside for, while or do");
|
|
}
|
|
|
|
void stack_stmt(label break_label, label cont_label)
|
|
{
|
|
register struct stmt_block *stmt_block = new_stmt_block();
|
|
|
|
stmt_block->next = stmt_stack;
|
|
stmt_block->st_break = break_label;
|
|
stmt_block->st_continue = cont_label;
|
|
stmt_stack = stmt_block;
|
|
}
|
|
|
|
void unstack_stmt(void)
|
|
{
|
|
/* unstack_stmt() unstacks the data of a statement
|
|
which may contain break or continue
|
|
*/
|
|
register struct stmt_block *sbp = stmt_stack;
|
|
stmt_stack = sbp->next;
|
|
free_stmt_block(sbp);
|
|
}
|
|
|
|
static label prc_name;
|
|
|
|
void prc_entry(char* name)
|
|
{
|
|
if (options['p']) {
|
|
C_df_dlb(prc_name = data_label());
|
|
C_rom_scon(name, (arith) (strlen(name) + 1));
|
|
C_lae_dlb(prc_name, (arith) 0);
|
|
C_cal("procentry");
|
|
C_asp(pointer_size);
|
|
}
|
|
}
|
|
|
|
void prc_exit(void)
|
|
{
|
|
if (options['p']) {
|
|
C_lae_dlb(prc_name, (arith) 0);
|
|
C_cal("procexit");
|
|
C_asp(pointer_size);
|
|
}
|
|
}
|
|
|
|
#ifdef DBSYMTAB
|
|
void db_line(char *file, unsigned int line)
|
|
{
|
|
static unsigned oldline;
|
|
static char *oldfile;
|
|
|
|
if (file != oldfile || line != oldline) {
|
|
C_ms_std((char *) 0, N_SLINE, (int) line);
|
|
oldline = line;
|
|
oldfile = file;
|
|
}
|
|
}
|
|
#endif /* DBSYMTAB */
|