245 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			245 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
| /*
 | |
|  * (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
 | |
|  */
 | |
| 
 | |
| /* O V E R A L L   S T R U C T U R E */
 | |
| 
 | |
| /* $Header$ */
 | |
| 
 | |
| {
 | |
| #include	"debug.h"
 | |
| 
 | |
| #include	<alloc.h>
 | |
| #include	<em_arith.h>
 | |
| #include	<em_label.h>
 | |
| 
 | |
| #include	"main.h"
 | |
| #include	"idf.h"
 | |
| #include	"LLlex.h"
 | |
| #include	"scope.h"
 | |
| #include	"def.h"
 | |
| #include	"type.h"
 | |
| #include	"node.h"
 | |
| #include	"f_info.h"
 | |
| #include	"warning.h"
 | |
| 
 | |
| }
 | |
| /*
 | |
| 	The grammar as given by Wirth is already almost LL(1); the
 | |
| 	main problem is that the full form of a qualified designator
 | |
| 	may be:
 | |
| 		[ module_ident '.' ]* IDENT [ '.' field_ident ]*
 | |
| 	which is quite confusing to an LL(1) parser. Rather than
 | |
| 	resorting to context-sensitive techniques, I have decided
 | |
| 	to render this as:
 | |
| 		IDENT [ '.' IDENT ]*
 | |
| 	on the grounds that it is quite natural to consider the first
 | |
| 	IDENT to be the name of the object and regard the others as
 | |
| 	field identifiers.
 | |
| */
 | |
| 
 | |
| %lexical LLlex;
 | |
| 
 | |
| %start	CompUnit, CompilationUnit;
 | |
| %start	DefModule, DefinitionModule;
 | |
| 
 | |
| ModuleDeclaration
 | |
| {
 | |
| 	register struct def *df;
 | |
| 	struct node *exportlist = 0;
 | |
| 	int qualified;
 | |
| } :
 | |
| 	MODULE IDENT	{ df = DefineLocalModule(dot.TOK_IDF); }
 | |
| 	priority(df)
 | |
| 	';'
 | |
| 	import(1)*
 | |
| 	export(&qualified, &exportlist)?
 | |
| 	block(&(df->mod_body))
 | |
| 	IDENT		{ if (exportlist) {
 | |
| 				EnterExportList(exportlist, qualified);
 | |
| 			  }
 | |
| 			  close_scope(SC_CHKFORW|SC_CHKPROC|SC_REVERSE);
 | |
| 			  match_id(df->df_idf, dot.TOK_IDF);
 | |
| 			}
 | |
| ;
 | |
| 
 | |
| priority(register struct def *df;):
 | |
| 	[
 | |
| 		'[' ConstExpression(&(df->mod_priority)) ']'
 | |
| 			{ if (!(df->mod_priority->nd_type->tp_fund &
 | |
| 				T_CARDINAL)) {
 | |
| 				node_error(df->mod_priority,
 | |
| 					   "illegal priority");
 | |
| 			  }
 | |
| 			}
 | |
| 	|
 | |
| 			{ df->mod_priority = 0; }
 | |
| 	]
 | |
| ;
 | |
| 
 | |
| export(int *QUALflag; struct node **ExportList;):
 | |
| 	EXPORT
 | |
| 	[
 | |
| 		QUALIFIED
 | |
| 			{ *QUALflag = D_QEXPORTED; }
 | |
| 	|
 | |
| 			{ *QUALflag = D_EXPORTED; }
 | |
| 	]
 | |
| 	IdentList(ExportList) ';'
 | |
| ;
 | |
| 
 | |
| import(int local;)
 | |
| {
 | |
| 	struct node *ImportList;
 | |
| 	register struct node *FromId = 0;
 | |
| 	register struct def *df;
 | |
| 	extern struct def *GetDefinitionModule();
 | |
| } :
 | |
| 	[ FROM
 | |
| 	  IDENT		{ FromId = MkLeaf(Name, &dot);
 | |
| 			  if (local) df = lookfor(FromId,enclosing(CurrVis),0);
 | |
| 			  else df = GetDefinitionModule(dot.TOK_IDF, 1);
 | |
| 			}
 | |
| 	]?
 | |
| 	IMPORT IdentList(&ImportList) ';'
 | |
| 	/*
 | |
| 	   When parsing a global module, this is the place where we must
 | |
| 	   read already compiled definition modules.
 | |
| 	   If the FROM clause is present, the identifier in it is a module
 | |
| 	   name, otherwise the names in the import list are module names.
 | |
| 	*/
 | |
| 			{ if (FromId) {
 | |
| 				EnterFromImportList(ImportList, df, FromId);
 | |
| 			  }
 | |
| 			  else EnterImportList(ImportList, local);
 | |
| 			}
 | |
| ;
 | |
| 
 | |
| DefinitionModule
 | |
| {
 | |
| 	register struct def *df;
 | |
| 	struct node *exportlist;
 | |
| 	int dummy;
 | |
| 	extern struct idf *DefId;
 | |
| 	extern int ForeignFlag;
 | |
| } :
 | |
| 	DEFINITION
 | |
