ack/lang/cem/cemcom.ansi/idf.c

681 lines
15 KiB
C
Raw Permalink Normal View History

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
/* IDENTIFIER FIDDLING & SYMBOL TABLE HANDLING */
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include "parameters.h"
1989-02-07 11:04:05 +00:00
#include <em_reg.h>
#include <alloc.h>
#include "idf.h"
1989-02-07 11:04:05 +00:00
#include "arith.h"
#include "align.h"
#include "LLlex.h"
#include "level.h"
#include "stack.h"
#include "label.h"
#include "def.h"
#include "type.h"
#include "proto.h"
#include "struct.h"
#include "declar.h"
#include "decspecs.h"
#include "sizes.h"
#include "print.h"
#include "util.h"
#include "stab.h"
#include "code.h"
#include "error.h"
#include "ch3.h"
1989-02-07 11:04:05 +00:00
#include "Lpars.h"
extern char options[];
extern arith NewLocal();
extern char *symbol2str();
1989-02-07 11:04:05 +00:00
#ifdef DEBUG
#define IDF_DEBUG
#endif
1989-02-07 11:04:05 +00:00
#include <idf_pkg.body>
1989-02-07 11:04:05 +00:00
struct idf *gen_idf(void)
1989-02-07 11:04:05 +00:00
{
/* A new idf is created out of nowhere, to serve as an
anonymous name.
*/
1989-02-07 11:04:05 +00:00
static int name_cnt;
1992-10-22 16:56:28 +00:00
char *s = Malloc(strlen(dot.tk_file) + 50);
1989-02-07 11:04:05 +00:00
sprint(s, "#%d in %s, line %u", ++name_cnt, dot.tk_file, dot.tk_line);
s = Realloc(s, strlen(s) + 1);
1992-10-22 16:56:28 +00:00
return str2idf(s, 0);
1989-02-07 11:04:05 +00:00
}
int is_anon_idf(struct idf *idf)
1989-02-07 11:04:05 +00:00
{
return idf->id_text[0] == '#';
}
void declare_idf(struct decspecs *ds, struct declarator *dc, int lvl)
1989-02-07 11:04:05 +00:00
{
/* The identifier inside dc is declared on the level lvl, with
properties deduced from the decspecs ds and the declarator
dc.
The level is given explicitly to be able to insert, e.g.,
labels on the outermost level inside the function.
This routine implements the rich semantics of C
declarations.
*/
1989-02-07 11:04:05 +00:00
register struct idf *idf = dc->dc_idf;
register int sc = ds->ds_sc;
/* This local copy is essential:
char b(), c;
makes b GLOBAL and c AUTO.
*/
register struct def *def = idf->id_def; /* may be NULL */
1989-02-07 11:04:05 +00:00
register struct type *type;
struct stack_level *stl = stack_level_of(lvl);
char formal_array = 0;
1989-02-07 11:04:05 +00:00
/* determine the present type */
if (ds->ds_type == 0)
{
1989-02-07 11:04:05 +00:00
/* at the L_FORMAL1 level there is no type specified yet
*/
assert(lvl == L_FORMAL1);
type = int_type; /* may change at L_FORMAL2 */
1989-02-07 11:04:05 +00:00
}
else
{
1989-02-07 11:04:05 +00:00
/* combine the decspecs and the declarator into one type */
type = declare_type(ds->ds_type, dc);
if (type->tp_size <= (arith) 0 && actual_declaration(sc, type))
{
if (type->tp_size == (arith) -1)
{
1989-02-07 11:04:05 +00:00
/* the type is not yet known,
but it has to be:
*/
if (type->tp_fund != VOID)
{
if (level != L_GLOBAL)
error("unknown %s-type", symbol2str(type->tp_fund));
}
else
error("void is not a complete type");
1989-02-07 11:04:05 +00:00
}
else
strict("%s has size 0", idf->id_text);
1989-02-07 11:04:05 +00:00
}
}
/* some additional work for formal definitions */
if (lvl == L_FORMAL2)
{
switch (type->tp_fund)
{
1989-02-07 11:04:05 +00:00
case FUNCTION:
warning("%s is a function; cannot be formal", idf->id_text);
type = construct_type(POINTER, type, 0, (arith) 0,
NO_PROTO);
1989-02-07 11:04:05 +00:00
break;
case ARRAY: /* 3.7.1 */
type = construct_type(POINTER, type->tp_up, 0, (arith) 0,
NO_PROTO);
1989-02-07 11:04:05 +00:00
formal_array = 1;
break;
case FLOAT:
1989-02-07 11:04:05 +00:00
case CHAR:
case SHORT:
/* The conversion is done in formal_cvt(). It is
* not done when the type is float and there is a
* prototype.
*/
1989-02-07 11:04:05 +00:00
break;
}
}
/* The tests on types, postponed from do_decspecs(), can now
be performed.
*/
1989-02-07 11:04:05 +00:00
/* update the storage class */
if (type && type->tp_fund == FUNCTION)
{
if (lvl != L_GLOBAL)
{ /* 3.5.1 */
if (sc == 0)
sc = GLOBAL;
else if (sc != EXTERN && sc != TYPEDEF)
{
error("illegal storage class %s for function with block-scope",
symbol2str(sc));
ds->ds_sc = sc = EXTERN;
}
1989-02-07 11:04:05 +00:00
}
else if (sc == 0)
sc = GLOBAL;
1989-02-07 11:04:05 +00:00
}
else /* non-FUNCTION */
if (sc == 0)
sc = lvl == L_GLOBAL ? GLOBAL :
lvl == L_FORMAL1 || lvl == L_FORMAL2 ? FORMAL : AUTO;
1989-02-07 11:04:05 +00:00
#ifdef LINT
check_hiding(idf, lvl, sc); /* of some idf by this idf */
#endif /* LINT */
if (def && lvl == L_LOCAL && def->df_level == L_FORMAL2)
{
error("%s redeclared", idf->id_text);
}
1989-02-07 11:04:05 +00:00
if (def
&& (def->df_level == lvl || (lvl != L_GLOBAL && def->df_level > lvl)
|| (lvl == L_GLOBAL && def->df_level == L_PROTO && def->next
&& def->next->df_level == L_GLOBAL)))
{
1989-02-07 11:04:05 +00:00
/* There is already a declaration for idf on this
level, or even more inside.
The rules differ for different levels.
*/
switch (lvl)
{
1989-02-07 11:04:05 +00:00
case L_GLOBAL:
global_redecl(idf, sc, type);
def->df_file = idf->id_file;
def->df_line = idf->id_line;
break;
case L_FORMAL1: /* formal declaration */
1989-02-07 11:04:05 +00:00
error("formal %s redeclared", idf->id_text);
break;
case L_FORMAL2: /* formal definition */
default: /* local */
if (sc != EXTERN)
error("%s redeclared", idf->id_text);
1989-02-07 11:04:05 +00:00
break;
}
}
else /* the idf is unknown on this level */
if (lvl == L_FORMAL2 && sc != ENUM && good_formal(def, idf))
{
1989-02-07 11:04:05 +00:00
/* formal declaration, update only */
def->df_type = type;
def->df_formal_array = formal_array;
def->df_sc = sc;
def->df_level = L_FORMAL2; /* CJ */
1989-02-07 11:04:05 +00:00
def->df_file = idf->id_file;
def->df_line = idf->id_line;
}
else
{ /* fill in the def block */
1989-02-07 11:04:05 +00:00
register struct def *newdef = new_def();
newdef->next = def;
newdef->df_level = lvl;
newdef->df_type = type;
newdef->df_sc = sc;
newdef->df_file = idf->id_file;
newdef->df_line = idf->id_line;
#ifdef LINT
1990-12-07 14:42:26 +00:00
newdef->df_set = 0;
newdef->df_firstbrace = 0;
#endif /* LINT */
1989-02-07 11:04:05 +00:00
/* link it into the name list in the proper place */
idf->id_def = newdef;
update_ahead(idf);
stack_idf(idf, stl);
/* We now calculate the address.
Globals have names and don't get addresses, they
get numbers instead (through data_label()).
Formals are handled by declare_formals().
So here we hand out local addresses only.
*/
if (lvl >= L_LOCAL)
{
assert(sc);
switch (sc)
{
1989-02-07 11:04:05 +00:00
case REGISTER:
case AUTO:
if (type->tp_size == (arith) -1 && type->tp_fund != ARRAY)
{
error("size of local %s unknown", idf->id_text);
/** type = idf->id_def->df_type = int_type; **/
1989-02-07 11:04:05 +00:00
}
if (type->tp_size != (arith) -1)
{
newdef->df_address = NewLocal(type->tp_size, type->tp_align,
regtype(type), sc);
}
1989-02-07 11:04:05 +00:00
break;
case STATIC:
newdef->df_address = (arith) data_label();
break;
}
}
}
}
int actual_declaration(int sc, struct type *tp)
1989-02-07 11:04:05 +00:00
{
/* An actual_declaration needs space, right here and now.
*/
1989-02-07 11:04:05 +00:00
register int fund = tp->tp_fund;
1989-02-07 11:04:05 +00:00
if (sc == ENUM || sc == TYPEDEF) /* virtual declarations */
return 0;
if (fund == FUNCTION || fund == ARRAY)
/* allocation solved in other ways */
return 0;
if (sc == EXTERN && fund == VOID)
{
1994-04-19 10:04:53 +00:00
/* strange, but should be accepted */
return 0;
}
1989-02-07 11:04:05 +00:00
/* to be allocated */
return 1;
}
void global_redecl(register struct idf *idf, int new_sc, struct type *tp)
1989-02-07 11:04:05 +00:00
{
/* A global identifier may be declared several times,
provided the declarations do not conflict; they might
conflict in type (or supplement each other in the case of
an array) or they might conflict or supplement each other
in storage class.
*/
1989-02-07 11:04:05 +00:00
register struct def *def = idf->id_def;
while (def->df_level != L_GLOBAL)
def = def->next;
if (!equal_type(tp, def->df_type, 0, 1))
{
1989-02-07 11:04:05 +00:00
error("redeclaration of %s with different type", idf->id_text);
1990-05-03 16:18:22 +00:00
return;
}
else
update_proto(tp, def->df_type);
if (tp->tp_fund == ARRAY)
{
1989-02-07 11:04:05 +00:00
/* Multiple array declaration; this may be interesting */
if (tp->tp_size < 0)
{ /* new decl has [] */
1989-02-07 11:04:05 +00:00
/* nothing new */
}
else if (def->df_type->tp_size < 0)
{ /* old decl has [] */
1989-02-07 11:04:05 +00:00
def->df_type = tp;
}
}
if (tp->tp_fund == FUNCTION && new_sc == GLOBAL)
{
/* see 3.1.2.2 */
new_sc = EXTERN;
1989-02-07 11:04:05 +00:00
}
/* Now we may be able to update the storage class.
Clean out this mess as soon as we know all the possibilities
for new_sc.
For now we have:
EXTERN: we have seen the word "extern"
GLOBAL: the item was declared on the outer
level, without either "extern" or
"static".
STATIC: we have seen the word "static"
*/
switch (def->df_sc)
{ /* the old storage class */
1989-02-07 11:04:05 +00:00
case EXTERN:
switch (new_sc)
{ /* the new storage class */
1989-02-07 11:04:05 +00:00
case STATIC:
warning("%s redeclared static", idf->id_text);
/* fallthrough */
case GLOBAL:
def->df_sc = new_sc;
/* fallthrough */
case EXTERN:
1989-02-07 11:04:05 +00:00
break;
default:
crash("bad storage class");
/*NOTREACHED*/
}
break;
case GLOBAL:
switch (new_sc)
{ /* the new storage class */
case STATIC: /* linkage disagreement */
warning("%s redeclared static", idf->id_text);
def->df_sc = new_sc;
/* fallthrough */
1989-02-07 11:04:05 +00:00
case GLOBAL:
case EXTERN:
1989-02-07 11:04:05 +00:00
break;
default:
crash("bad storage class");
/*NOTREACHED*/
}
break;
case STATIC:
switch (new_sc)
{ /* the new storage class */
case GLOBAL: /* linkage disagreement */
1989-02-07 11:04:05 +00:00
case EXTERN:
warning("%s is already declared static", idf->id_text);
/* fallthrough */
1989-02-07 11:04:05 +00:00
case STATIC:
break;
default:
crash("bad storage class");
/*NOTREACHED*/
}
break;
case ENUM:
case TYPEDEF:
error("illegal redeclaration of %s", idf->id_text);
break;
default:
crash("bad storage class");
/*NOTREACHED*/
}
}
int good_formal(register struct def *def, register struct idf *idf)
1989-02-07 11:04:05 +00:00
{
/* Succeeds if def is a proper L_FORMAL1 definition and
gives an error message otherwise.
*/
if (!def || def->df_level != L_FORMAL1)
{ /* not in parameter list */
1989-02-07 11:04:05 +00:00
if (!is_anon_idf(idf))
error("%s not in parameter list", idf->id_text);
return 0;
}
assert(def->df_sc == FORMAL); /* CJ */
1989-02-07 11:04:05 +00:00
return 1;
}
void declare_params(struct declarator *dc)
1989-02-07 11:04:05 +00:00
{
/* Declares the formal parameters if they exist.
*/
1989-02-07 11:04:05 +00:00
register struct formal *fm = dc->dc_formal;
while (fm)
{
1989-02-07 11:04:05 +00:00
declare_parameter(fm->fm_idf);
fm = fm->next;
}
}
void idf_initialized(register struct idf *idf)
1989-02-07 11:04:05 +00:00
{
/* The topmost definition of idf is set to initialized.
*/
register struct def *def = idf->id_def; /* the topmost */
while (def->df_level <= L_PROTO)
def = def->next;
1989-02-07 11:04:05 +00:00
if (def->df_initialized)
error("multiple initialization of %s", idf->id_text);
if (def->df_sc == TYPEDEF)
{
error("typedef cannot be initialized");
return;
1989-02-07 11:04:05 +00:00
}
def->df_initialized = 1;
}
void declare_parameter(struct idf *idf)
1989-02-07 11:04:05 +00:00
{
/* idf is declared as a formal.
*/
1989-02-07 11:04:05 +00:00
add_def(idf, FORMAL, int_type, level);
}
void declare_enum(struct type *tp, struct idf *idf, arith l)
1989-02-07 11:04:05 +00:00
{
/* idf is declared as an enum constant with value l.
*/
1989-02-07 11:04:05 +00:00
add_def(idf, ENUM, tp, level);
idf->id_def->df_address = l;
}
void check_formals(struct idf *idf, struct declarator *dc)
{
register struct formal *fm = dc->dc_formal;
register struct proto *pl = idf->id_def->df_type->tp_proto;
register struct decl_unary *du = dc->dc_decl_unary;
if (!du)
{ /* error or typdef'ed function */
error("illegal definition of %s", idf->id_text);
return;
}
while (du
&& (du->du_fund != FUNCTION || du->next != (struct decl_unary *) 0))
{
du = du->next;
}
if (!du)
return; /* terrible error, signalled earlier */
if (du->du_proto)
return;
if (pl)
{
1990-04-06 15:37:16 +00:00
/* Don't give a warning about an old-style definition,
* since the arguments will be checked anyway.
*/
if (pl->pl_flag & PL_ELLIPSIS)
{
if (!(du->du_proto) && !(pl->pl_flag & PL_ERRGIVEN))
error("ellipsis terminator in previous declaration");
pl = pl->next;
}
else if (pl->pl_flag & PL_VOID)
{
pl = pl->next; /* should be 0 */
}
while (fm && pl)
{
if (!equal_type(promoted_type(fm->fm_idf->id_def->df_type),
pl->pl_type, -1, 1))
{
if (!(pl->pl_flag & PL_ERRGIVEN))
error("incorrect type for parameter %s",
fm->fm_idf->id_text);
pl->pl_flag |= PL_ERRGIVEN;
}
fm = fm->next;
pl = pl->next;
}
if (pl || fm)
{
error("incorrect number of parameters");
}
}
else
{ /* make a pseudo-prototype */
register struct proto *lpl = new_proto();
1990-04-06 15:37:16 +00:00
if (!options['o'])
warning("'%s' old-fashioned function definition",
dc->dc_idf->id_text);
while (fm)
{
if (pl == 0)
pl = lpl;
else
{
lpl->next = new_proto();
lpl = lpl->next;
}
lpl->pl_flag = PL_FORMAL;
lpl->pl_idf = fm->fm_idf;
lpl->pl_type = promoted_type(fm->fm_idf->id_def->df_type);
fm = fm->next;
}
if (pl == 0)
{ /* make func(void) */
pl = lpl;
pl->pl_type = void_type;
1990-04-06 15:37:16 +00:00
pl->pl_flag = PL_FORMAL | PL_VOID;
}
idf->id_def->df_type->tp_pseudoproto = pl;
}
free_formals(dc->dc_formal);
dc->dc_formal = 0;
}
void declare_formals(struct idf *idf, arith *fp)
1989-02-07 11:04:05 +00:00
{
/* Declares those formals as int that haven't been declared
by the user.
An address is assigned to each formal parameter.
The total size of the formals is returned in *fp;
*/
1989-02-07 11:04:05 +00:00
register struct stack_entry *se = stack_level_of(L_FORMAL1)->sl_entry;
arith f_offset = (arith) 0;
1989-02-07 11:04:05 +00:00
register int nparams = 0;
int hasproto;
struct def *df = idf->id_def;
/* When one of the formals has the same name as the function,
it hides the function def. Get it.
*/
while (se)
{
if (se->se_idf == idf)
{
df = df->next;
break;
}
se = se->next;
}
se = stack_level_of(L_FORMAL1)->sl_entry;
hasproto = df->df_type->tp_proto != 0;
1989-02-07 11:04:05 +00:00
#ifdef DEBUG
if (options['t'])
dumpidftab("start declare_formals", 0);
#endif /* DEBUG */
if (is_struct_or_union(df->df_type->tp_up->tp_fund))
{
/* create space for address of return value */
f_offset = pointer_size;
}
while (se)
{
df = se->se_idf->id_def;
/* this stacklevel may also contain tags. ignore them */
if (!df || df->df_level < L_FORMAL1)
{
se = se->next;
continue;
}
df->df_address = f_offset;
1989-02-07 11:04:05 +00:00
/* the alignment convention for parameters is: align on
word boundaries, i.e. take care that the following
parameter starts on a new word boundary.
*/
if (!hasproto && df->df_type->tp_fund == FLOAT
&& df->df_type->tp_size != double_size)
{
f_offset = align(f_offset + double_size, (int) word_size);
}
else
f_offset = align(f_offset + df->df_type->tp_size, (int) word_size);
RegisterAccount(df->df_address, df->df_type->tp_size,
regtype(df->df_type), df->df_sc);
/* cvt int to char or short and double to float, if necessary
*/
formal_cvt(hasproto, df);
df->df_level = L_FORMAL2; /* CJ */
1989-02-07 11:04:05 +00:00
if (nparams++ >= STDC_NPARAMS)
strict("number of formal parameters exceeds ANSI limit");
#ifdef DBSYMTAB
if (options['g'])
{
stb_string(df, FORMAL, se->se_idf->id_text);
}
#endif /* DBSYMTAB */
se = se->next;
1989-02-07 11:04:05 +00:00
}
*fp = f_offset;
}
int regtype(struct type *tp)
1989-02-07 11:04:05 +00:00
{
switch (tp->tp_fund)
{
1989-02-07 11:04:05 +00:00
case INT:
case LONG:
return reg_any;
case FLOAT:
case DOUBLE:
case LNGDBL:
1989-02-07 11:04:05 +00:00
return reg_float;
case POINTER:
return reg_pointer;
}
return -1;
}
void add_def(struct idf *idf, int sc, struct type *tp, int lvl)
1989-02-07 11:04:05 +00:00
{
/* The identifier idf is declared on level lvl with storage
class sc and type tp, through a faked C declaration.
This is probably the wrong way to structure the problem,
but it will have to do for the time being.
*/
struct decspecs Ds;
struct declarator Dc;
1989-02-07 11:04:05 +00:00
Ds = null_decspecs;
Ds.ds_type = tp;
Ds.ds_sc = sc;
Dc = null_declarator;
Dc.dc_idf = idf;
declare_idf(&Ds, &Dc, lvl);
}
void update_ahead(register struct idf *idf)
1989-02-07 11:04:05 +00:00
{
/* The tk_symb of the token ahead is updated in the light of new
information about the identifier idf.
*/
1989-02-07 11:04:05 +00:00
register int tk_symb = AHEAD;
if ((tk_symb == IDENTIFIER || tk_symb == TYPE_IDENTIFIER)
&& ahead.tk_idf == idf)
1989-02-07 11:04:05 +00:00
AHEAD = idf->id_def && idf->id_def->df_sc == TYPEDEF ?
TYPE_IDENTIFIER :
IDENTIFIER;
1989-02-07 11:04:05 +00:00
}
void free_formals(register struct formal *fm)
1989-02-07 11:04:05 +00:00
{
while (fm)
{
1989-02-07 11:04:05 +00:00
struct formal *tmp = fm->next;
free_formal(fm);
fm = tmp;
}
}