603 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			603 lines
		
	
	
	
		
			11 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".
 | |
|  *
 | |
|  * Author: Ceriel J.H. Jacobs
 | |
|  */
 | |
| 
 | |
| /* L E X I C A L   A N A L Y S E R   F O R   M O D U L A - 2 */
 | |
| 
 | |
| /* $Header$ */
 | |
| 
 | |
| #include	"debug.h"
 | |
| #include	"idfsize.h"
 | |
| #include	"numsize.h"
 | |
| #include	"strsize.h"
 | |
| 
 | |
| #include	<alloc.h>
 | |
| #include	<em_arith.h>
 | |
| #include	<em_label.h>
 | |
| #include	<assert.h>
 | |
| 
 | |
| #include	"LLlex.h"
 | |
| #include	"input.h"
 | |
| #include	"f_info.h"
 | |
| #include	"Lpars.h"
 | |
| #include	"class.h"
 | |
| #include	"idf.h"
 | |
| #include	"def.h"
 | |
| #include	"type.h"
 | |
| #include	"const.h"
 | |
| #include	"warning.h"
 | |
| 
 | |
| long str2long();
 | |
| 
 | |
| t_token		dot,
 | |
| 		aside;
 | |
| t_type		*toktype;
 | |
| int		idfsize = IDFSIZE;
 | |
| int		ForeignFlag;
 | |
| #ifdef DEBUG
 | |
| extern int	cntlines;
 | |
| #endif
 | |
| 
 | |
| extern char	options[];
 | |
| 
 | |
| STATIC
 | |
| SkipComment()
 | |
