423 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			423 lines
		
	
	
	
		
			8.1 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".
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 *  L L G E N
 | 
						|
 *
 | 
						|
 *  An Extended LL(1) Parser Generator
 | 
						|
 *
 | 
						|
 *  Author : Ceriel J.H. Jacobs
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * tokens.g
 | 
						|
 * Defines the tokens for the grammar of LLgen.
 | 
						|
 * The lexical analyser and LLmessage are also included here. 
 | 
						|
 */
 | 
						|
 | 
						|
{
 | 
						|
# include "types.h"
 | 
						|
# include "io.h"
 | 
						|
# include "extern.h"
 | 
						|
# include "assert.h"
 | 
						|
# include "cclass.h"
 | 
						|
 | 
						|
# ifndef NORCSID
 | 
						|
static string	rcsidc = "$Header$";
 | 
						|
# endif
 | 
						|
 | 
						|
/* Here are defined : */
 | 
						|
extern int	scanner();
 | 
						|
extern 		LLmessage();
 | 
						|
extern int	input();
 | 
						|
extern		unput();
 | 
						|
extern		skipcomment();
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
STATIC		linedirective();
 | 
						|
# endif
 | 
						|
STATIC string	cpy();
 | 
						|
STATIC string	vallookup();
 | 
						|
}
 | 
						|
/* Classes */
 | 
						|
 | 
						|
%token  C_IDENT ;	/* lextoken.t_string contains the identifier read */
 | 
						|
%token	C_NUMBER ;	/* lextoken.t_num contains the number read */
 | 
						|
%token	C_LITERAL ;	/* lextoken.t_string contains the literal read */
 | 
						|
 | 
						|
/* Keywords */
 | 
						|
 | 
						|
%token	C_TOKEN ;
 | 
						|
%token	C_START ;
 | 
						|
%token	C_IF ;
 | 
						|
%token	C_WHILE ;
 | 
						|
%token	C_PERSISTENT ;
 | 
						|
%token	C_FIRST ;
 | 
						|
%token	C_LEXICAL ;
 | 
						|
%token	C_ONERROR ;
 | 
						|
%token	C_AVOID ;
 | 
						|
%token	C_PREFER ;
 | 
						|
%token	C_DEFAULT ;
 | 
						|
 | 
						|
