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

281 lines
6.8 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
/* S T A C K / U N S T A C K R O U T I N E S */
#include "parameters.h"
1989-02-07 11:04:05 +00:00
#include <system.h>
1990-12-07 14:42:26 +00:00
#ifndef LINT
1989-02-07 11:04:05 +00:00
#include <em.h>
1990-12-07 14:42:26 +00:00
#else
#include "l_em.h"
#endif /* LINT */
1989-02-07 11:04:05 +00:00
#include <alloc.h>
#include "idf.h"
1989-02-07 11:04:05 +00:00
#include "Lpars.h"
#include "arith.h"
#include "stack.h"
#include "type.h"
#include "def.h"
#include "struct.h"
#include "level.h"
#include "mes.h"
#include "code.h"
#include "util.h"
#include "error.h"
1989-02-07 11:04:05 +00:00
/* #include <em_reg.h> */
extern char options[];
static struct stack_level UniversalLevel;
struct stack_level *local_level = &UniversalLevel;
/* The main reason for having this secondary stacking
mechanism besides the linked lists pointed to by the idf's
is efficiency.
To remove the idf's of a given level, one could scan the
hash table and chase down the idf chains; with a hash
table size of 100 this is feasible, but with a size of say
100000 this becomes painful. Therefore all idf's are also
kept in a stack of sets, one set for each level.
*/
int level; /* Always equal to local_level->sl_level. */
void stack_level(void) {
1989-02-07 11:04:05 +00:00
/* A new level is added on top of the identifier stack.
*/
register struct stack_level *stl = new_stack_level();
register struct stack_level *loclev = local_level;
loclev->sl_next = stl;
stl->sl_previous = loclev;
stl->sl_level = ++level;
stl->sl_max_block = loclev->sl_max_block;
local_level = stl;
#ifdef LINT
lint_start_local();
#endif /* LINT */
1989-02-07 11:04:05 +00:00
}
void stack_idf(
struct idf *idf,
register struct stack_level *stl)
1989-02-07 11:04:05 +00:00
{
/* The identifier idf is inserted in the stack on level stl,
but only if it is not already present at this level.
1989-02-07 11:04:05 +00:00
*/
register struct stack_entry *se;
se = stl->sl_entry;
while (se) {
if (se->se_idf == idf) return;
se = se->next;
}
se = new_stack_entry();
1989-02-07 11:04:05 +00:00
/* link it into the stack level */
se->next = stl->sl_entry;
se->se_idf = idf;
stl->sl_entry = se;
}
struct stack_level *stack_level_of(int lvl)
1989-02-07 11:04:05 +00:00
{
/* The stack_level corresponding to level lvl is returned.
The stack should probably be an array, to be extended with
realloc where needed.
*/
register struct stack_level *stl;
if (lvl == level)
return local_level;
stl = &UniversalLevel;
while (stl->sl_level != lvl) {
1989-02-07 11:04:05 +00:00
stl = stl->sl_next;
}
1989-02-07 11:04:05 +00:00
return stl;
}
void unstack_level(void)
1989-02-07 11:04:05 +00:00
{
/* The top level of the identifier stack is removed.
*/
struct stack_level *lastlvl;
#ifdef DEBUG
if (options['t'])
dumpidftab("before unstackidfs", 0);
#endif /* DEBUG */
1989-02-07 11:04:05 +00:00
#ifdef LINT
1990-12-07 14:42:26 +00:00
lint_end_local(local_level);
#endif /* LINT */
1989-02-07 11:04:05 +00:00
/* The implementation below is more careful than strictly
necessary. Optimists may optimize it afterwards.
*/
while (local_level->sl_entry) {
register struct stack_entry *se = local_level->sl_entry;
register struct idf *idf = se->se_idf;
register struct def *def;
register struct sdef *sdef;
register struct tag *tag;
/* unlink it from the local stack level */
local_level->sl_entry = se->next;
free_stack_entry(se);
if (level == L_LOCAL && (def = idf->id_label)) {
unstack_label(idf);
free_def(def);
idf->id_label = 0;
}
1989-02-07 11:04:05 +00:00
while ((def = idf->id_def) && def->df_level >= level) {
/* unlink it from the def list under the idf block */
if (def->df_sc == REGISTER || def->df_sc == AUTO)
1989-02-07 11:04:05 +00:00
FreeLocal(def->df_address);
idf->id_def = def->next;
free_def(def);
update_ahead(idf);
}
while ( (sdef = idf->id_sdef)
&& sdef->sd_level >= level
) {
/* unlink it from the sdef list under the idf block */
idf->id_sdef = sdef->next;
free_sdef(sdef);
}
while ( (tag = idf->id_tag)
1989-02-07 11:04:05 +00:00
&& tag->tg_level >= level
) {
/* unlink it from the struct list under the idf block */
idf->id_tag = tag->next;
1989-02-07 11:04:05 +00:00
free_tag(tag);
}
}
/* Unlink the local stack level from the stack.
*/
lastlvl = local_level;
local_level = local_level->sl_previous;
if (level >= L_LOCAL) {
local_level->sl_max_block = lastlvl->sl_max_block;
}
free_stack_level(lastlvl);
local_level->sl_next = (struct stack_level *) 0;
level = local_level->sl_level;
#ifdef DEBUG
if (options['t'])
dumpidftab("after unstackidfs", 0);
#endif /* DEBUG */
1989-02-07 11:04:05 +00:00
}
void unstack_world(void)
1989-02-07 11:04:05 +00:00
{
/* The global level of identifiers is scanned, and final
decisions are taken about such issues as
extern/static/global and un/initialized.
Effects on the code generator: initialised variables
have already been encoded while the uninitialised ones
are not and have to be encoded at this moment.
*/
register struct stack_entry *se = local_level->sl_entry;
#ifdef LINT
1990-12-07 14:42:26 +00:00
lint_end_global(local_level);
#endif /* LINT */
1989-02-07 11:04:05 +00:00
#ifdef GEN_NM_LIST
1989-02-07 11:04:05 +00:00
open_name_list();
#endif /* GEN_NM_LIST */
1989-02-07 11:04:05 +00:00
while (se) {
register struct idf *idf = se->se_idf;
register struct def *def = idf->id_def;
if (!def) {
/* global selectors, etc. */
se = se->next;
continue;
}
#ifdef DEBUG
if (options['a']) {
char *symbol2str();
print("\"%s\", %s, %s, %s, %s\n",
idf->id_text,
(def->df_alloc == 0) ? "no alloc" :
(def->df_alloc == ALLOC_SEEN) ? "alloc seen" :
(def->df_alloc == ALLOC_DONE) ? "alloc done" :
"illegal alloc info",
symbol2str(def->df_sc),
def->df_initialized ? "init" : "no init",
def->df_used ? "used" : "not used");
}
#endif /* DEBUG */
if (def->df_sc == STATIC
&& def->df_type->tp_fund == FUNCTION
&& !def->df_initialized) {
1989-02-07 11:04:05 +00:00
/* orphaned static function */
warning("static function %s never defined, %s"
, idf->id_text
, "changed to extern");
1989-02-07 11:04:05 +00:00
def->df_sc = EXTERN;
}
if (def->df_alloc == ALLOC_SEEN
&& !def->df_initialized) {
1989-02-07 11:04:05 +00:00
/* space must be allocated */
bss(idf);
#ifdef GEN_NM_LIST
1989-02-07 11:04:05 +00:00
if (def->df_sc != STATIC)
namelist(idf->id_text); /* may be common */
#endif /* GEN_NM_LIST */
1989-02-07 11:04:05 +00:00
def->df_alloc = ALLOC_DONE; /* see Note below */
}
se = se->next;
}
/* Note:
df_alloc must be set to ALLOC_DONE because the idf entry
may occur several times in the list.
The reason for this is that the same name may be used
for different purposes on the same level, e.g.
struct s {int s;} s;
is a legal definition and contains 3 defining occurrences
of s.
Each definition has been entered into the identifier stack.
Although only one of them concerns a variable, we meet the
s 3 times when scanning the identifier stack.
*/
}
#ifdef GEN_NM_LIST
1989-02-07 11:04:05 +00:00
/* A list of potential common names is kept, to be fed to
an understanding loader. The list is written to a file
the name of which is nmlist. If nmlist == NULL, no name
list is generated.
*/
extern char *nmlist; /* BAH! -- main.c */
static File *nfp = 0;
void open_name_list(void)
1989-02-07 11:04:05 +00:00
{
if (nmlist && sys_open(nmlist, OP_WRITE, &nfp) == 0)
fatal("cannot create namelist %s", nmlist);
}
void namelist(char *nm)
1989-02-07 11:04:05 +00:00
{
if (nmlist) {
sys_write(nfp, nm, strlen(nm));
sys_write(nfp, "\n", 1);
}
}
#endif /* GEN_NM_LIST */