| {
 | |
| 	/*	Skip Modula-2 comments (* ... *).
 | |
| 		Note that comments may be nested (par. 3.5).
 | |
| 	*/
 | |
| 	register int ch, c;
 | |
| 	register int CommentLevel = 0;
 | |
| 
 | |
| 	LoadChar(ch);
 | |
| 	if (ch == '$') {
 | |
| 		LoadChar(ch);
 | |
| 		switch(ch) {
 | |
| 		case 'F':
 | |
| 			/* Foreign; This definition module has an
 | |
| 			   implementation in another language.
 | |
| 			   In this case, don't generate prefixes in front
 | |
| 			   of the names. Also, don't generate call to
 | |
| 			   initialization routine.
 | |
| 			*/
 | |
| 			ForeignFlag = D_FOREIGN;
 | |
| 			break;
 | |
| 		case 'U':
 | |
| 			inidf['_'] = 1;
 | |
| 			break;
 | |
| 		case 'A': /* Extra array bound checks, on or off */
 | |
| 		case 'R': /* Range checks, on or off */
 | |
| 		{
 | |
| 			int on_on_minus = ch == 'R';
 | |
| 			LoadChar(c);
 | |
| 			if (c == '-') {
 | |
| 				options[ch] = on_on_minus;
 | |
| 				break;
 | |
| 			}
 | |
| 			if (c == '+') {
 | |
| 				options[ch] = !on_on_minus;
 | |
| 				break;
 | |
| 			}
 | |
| 			ch = c;
 | |
| 		}
 | |
| 			/* fall through */
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	for (;;) {
 | |
| 		if (!(ch & 0200) && class(ch) == STNL) {
 | |
| 			LineNumber++;
 | |
| #ifdef DEBUG
 | |
| 			cntlines++;
 | |
| #endif
 | |
| 		}
 | |
| 		else if (ch == '(') {
 | |
| 			LoadChar(ch);
 | |
| 			if (ch == '*') CommentLevel++;
 | |
| 			else continue;
 | |
| 		}
 | |
| 		else if (ch == '*') {
 | |
| 			LoadChar(ch);
 | |
| 			if (ch == ')') {
 | |
| 				CommentLevel--;
 | |
| 				if (CommentLevel < 0) break;
 | |
| 			}
 | |
| 			else continue;
 | |
| 		}
 | |
| 		else if (ch == EOI) {
 | |
| 			lexerror("unterminated comment");
 | |
| 			PushBack();
 | |
| 			break;
 | |
| 		}
 | |
| 		LoadChar(ch);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| STATIC struct string *
 | |
| GetString(upto)
 | |
| {
 | |
| 	/*	Read a Modula-2 string, delimited by the character "upto".
 | |
| 	*/
 | |
| 	register int ch;
 | |
| 	register struct string *str = (struct string *)
 | |
| 			Malloc((unsigned) sizeof(struct string));
 | |
| 	register char *p;
 | |
| 	register int len;
 | |
| 	
 | |
| 	len = ISTRSIZE;
 | |
| 	str->s_str = p = Malloc((unsigned int) ISTRSIZE);
 | |
| 	while (LoadChar(ch), ch != upto)	{
 | |
| 		if (!(ch & 0200) && class(ch) == STNL)	{
 | |
| 			lexerror("newline in string");
 | |
| 			LineNumber++;
 | |
| #ifdef DEBUG
 | |
| 			cntlines++;
 | |
| #endif
 | |
| 			break;
 | |
| 		}
 | |
| 		if (ch == EOI)	{
 | |
| 			lexerror("end-of-file in string");
 | |
| 			break;
 | |
| 		}
 | |
| 		*p++ = ch;
 | |
| 		if (p - str->s_str == len)	{
 | |
| 			str->s_str = Realloc(str->s_str,
 | |
| 				(unsigned int) len + RSTRSIZE);
 | |
| 			p = str->s_str + len;
 | |
| 			len += RSTRSIZE;
 | |
| 		}
 | |
| 	}
 | |
| 	str->s_length = p - str->s_str;
 | |
| 	len = (str->s_length+(int)word_size) & ~((int)word_size-1);
 | |
| 	while (p - str->s_str < len) {
 | |
| 		*p++ = '\0';
 | |
| 	}
 | |
| 	str->s_str = Realloc(str->s_str, (unsigned) len);
 | |
| 	if (str->s_length == 0) str->s_length = 1;
 | |
| 	/* ??? string length at least 1 ??? */
 | |
| 	return str;
 | |
| }
 | |
| 
 | |
| static char *s_error = "illegal line directive";
 | |
| 
 | |
| STATIC int
 | |
| getch()
 | |
| {
 | |
| 	register int ch;
 | |
| 
 | |
| 	while (LoadChar(ch), (ch & 0200) && ch != EOI) {
 | |
| 		error("non-ascii '\\%03o' read", ch & 0377);
 | |
| 	}
 | |
| 	return ch;
 | |
| }
 | |
| 
 | |
| CheckForLineDirective()
 | |
| {
 | |
| 	register int ch = getch();
 | |
| 	register int	i = 0;
 | |
| 	char		buf[IDFSIZE];
 | |
| 	register char	*c = buf;
 | |
| 
 | |
| 
 | |
| 	if (ch != '#') {
 | |
| 		PushBack();
 | |
| 		return;
 | |
| 	}
 | |
| 	do {	/*
 | |
| 		 * Skip to next digit
 | |
| 		 * Do not skip newlines
 | |
| 		 */
 | |
| 		ch = getch();
 | |
| 		if (class(ch) == STNL || class(ch) == STEOI) {
 | |
| 			LineNumber++;
 | |
| 			error(s_error);
 | |
| 			return;
 | |
| 		}
 | |
| 	} while (class(ch) != STNUM);
 | |
| 	while (class(ch) == STNUM)  {
 | |
| 		i = i*10 + (ch - '0');
 | |
| 		ch = getch();
 | |
| 	}
 | |
| 	while (ch != '"' && class(ch) != STNL && class(ch) != STEOI)
 | |
| 		ch = getch();
 | |
| 	if (ch == '"') {
 | |
| 		c = buf;
 | |
| 		do {
 | |
| 			ch = getch();
 | |
| 			if (c < &buf[IDFSIZE]) *c++ = ch;
 | |
| 			if (class(ch) == STNL || class(ch) == STEOI) {
 | |
| 				LineNumber++;
 | |
| 				error(s_error);
 | |
| 				return;
 | |
| 			}
 | |
| 		} while (ch != '"');
 | |
| 		*--c = '\0';
 | |
| 		do {
 | |
| 			ch = getch();
 | |
| 		} while (class(ch) != STNL && class(ch) != STEOI);
 | |
| 		/*
 | |
| 		 * Remember the file name
 | |
| 		 */
 | |
| 		if (class(ch) == STNL && strcmp(FileName,buf)) {
 | |
| 			FileName = Salloc(buf,(unsigned) strlen(buf) + 1);
 | |
| 		}
 | |
| 	}
 | |
| 	if (class(ch) == STEOI) {
 | |
| 		error(s_error);
 | |
| 		return;
 | |
| 	}
 | |
| 	LineNumber = i;
 | |
| }
 | |
| 
 | |
| int
 | |
| LLlex()
 | |
| {
 | |
| 	/*	LLlex() is the Lexical Analyzer.
 | |
| 		The putting aside of tokens is taken into account.
 | |
| 	*/
 | |
| 	register t_token *tk = ˙
 | |
| 	char buf[(IDFSIZE > NUMSIZE ? IDFSIZE : NUMSIZE) + 2];
 | |
| 	register int ch, nch;
 | |
| 
 | |
| 	toktype = error_type;
 | |
| 
 | |
| 	if (ASIDE)	{	/* a token is put aside		*/
 | |
| 		*tk = aside;
 | |
| 		ASIDE = 0;
 | |
| 		return tk->tk_symb;
 | |
| 	}
 | |
| 
 | |
| again:
 | |
| 	ch = getch();
 | |
| 	tk->tk_lineno = LineNumber;
 | |
| 
 | |
| 	switch (class(ch))	{
 | |
| 
 | |
| 	case STNL:
 | |
| 		LineNumber++;
 | |
| #ifdef DEBUG
 | |
| 		cntlines++;
 | |
| #endif
 | |
| 		CheckForLineDirective();
 | |
| 		goto again;
 | |
| 
 | |
| 	case STSKIP:
 | |
| 		goto again;
 | |
| 
 | |
| 	case STGARB:
 | |
| 		if ((unsigned) ch - 040 < 0137)	{
 | |
| 			lexerror("garbage char %c", ch);
 | |
| 		}
 | |
| 		else	lexerror("garbage char \\%03o", ch);
 | |
| 		goto again;
 | |
| 
 | |
| 	case STSIMP:
 | |
| 		if (ch == '(')	{
 | |
| 			LoadChar(nch);
 | |
| 			if (nch == '*')	{
 | |
| 				SkipComment();
 | |
| 				goto again;
 | |
| 			}
 | |
| 			PushBack();
 | |
| 		}
 | |
| 		if (ch == '&') return tk->tk_symb = AND;
 | |
| 		if (ch == '~') return tk->tk_symb = NOT;
 | |
| 		return tk->tk_symb = ch;
 | |
| 
 | |
| 	case STCOMP:
 | |
| 		LoadChar(nch);
 | |
| 		switch (ch)	{
 | |
| 
 | |
| 		case '.':
 | |
| 			if (nch == '.')	{
 | |
| 				return tk->tk_symb = UPTO;
 | |
| 			}
 | |
| 			break;
 | |
| 
 | |
| 		case ':':
 | |
| 			if (nch == '=')	{
 | |
| 				return tk->tk_symb = BECOMES;
 | |
| 			}
 | |
| 			break;
 | |
| 
 | |
| 		case '<':
 | |
| 			if (nch == '=')	{
 | |
| 				return tk->tk_symb = LESSEQUAL;
 | |
| 			}
 | |
| 			if (nch == '>') {
 | |
| 				return tk->tk_symb = '#';
 | |
| 			}
 | |
| 			break;
 | |
| 
 | |
| 		case '>':
 | |
| 			if (nch == '=')	{
 | |
| 				return tk->tk_symb = GREATEREQUAL;
 | |
| 			}
 | |
| 			break;
 | |
| 
 | |
| 		default :
 | |
| 			crash("(LLlex, STCOMP)");
 | |
| 		}
 | |
| 		PushBack();
 | |
| 		return tk->tk_symb = ch;
 | |
| 
 | |
| 	case STIDF:
 | |
| 	{
 | |
| 		register char *tag = &buf[0];
 | |
| 		register t_idf *id;
 | |
| 
 | |
| 		do	{
 | |
| 			if (tag - buf < idfsize) *tag++ = ch;
 | |
| 			LoadChar(ch);
 | |
| 			if (ch == '_' && *(tag-1) == '_') {
 | |
| 				lexerror("an identifier may not contain two consecutive underscores");
 | |
| 			}
 | |
| 		} while(in_idf(ch));
 | |
| 
 | |
| 		PushBack();
 | |
| 		*tag = '\0';
 | |
| 		if (*(tag - 1) == '_') {
 | |
| 			lexerror("last character of an identifier may not be an underscore");
 | |
| 		}
 | |
| 
 | |
| 		tk->TOK_IDF = id = str2idf(buf, 1);
 | |
| 		return tk->tk_symb = id->id_reserved ? id->id_reserved : IDENT;
 | |
| 	}
 | |
| 
 | |
| 	case STSTR: {
 | |
| 		register struct string *str = GetString(ch);
 | |
| 
 | |
| 		if (str->s_length == 1) {
 | |
| 			tk->TOK_INT = *(str->s_str) & 0377;
 | |
| 			toktype = char_type;
 | |
| 			free(str->s_str);
 | |
| 			free((char *) str);
 | |
| 		}
 | |
| 		else {
 | |
| 			tk->tk_data.tk_str = str;
 | |
| 			if (! fit((arith)(str->s_length), (int) word_size)) {
 | |
| 				lexerror("string too long");
 | |
| 			}
 | |
| 			toktype = standard_type(T_STRING, 1, (arith)(str->s_length));
 | |
| 		}
 | |
| 		return tk->tk_symb = STRING;
 | |
| 		}
 | |
| 
 | |
| 	case STNUM:
 | |
| 	{
 | |
| 		/*	The problem arising with the "parsing" of a number
 | |
| 			is that we don't know the base in advance so we
 | |
| 			have to read the number with the help of a rather
 | |
| 			complex finite automaton.
 | |
| 		*/
 | |
| 		enum statetp {Oct,OptHex,Hex,Dec,OctEndOrHex,End,OptReal,Real};
 | |
| 		register enum statetp state;
 | |
| 		register int base = 8;
 | |
| 		register char *np = &buf[1];
 | |
| 					/* allow a '-' to be added	*/
 | |
| 
 | |
| 		buf[0] = '-';
 | |
| 		*np++ = ch;
 | |
| 		state = is_oct(ch) ? Oct : Dec;
 | |
| 		LoadChar(ch);
 | |
| 		for (;;) {
 | |
| 			switch(state) {
 | |
| 			case Oct:
 | |
| 				while (is_oct(ch))	{
 | |
| 					if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 					LoadChar(ch);
 | |
| 				}
 | |
| 				if (ch == 'B' || ch == 'C') {
 | |
| 					state = OctEndOrHex;
 | |
| 					break;
 | |
| 				}
 | |
| 				/* Fall Through */
 | |
| 			case Dec:
 | |
| 				base = 10;
 | |
| 				while (is_dig(ch))	{
 | |
| 					if (np < &buf[NUMSIZE]) {
 | |
| 						*np++ = ch;
 | |
| 					}
 | |
| 					LoadChar(ch);
 | |
| 				}
 | |
| 				if (ch == 'D') state = OptHex;
 | |
| 				else if (is_hex(ch)) state = Hex;
 | |
| 				else if (ch == '.') state = OptReal;
 | |
| 				else {
 | |
| 					state = End;
 | |
| 					if (ch == 'H') base = 16;
 | |
| 					else PushBack();
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case OptHex:
 | |
| 				LoadChar(ch);
 | |
| 				if (is_hex(ch)) {
 | |
| 					if (np < &buf[NUMSIZE]) *np++ = 'D';
 | |
| 					state = Hex;
 | |
| 				}
 | |
| 				else {
 | |
| 					state = End;
 | |
| 					ch = 'D';
 | |
| 					PushBack();
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case Hex:
 | |
| 				while (is_hex(ch))	{
 | |
| 					if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 					LoadChar(ch);
 | |
| 				}
 | |
| 				base = 16;
 | |
| 				state = End;
 | |
| 				if (ch != 'H') {
 | |
| 					lexerror("H expected after hex number");
 | |
| 					PushBack();
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case OctEndOrHex:
 | |
| 				if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 				LoadChar(ch);
 | |
| 				if (ch == 'H') {
 | |
| 					base = 16;
 | |
| 					state = End;
 | |
| 					break;
 | |
| 				}
 | |
| 				if (is_hex(ch)) {
 | |
| 					state = Hex;
 | |
| 					break;
 | |
| 				}
 | |
| 				PushBack();
 | |
| 				ch = *--np;
 | |
| 				*np++ = '\0';
 | |
| 				/* Fall through */
 | |
| 				
 | |
| 			case End: {
 | |
| 				int sgnswtch = 0;
 | |
| 
 | |
| 				*np = '\0';
 | |
| 				if (np >= &buf[NUMSIZE]) {
 | |
| 					tk->TOK_INT = 1;
 | |
| 					lexerror("constant too long");
 | |
| 				}
 | |
| 				else {
 | |
| 					np = &buf[1];
 | |
| 					while (*np == '0') np++;
 | |
| 					tk->TOK_INT = 0;
 | |
| 					while (*np) {
 | |
| 						arith old = tk->TOK_INT;
 | |
| 						int c;
 | |
| 
 | |
| 						if (is_dig(*np)) {
 | |
| 							c = *np++ - '0';
 | |
| 						}
 | |
| 						else {
 | |
| 							assert(is_hex(*np));
 | |
| 							c = *np++ - 'A' + 10;
 | |
| 						}
 | |
| 						tk->TOK_INT = tk->TOK_INT*base
 | |
| 							+ c;
 | |
| 						sgnswtch += (old < 0) ^
 | |
| 							    (tk->TOK_INT < 0);
 | |
| 					}
 | |
| 				}
 | |
| 				toktype = card_type;
 | |
| 				if (sgnswtch >= 2) {
 | |
| lexwarning(W_ORDINARY, "overflow in constant");
 | |
| 				}
 | |
| 				else if (ch == 'C' && base == 8) {
 | |
| 					toktype = char_type;
 | |
| 					if (sgnswtch != 0 || tk->TOK_INT>255) {
 | |
| lexwarning(W_ORDINARY, "character constant out of range");
 | |
| 					}
 | |
| 				}
 | |
| 				else if (ch == 'D' && base == 10) {
 | |
| 					if (sgnswtch != 0 ||
 | |
| 					    tk->TOK_INT > max_int[(int)long_size]) {
 | |
| lexwarning(W_ORDINARY, "overflow in constant");
 | |
| 					}
 | |
| 					toktype = longint_type;
 | |
| 				}
 | |
| 				else if (sgnswtch == 0 &&
 | |
| 					 tk->TOK_INT<=max_int[(int)int_size]) {
 | |
| 					toktype = intorcard_type;
 | |
| 				}
 | |
| 				else if (! chk_bounds(tk->TOK_INT,
 | |
| 						      full_mask[(int)int_size],
 | |
| 						      T_CARDINAL)) {
 | |
| lexwarning(W_ORDINARY, "overflow in constant");
 | |
| 				}
 | |
| 				return tk->tk_symb = INTEGER;
 | |
| 				}
 | |
| 
 | |
| 			case OptReal:
 | |
| 				/*	The '.' could be the first of the '..'
 | |
| 					token. At this point, we need a
 | |
| 					look-ahead of two characters.
 | |
| 				*/
 | |
| 				LoadChar(ch);
 | |
| 				if (ch == '.') {
 | |
| 					/*	Indeed the '..' token
 | |
| 					*/
 | |
| 					PushBack();
 | |
| 					PushBack();
 | |
| 					state = End;
 | |
| 					base = 10;
 | |
| 					break;
 | |
| 				}
 | |
| 				state = Real;
 | |
| 				break;
 | |
| 			}
 | |
| 			if (state == Real) break;
 | |
| 		}
 | |
| 
 | |
| 		/* a real real constant */
 | |
| 		if (np < &buf[NUMSIZE]) *np++ = '.';
 | |
| 
 | |
| 		toktype = real_type;
 | |
| 
 | |
| 		while (is_dig(ch)) {
 | |
| 			/* 	Fractional part
 | |
| 			*/
 | |
| 			if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 			LoadChar(ch);
 | |
| 		}
 | |
| 
 | |
| 		if (ch == 'D') {
 | |
| 			toktype = longreal_type;
 | |
| 			LoadChar(ch);
 | |
| 			if (ch == '+' || ch == '-' || is_dig(ch)) {
 | |
| 				ch = 'E';
 | |
| 				PushBack();
 | |
| 			}
 | |
| 		}
 | |
| 		if (ch == 'E') {
 | |
| 			/*	Scale factor
 | |
| 			*/
 | |
| 			if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 			LoadChar(ch);
 | |
| 			if (ch == '+' || ch == '-') {
 | |
| 				/*	Signed scalefactor
 | |
| 				*/
 | |
| 				if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 				LoadChar(ch);
 | |
| 			}
 | |
| 			if (is_dig(ch)) {
 | |
| 				do {
 | |
| 					if (np < &buf[NUMSIZE]) *np++ = ch;
 | |
| 					LoadChar(ch);
 | |
| 				} while (is_dig(ch));
 | |
| 			}
 | |
| 			else {
 | |
| 				lexerror("bad scale factor");
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		*np++ = '\0';
 | |
| 		PushBack();
 | |
| 
 | |
| 		if (np >= &buf[NUMSIZE]) {
 | |
| 			tk->TOK_REL = Salloc("0.0", 5);
 | |
| 			lexerror("real constant too long");
 | |
| 		}
 | |
| 		else	tk->TOK_REL = Salloc(buf, (unsigned) (np - buf)) + 1;
 | |
| 		return tk->tk_symb = REAL;
 | |
| 
 | |
| 		/*NOTREACHED*/
 | |
| 	}
 | |
| 
 | |
| 	case STEOI:
 | |
| 		return tk->tk_symb = -1;
 | |
| 
 | |
| 	case STCHAR:
 | |
| 	default:
 | |
| 		crash("(LLlex) Impossible character class");
 | |
| 		/*NOTREACHED*/
 | |
| 	}
 | |
| 	/*NOTREACHED*/
 | |
| }
 |