%lexical scanner ;
 | 
						|
 | 
						|
{
 | 
						|
 | 
						|
/*
 | 
						|
 * Structure for a keyword
 | 
						|
 */
 | 
						|
 | 
						|
typedef struct keyword {
 | 
						|
	string	w_word;
 | 
						|
	int	w_value;
 | 
						|
} t_keyw, *p_keyw;
 | 
						|
 | 
						|
/*
 | 
						|
 * The list of keywords, the most often used keywords come first.
 | 
						|
 * Linear search is used, as there are not many keywords
 | 
						|
 */
 | 
						|
 | 
						|
static t_keyw resword[] = {
 | 
						|
	{ "token",	C_TOKEN	},
 | 
						|
	{ "avoid",	C_AVOID	},
 | 
						|
	{ "prefer",	C_PREFER	},
 | 
						|
	{ "persistent", C_PERSISTENT	},
 | 
						|
	{ "default",	C_DEFAULT	},
 | 
						|
	{ "if",		C_IF	},
 | 
						|
	{ "while",	C_WHILE	},
 | 
						|
	{ "first",	C_FIRST	},
 | 
						|
	{ "start",	C_START	},
 | 
						|
	{ "lexical",	C_LEXICAL	},
 | 
						|
	{ "onerror",	C_ONERROR	},
 | 
						|
	{ 0,		0	}
 | 
						|
};
 | 
						|
 | 
						|
static t_token	savedtok;	/* to save lextoken in case of an insertion */
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
static  int	nostartline;	/* = 0 if at the start of a line */
 | 
						|
# endif
 | 
						|
 | 
						|
scanner() {
 | 
						|
	/*
 | 
						|
	 * Lexical analyser, what else
 | 
						|
	 */
 | 
						|
	register int	ch;		/* Current char */
 | 
						|
	register char *p = ltext;
 | 
						|
	int		reserved = 0;	/* reserved word? */
 | 
						|
	char		*max = <ext[LTEXTSZ - 1];
 | 
						|
 | 
						|
	if (savedtok.t_tokno) {
 | 
						|
				/* A token has been inserted.
 | 
						|
				 * Now deliver the last lextoken again
 | 
						|
				 */
 | 
						|
		lextoken = savedtok;
 | 
						|
		savedtok.t_tokno = 0;
 | 
						|
		return lextoken.t_tokno;
 | 
						|
	}
 | 
						|
	for (;;) {
 | 
						|
		ch = input();
 | 
						|
		if (ch == EOF) return ch;
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
		if (ch == '#' && !nostartline) {
 | 
						|
			linedirective();
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
# endif
 | 
						|
		switch(c_class[ch]) {
 | 
						|
		  case ISLIT :
 | 
						|
			for (;;) {
 | 
						|
				ch = input();
 | 
						|
				if (ch == '\n' || ch == EOF) {
 | 
						|
					error(linecount,"Missing '");
 | 
						|
					break;
 | 
						|
				}
 | 
						|
				if (ch == '\'') break;
 | 
						|
				if (ch == '\\') {
 | 
						|
					*p++ = ch;
 | 
						|
					ch = input();
 | 
						|
				}
 | 
						|
				*p++ = ch;
 | 
						|
				if (p > max) p--;
 | 
						|
			}
 | 
						|
			*p = '\0';
 | 
						|
			lextoken.t_string = ltext;
 | 
						|
			return C_LITERAL;
 | 
						|
		  case ISCOM :
 | 
						|
			skipcomment(0);
 | 
						|
			/* Fall through */
 | 
						|
		  case ISSPA :
 | 
						|
			continue;
 | 
						|
		  case ISDIG : {
 | 
						|
			register i = 0;
 | 
						|
			do {
 | 
						|
				i = 10 * i + (ch - '0');
 | 
						|
				ch= input();
 | 
						|
			} while (c_class[ch] == ISDIG);
 | 
						|
			lextoken.t_num = i;
 | 
						|
			unput(ch);
 | 
						|
			return C_NUMBER; }
 | 
						|
		  default:
 | 
						|
			return ch;
 | 
						|
		  case ISKEY :
 | 
						|
			reserved = 1;
 | 
						|
			ch = input();
 | 
						|
			/* Fall through */
 | 
						|
		  case ISLET :
 | 
						|
			do {
 | 
						|
				if (reserved && ch >= 'A' && ch <= 'Z') {
 | 
						|
					ch += 'a' - 'A';
 | 
						|
				}
 | 
						|
				*p++ = ch;
 | 
						|
				if (p > max) p--;
 | 
						|
				ch = input();
 | 
						|
			} while (c_class[ch] == ISDIG || c_class[ch] == ISLET);
 | 
						|
			unput(ch);
 | 
						|
			*p = '\0';
 | 
						|
			if (reserved) {	/*
 | 
						|
			 		 * Now search for the keyword
 | 
						|
					 */
 | 
						|
				register p_keyw w;
 | 
						|
 | 
						|
				w = resword;
 | 
						|
				while (w->w_word) {
 | 
						|
					if (! strcmp(ltext,w->w_word)) {
 | 
						|
						/*
 | 
						|
						 * Return token number.
 | 
						|
						 */
 | 
						|
						return w->w_value;
 | 
						|
					}
 | 
						|
					w++;
 | 
						|
				}
 | 
						|
				error(linecount,"Illegal reserved word");
 | 
						|
			}
 | 
						|
			lextoken.t_string = ltext;
 | 
						|
			return C_IDENT;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int	backupc;	/* for unput() */
 | 
						|
static int	nonline;	/* = 1 if last char read was a newline */
 | 
						|
 | 
						|
input() {
 | 
						|
	/*
 | 
						|
	 * Low level input routine, used by all other input routines
 | 
						|
	 */
 | 
						|
	register	c;
 | 
						|
 | 
						|
        if (c = backupc) {
 | 
						|
			/* Last char was "unput()". Deliver it again
 | 
						|
			 */
 | 
						|
		backupc = 0;
 | 
						|
                return c;
 | 
						|
	}
 | 
						|
	if ((c = getc(finput)) == EOF) return c;
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
	nostartline = 1;
 | 
						|
# endif
 | 
						|
	if (!nonline) {
 | 
						|
		linecount++;
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
		nostartline = 0;
 | 
						|
# endif
 | 
						|
		nonline = 1;
 | 
						|
	}
 | 
						|
	if (c == '\n') nonline = 0;
 | 
						|
	return c;
 | 
						|
}
 | 
						|
 | 
						|
unput(c) {
 | 
						|
	/*
 | 
						|
	 * "unread" c
 | 
						|
	 */
 | 
						|
	backupc = c;
 | 
						|
}
 | 
						|
 | 
						|
skipcomment(flag) {
 | 
						|
	/*
 | 
						|
	 * Skip comment. If flag != 0, the comment is inside a fragment
 | 
						|
	 * of C-code, so the newlines in it must be copied to enable the
 | 
						|
	 * C-compiler to keep a correct line count
 | 
						|
	 */
 | 
						|
	register int	ch;
 | 
						|
	int		saved;	/* line count on which comment starts */
 | 
						|
 | 
						|
	saved = linecount;
 | 
						|
	if (input() != '*') error(linecount,"Illegal comment");
 | 
						|
	do {
 | 
						|
		ch = input();
 | 
						|
		while (ch == '*') {
 | 
						|
			if ((ch = input()) == '/') return;
 | 
						|
		}
 | 
						|
		if (flag && ch == '\n') putc(ch,fact);
 | 
						|
	} while (ch != EOF);
 | 
						|
	error(saved,"Comment does not terminate");
 | 
						|
}
 | 
						|
 | 
						|
# ifdef LINE_DIRECTIVE
 | 
						|
STATIC
 | 
						|
linedirective() {
 | 
						|
	/*
 | 
						|
	 * Read a line directive
 | 
						|
	 */
 | 
						|
	register int	ch;
 | 
						|
	register int	i;
 | 
						|
	string		s_error = "Illegal line directive";
 | 
						|
	string		store();
 | 
						|
	register string	c;
 | 
						|
 | 
						|
	do {	/*
 | 
						|
		 * Skip to next digit
 | 
						|
		 * Do not skip newlines
 | 
						|
		 */
 | 
						|
		ch = input();
 | 
						|
	} while (ch != '\n' && c_class[ch] != ISDIG);
 | 
						|
	if (ch == '\n') {
 | 
						|
		error(linecount,s_error);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	i = 0;
 | 
						|
	do  {
 | 
						|
		i = i*10 + (ch - '0');
 | 
						|
		ch = input();
 | 
						|
	} while (c_class[ch] == ISDIG);
 | 
						|
	while (ch != '\n' && ch != '"') ch = input();
 | 
						|
	if (ch == '"') {
 | 
						|
		c = ltext;
 | 
						|
		do {
 | 
						|
			*c++ = ch = input();
 | 
						|
		} while (ch != '"' && ch != '\n');
 | 
						|
		if (ch == '\n') {
 | 
						|
			error(linecount,s_error);
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		*--c = '\0';
 | 
						|
		do {
 | 
						|
			ch = input();
 | 
						|
		} while (ch != '\n');
 | 
						|
		/*
 | 
						|
		 * Remember the file name
 | 
						|
		 */
 | 
						|
		if (strcmp(f_input,ltext)) f_input = store(ltext);
 | 
						|
	}
 | 
						|
	linecount = i;
 | 
						|
}
 | 
						|
# endif
 | 
						|
 | 
						|
STATIC string
 | 
						|
vallookup(s) {
 | 
						|
	/*
 | 
						|
	 * Look up the keyword that has token number s
 | 
						|
	 */
 | 
						|
	register p_keyw p = resword;
 | 
						|
 | 
						|
	while (p->w_value) {
 | 
						|
		if (p->w_value == s) return p->w_word;
 | 
						|
		p++;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
STATIC string
 | 
						|
cpy(s,p,inserted) register string p; {
 | 
						|
	/*
 | 
						|
	 * Create a piece of error message for token s and put it at p.
 | 
						|
	 * inserted = 0 if the token s was deleted (in which case we have
 | 
						|
	 * attributes), else it was inserted
 | 
						|
	 */
 | 
						|
	register string t = 0;
 | 
						|
 | 
						|
	switch(s) {
 | 
						|
	  case C_IDENT : 	
 | 
						|
		if (!inserted) t = lextoken.t_string;
 | 
						|
		else t = "identifier";
 | 
						|
		break;
 | 
						|
	  case C_NUMBER :
 | 
						|
		t = "number";
 | 
						|
		break;
 | 
						|
	  case C_LITERAL :
 | 
						|
		if (!inserted) {
 | 
						|
			*p++ = '\'';
 | 
						|
			t = lextoken.t_string;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		t = "literal";
 | 
						|
		break;
 | 
						|
	  case EOFILE :
 | 
						|
		t = "endoffile";
 | 
						|
		break;
 | 
						|
	}
 | 
						|
	if (!t && (t = vallookup(s))) {
 | 
						|
		*p++ = '%';
 | 
						|
	}
 | 
						|
	if (t) {	/*
 | 
						|
			 * We have a string for the token. Copy it
 | 
						|
			 */
 | 
						|
		while (*t) *p++ = *t++;
 | 
						|
		if (s == C_LITERAL && !inserted) {
 | 
						|
			*p++ = '\'';
 | 
						|
		}
 | 
						|
		return p;
 | 
						|
	}
 | 
						|
	/*
 | 
						|
	 * The token is a literal
 | 
						|
	 */
 | 
						|
	*p++ = '\'';
 | 
						|
	if (s >= 040 && s <= 0176) *p++ = s;
 | 
						|
	else {
 | 
						|
	    *p++ = '\\';
 | 
						|
	    switch(s) {
 | 
						|
	      case '\b' : *p++ = 'b'; break;
 | 
						|
	      case '\f' : *p++ = 'f'; break;
 | 
						|
	      case '\n' : *p++ = 'n'; break;
 | 
						|
	      case '\r' : *p++ = 'r'; break;
 | 
						|
	      case '\t' : *p++ = 't'; break;
 | 
						|
	      default : *p++='0'+((s&0377)>>6); *p++='0'+((s>>3)&07);
 | 
						|
		        *p++='0'+(s&07);
 | 
						|
	    }
 | 
						|
	}
 | 
						|
	*p++ = '\'';
 | 
						|
	return p;
 | 
						|
}
 | 
						|
 | 
						|
LLmessage(d) {
 | 
						|
	/*
 | 
						|
	 * d is either 0, in which case the current token has been deleted,
 | 
						|
	 * or non-zero, in which case it represents a token that is inserted
 | 
						|
	 * before the current token
 | 
						|
	 */
 | 
						|
	register string	s,t;
 | 
						|
	char		buf[128];
 | 
						|
 | 
						|
	nerrors++;
 | 
						|
	s = buf;
 | 
						|
	if (d == 0) {
 | 
						|
		s = cpy(LLsymb,s,0);
 | 
						|
		t = " deleted";
 | 
						|
		do *s++ = *t; while (*t++);
 | 
						|
	} else {
 | 
						|
		s = cpy(d,s,1);
 | 
						|
		t = " inserted in front of ";
 | 
						|
		do *s++ = *t++; while (*t);
 | 
						|
		s = cpy(LLsymb,s,0);
 | 
						|
		*s = '\0';
 | 
						|
	}
 | 
						|
	error(linecount, "%s", buf);
 | 
						|
			/* Don't change this line to 
 | 
						|
			 * error(linecount, buf).
 | 
						|
			 * The string in "buf" might contain '%' ...
 | 
						|
			 */
 | 
						|
	if (d) {	/*
 | 
						|
			 * Save the current token and make up some
 | 
						|
			 * attributes for the inserted token
 | 
						|
			 */
 | 
						|
		savedtok = lextoken;
 | 
						|
		savedtok.t_tokno = LLsymb;
 | 
						|
		if (d == C_IDENT) lextoken.t_string = "dummy_identifier";
 | 
						|
		else if (d == C_LITERAL) lextoken.t_string = "dummy_literal";
 | 
						|
		else if (d == C_NUMBER) lextoken.t_num = 1;
 | 
						|
	}
 | 
						|
}
 | 
						|
}
 |