| 	MODULE IDENT	{ df = define(dot.TOK_IDF, GlobalScope, D_MODULE);
 | |
| 			  df->df_flags |= D_BUSY;
 | |
| 			  df->df_flags |= ForeignFlag;
 | |
| 			  if (!Defined) Defined = df;
 | |
| 		  	  CurrentScope->sc_definedby = df;
 | |
| 			  if (df->df_idf != DefId) {
 | |
| 				error("DEFINITION MODULE name is \"%s\", not \"%s\"",
 | |
| 					df->df_idf->id_text, DefId->id_text);
 | |
| 			  }
 | |
| 			  CurrentScope->sc_name = df->df_idf->id_text;
 | |
| 			  df->mod_vis = CurrVis;
 | |
| 			  df->df_type = standard_type(T_RECORD, 1, (arith) 1);
 | |
| 			  df->df_type->rec_scope = df->mod_vis->sc_scope;
 | |
| 			  DefinitionModule++;
 | |
| 			}
 | |
| 	';'
 | |
| 	import(0)* 
 | |
| 	[
 | |
| 		export(&dummy, &exportlist)
 | |
| 		/*	New Modula-2 does not have export lists in definition
 | |
| 			modules. Issue a warning.
 | |
| 		*/
 | |
| 			{ 
 | |
| node_warning(exportlist, W_OLDFASHIONED, "export list in definition module ignored");
 | |
| 				FreeNode(exportlist);
 | |
| 			}
 | |
| 	|
 | |
| 		/* empty */
 | |
| 	]
 | |
| 	definition* END IDENT
 | |
| 			{ register struct def *df1 = CurrentScope->sc_def;
 | |
| 			  while (df1) {
 | |
| 				/* Make all definitions "QUALIFIED EXPORT" */
 | |
| 				df1->df_flags |= D_QEXPORTED;
 | |
| 				df1 = df1->df_nextinscope;
 | |
| 			  }
 | |
| 			  DefinitionModule--;
 | |
| 			  match_id(df->df_idf, dot.TOK_IDF);
 | |
| 			  df->df_flags &= ~D_BUSY;
 | |
| 			}
 | |
| 	'.'
 | |
| ;
 | |
| 
 | |
| definition
 | |
| {
 | |
| 	register struct def *df;
 | |
| 	struct def *dummy;
 | |
| } :
 | |
| 	CONST [ %persistent ConstantDeclaration ';' ]*
 | |
| |
 | |
| 	TYPE
 | |
| 	[ %persistent
 | |
| 	  IDENT 	{ df = define(dot.TOK_IDF, CurrentScope, D_TYPE); }
 | |
| 	  [ '=' type(&(df->df_type))
 | |
| 	  | /* empty */
 | |
| 	    /*
 | |
| 	       Here, the exported type has a hidden implementation.
 | |
| 	       The export is said to be opaque.
 | |
| 	       It is restricted to pointer types.
 | |
| 	    */
 | |
| 	    		{ df->df_kind = D_HIDDEN;
 | |
| 			  df->df_type = construct_type(T_HIDDEN, NULLTYPE);
 | |
| 			}
 | |
| 	  ]
 | |
| 	  ';'
 | |
| 	]*
 | |
| |
 | |
| 	VAR [ %persistent VariableDeclaration ';' ]*
 | |
| |
 | |
| 	ProcedureHeading(&dummy, D_PROCHEAD)
 | |
| 	';'
 | |
| ;
 | |
| 
 | |
| ProgramModule
 | |
| {
 | |
| 	extern struct def *GetDefinitionModule();
 | |
| 	register struct def *df;
 | |
| } :
 | |
| 	MODULE
 | |
| 	IDENT	{ if (state == IMPLEMENTATION) {
 | |
| 			df = GetDefinitionModule(dot.TOK_IDF, 0);
 | |
| 			CurrVis = df->mod_vis;
 | |
| 			RemoveImports(&(CurrentScope->sc_def));
 | |
| 		  }
 | |
| 		  else {
 | |
| 			Defined = df = define(dot.TOK_IDF, GlobalScope, D_MODULE);
 | |
| 			open_scope(CLOSEDSCOPE);
 | |
| 			df->mod_vis = CurrVis;
 | |
| 			CurrentScope->sc_name = "_M2M";
 | |
| 		  	CurrentScope->sc_definedby = df;
 | |
| 		  }
 | |
| 		}
 | |
| 	priority(df)
 | |
| 	';' import(0)*
 | |
| 	block(&(df->mod_body)) IDENT
 | |
| 		{ close_scope(SC_CHKFORW|SC_CHKPROC|SC_REVERSE);
 | |
| 		  match_id(df->df_idf, dot.TOK_IDF);
 | |
| 		}
 | |
| 	'.'
 | |
| ;
 | |
| 
 | |
| Module:
 | |
| 	DEFINITION
 | |
| 				{ fatal("Compiling a definition module"); }
 | |
| |	%default
 | |
| 	[
 | |
| 		IMPLEMENTATION	{ state = IMPLEMENTATION; }
 | |
| 	|
 | |
| 		/* empty */	{ state = PROGRAM; }
 | |
| 	]
 | |
| 	ProgramModule
 | |
| ;
 | |
| 
 | |
| CompilationUnit:
 | |
| 	Module
 | |
| ;
 |