ack/lang/m2/comp/scope.C

242 lines
5.1 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-03-26 15:11:02 +00:00
/* S C O P E M E C H A N I S M */
/* $Header$ */
1986-05-01 19:06:53 +00:00
#include "debug.h"
1986-03-26 15:11:02 +00:00
#include <assert.h>
#include <alloc.h>
1986-03-27 17:37:41 +00:00
#include <em_arith.h>
#include <em_label.h>
1986-04-22 22:36:16 +00:00
1986-03-27 17:37:41 +00:00
#include "LLlex.h"
#include "idf.h"
1986-03-26 15:11:02 +00:00
#include "scope.h"
1986-03-27 17:37:41 +00:00
#include "type.h"
#include "def.h"
1986-04-08 18:15:46 +00:00
#include "node.h"
1986-04-22 22:36:16 +00:00
t_scope *PervasiveScope;
t_scopelist *CurrVis, *GlobalVis;
1986-05-16 17:15:36 +00:00
extern int proclevel;
static t_scopelist *PervVis;
1986-11-05 14:33:00 +00:00
extern char options[];
1986-03-26 15:11:02 +00:00
1986-10-06 20:36:30 +00:00
/* STATICALLOCDEF "scope" 10 */
1986-03-26 15:11:02 +00:00
1986-10-06 20:36:30 +00:00
/* STATICALLOCDEF "scopelist" 10 */
1986-04-28 18:06:58 +00:00
1986-04-15 17:51:53 +00:00
open_scope(scopetype)
1986-03-26 15:11:02 +00:00
{
1986-04-03 00:44:39 +00:00
/* Open a scope that is either open (automatic imports) or closed.
*/
register t_scope *sc = new_scope();
register t_scopelist *ls = new_scopelist();
1986-05-30 18:48:00 +00:00
1986-04-15 17:51:53 +00:00
assert(scopetype == OPENSCOPE || scopetype == CLOSEDSCOPE);
1986-05-30 18:48:00 +00:00
1986-04-15 17:51:53 +00:00
sc->sc_scopeclosed = scopetype == CLOSEDSCOPE;
1986-05-16 17:15:36 +00:00
sc->sc_level = proclevel;
1986-10-06 20:36:30 +00:00
ls->sc_scope = sc;
ls->sc_encl = CurrVis;
1986-04-28 18:06:58 +00:00
if (scopetype == OPENSCOPE) {
1987-07-16 19:51:40 +00:00
ls->sc_next = ls->sc_encl;
1986-03-26 15:11:02 +00:00
}
1987-07-16 19:51:40 +00:00
else ls->sc_next = PervVis;
1986-04-28 18:06:58 +00:00
CurrVis = ls;
1986-03-26 15:11:02 +00:00
}
t_scope *
open_and_close_scope(scopetype)
{
t_scope *sc;
open_scope(scopetype);
sc = CurrentScope;
1987-05-18 15:57:33 +00:00
close_scope(0);
return sc;
}
1986-06-06 02:22:09 +00:00
InitScope()
1986-03-26 15:11:02 +00:00
{
register t_scope *sc = new_scope();
register t_scopelist *ls = new_scopelist();
1986-03-26 15:11:02 +00:00
1986-04-15 17:51:53 +00:00
sc->sc_scopeclosed = 0;
1986-04-03 00:44:39 +00:00
sc->sc_def = 0;
1986-05-16 17:15:36 +00:00
sc->sc_level = proclevel;
1986-04-15 17:51:53 +00:00
PervasiveScope = sc;
1987-07-16 19:51:40 +00:00
ls->sc_next = 0;
1986-04-28 18:06:58 +00:00
ls->sc_encl = 0;
ls->sc_scope = PervasiveScope;
PervVis = ls;
CurrVis = ls;
1986-03-27 17:37:41 +00:00
}
1986-06-17 12:04:05 +00:00
STATIC
1986-04-15 17:51:53 +00:00
chk_proc(df)
register t_def *df;
1986-04-08 18:15:46 +00:00
{
1986-04-15 17:51:53 +00:00
/* Called at scope closing. Check all definitions, and if one
1986-10-06 20:36:30 +00:00
is a D_PROCHEAD, the procedure was not defined.
1986-12-01 10:06:53 +00:00
Also check that hidden types are defined.
1986-04-12 02:21:24 +00:00
*/
1986-04-15 17:51:53 +00:00
while (df) {
1986-12-01 10:06:53 +00:00
if (df->df_kind == D_HIDDEN) {
error("hidden type \"%s\" not declared",
df->df_idf->id_text);
}
else if (df->df_kind == D_PROCHEAD) {
1986-04-15 17:51:53 +00:00
/* A not defined procedure
*/
1986-10-06 20:36:30 +00:00
error("procedure \"%s\" not defined",
df->df_idf->id_text);
1986-04-15 17:51:53 +00:00
FreeNode(df->for_node);
}
df = df->df_nextinscope;
}
}
1986-06-17 12:04:05 +00:00
STATIC
1986-04-15 17:51:53 +00:00
chk_forw(pdf)
t_def **pdf;
1986-04-15 17:51:53 +00:00
{
/* Called at scope close. Look for all forward definitions and
if the scope was a closed scope, give an error message for
them, and otherwise move them to the enclosing scope.
*/
register t_def *df;
1986-10-06 20:36:30 +00:00
while (df = *pdf) {
1986-11-26 16:40:45 +00:00
if (df->df_kind == D_FORWTYPE) {
register t_def *df1 = df;
register t_node *nd = df->df_forw_node;
*pdf = df->df_nextinscope;
RemoveFromIdList(df);
df = lookfor(nd, CurrVis, 1, 0);
if (! df->df_kind & (D_ERROR|D_FTYPE|D_TYPE)) {
node_error(nd, "\"%s\" is not a type", df1->df_idf->id_text);
}
while (nd) {
1987-07-16 19:51:40 +00:00
nd->nd_type->tp_next = df->df_type;
nd = nd->nd_right;
}
FreeNode(df1->df_forw_node);
free_def(df1);
1986-12-01 10:06:53 +00:00
continue;
1986-11-26 16:40:45 +00:00
}
else if (df->df_kind == D_FTYPE) {
register t_node *nd = df->df_forw_node;
1986-11-26 16:40:45 +00:00
df->df_kind = D_TYPE;
while (nd) {
1987-07-16 19:51:40 +00:00
nd->nd_type->tp_next = df->df_type;
nd = nd->nd_right;
}
FreeNode(df->df_forw_node);
1986-11-26 16:40:45 +00:00
}
else if (df->df_kind & (D_FORWARD|D_FORWMODULE)) {
1986-04-15 17:51:53 +00:00
/* These definitions must be found in
the enclosing closed scope, which of course
may be the scope that is now closed!
*/
if (scopeclosed(CurrentScope)) {
/* Indeed, the scope was a closed
scope, so give error message
*/
1986-11-26 16:40:45 +00:00
node_error(df->for_node, "identifier \"%s\" not declared",
1986-10-06 20:36:30 +00:00
df->df_idf->id_text);
FreeNode(df->for_node);
1986-04-12 02:21:24 +00:00
}
1986-10-06 20:36:30 +00:00
else {
/* This scope was an open scope.
1986-04-15 17:51:53 +00:00
Maybe the definitions are in the
enclosing scope?
1986-04-12 02:21:24 +00:00
*/
register t_scopelist *ls =
1986-10-06 20:36:30 +00:00
nextvisible(CurrVis);
t_def *df1 = df->df_nextinscope;
1986-10-06 20:36:30 +00:00
if (df->df_kind == D_FORWMODULE) {
1987-07-16 19:51:40 +00:00
df->for_vis->sc_next = ls;
1986-04-12 02:21:24 +00:00
}
1986-10-06 20:36:30 +00:00
df->df_nextinscope = ls->sc_scope->sc_def;
ls->sc_scope->sc_def = df;
df->df_scope = ls->sc_scope;
1986-04-15 17:51:53 +00:00
*pdf = df1;
1986-10-06 20:36:30 +00:00
continue;
1986-04-12 02:21:24 +00:00
}
}
1986-10-06 20:36:30 +00:00
pdf = &df->df_nextinscope;
1986-04-12 02:21:24 +00:00
}
1986-04-08 18:15:46 +00:00
}
1986-04-17 09:28:09 +00:00
Reverse(pdf)
t_def **pdf;
1986-04-17 09:28:09 +00:00
{
/* Reverse the order in the list of definitions in a scope.
This is neccesary because this list is built in reverse.
1986-04-21 17:27:06 +00:00
Also, while we're at it, remove uninteresting definitions
1986-06-17 12:04:05 +00:00
from this list.
1986-04-17 09:28:09 +00:00
*/
register t_def *df, *df1;
1986-06-17 12:04:05 +00:00
#define INTERESTING D_MODULE|D_PROCEDURE|D_PROCHEAD|D_VARIABLE
1986-04-17 09:28:09 +00:00
df = 0;
df1 = *pdf;
1986-04-21 17:27:06 +00:00
while (df1) {
1986-10-06 20:36:30 +00:00
if (df1->df_kind & INTERESTING) {
t_def *prev = df;
1986-10-06 20:36:30 +00:00
df = df1;
1986-04-21 17:27:06 +00:00
df1 = df1->df_nextinscope;
1986-10-06 20:36:30 +00:00
df->df_nextinscope = prev;
1986-04-21 17:27:06 +00:00
}
1986-10-06 20:36:30 +00:00
else df1 = df1->df_nextinscope;
1986-04-17 09:28:09 +00:00
}
1986-10-06 20:36:30 +00:00
*pdf = df;
1986-04-17 09:28:09 +00:00
}
1986-04-15 17:51:53 +00:00
close_scope(flag)
1987-05-18 15:57:33 +00:00
register int flag;
1986-04-15 17:51:53 +00:00
{
/* Close a scope. If "flag" is set, check for forward declarations,
either POINTER declarations, or EXPORTs, or forward references
to MODULES
*/
register t_scope *sc = CurrentScope;
1986-04-15 17:51:53 +00:00
assert(sc != 0);
FreeNode(sc->sc_end);
sc->sc_end = dot2leaf(Link);
1986-04-15 17:51:53 +00:00
if (flag) {
DO_DEBUG(options['S'],(print("List of definitions in currently ended scope:\n"), DumpScope(sc->sc_def)));
1986-04-15 17:51:53 +00:00
if (flag & SC_CHKPROC) chk_proc(sc->sc_def);
if (flag & SC_CHKFORW) chk_forw(&(sc->sc_def));
1986-04-22 22:36:16 +00:00
if (flag & SC_REVERSE) Reverse(&(sc->sc_def));
1986-04-15 17:51:53 +00:00
}
1986-04-28 18:06:58 +00:00
CurrVis = enclosing(CurrVis);
1986-04-15 17:51:53 +00:00
}
#ifdef DEBUG
DumpScope(df)
register t_def *df;
1986-04-15 17:51:53 +00:00
{
while (df) {
PrDef(df);
df = df->df_nextinscope;
}
}
#endif