| #include "e.h" | 
 | #include "y.tab.h" | 
 | #include <ctype.h> | 
 | #include <errno.h> | 
 |  | 
 | #define	SSIZE	1000 | 
 | char	token[SSIZE]; | 
 | int	sp; | 
 |  | 
 | void	space(void); | 
 | void	dodef(tbl *); | 
 | void	define(int); | 
 | void	ifdef(void); | 
 | void	include(void); | 
 | void	delim(void); | 
 |  | 
 | int | 
 | yylex(void) | 
 | { | 
 | 	register int c; | 
 | 	tbl *tp; | 
 |  | 
 | begin: | 
 | 	while ((c = input()) == ' ' || c == '\n' || c == '\t') | 
 | 		; | 
 | 	yylval = c; | 
 | 	switch (c) { | 
 | 	case EOF: | 
 | 		ERROR "unexpected end of input inside equation" WARNING; | 
 | 		return(EOF); | 
 | 	case '~': | 
 | 		return(SPACE); | 
 | 	case '^': | 
 | 		return(THIN); | 
 | 	/* case '\t': | 
 | 		return(TAB); | 
 | 	*/ | 
 | 	case '{': | 
 | 		return('{'); | 
 | 	case '}': | 
 | 		return('}'); | 
 | 	case '"': | 
 | 		for (sp = 0; (c=input())!='"' && c != '\n'; ) { | 
 | 			if (c == '\\') | 
 | 				if ((c = input()) != '"') | 
 | 					token[sp++] = '\\'; | 
 | 			token[sp++] = c; | 
 | 			if (sp >= SSIZE) | 
 | 				ERROR "quoted string %.20s... too long", token FATAL; | 
 | 		} | 
 | 		token[sp] = '\0'; | 
 | 		yylval = (intptr_t)&token[0]; | 
 | 		if (c == '\n') | 
 | 			ERROR "missing \" in %.20s", token WARNING; | 
 | 		return(QTEXT); | 
 | 	} | 
 | 	if (!display && c == righteq) | 
 | 		return(EOF); | 
 |  | 
 | 	unput(c); | 
 | 	getstr(token, SSIZE); | 
 | 	dprintf(".\tlex token = |%s|\n", token); | 
 | 	if ((tp = lookup(deftbl, token)) != NULL) {	/* defined term */ | 
 | 		c = input(); | 
 | 		unput(c); | 
 | 		if (c == '(')	/* macro with args */ | 
 | 			dodef(tp); | 
 | 		else {		/* no args */ | 
 | 			unput(' '); | 
 | 			pbstr(tp->cval); | 
 | 			dprintf(".\tfound %s|=%s|\n", token, tp->cval); | 
 | 		} | 
 | 		goto begin; | 
 | 	} | 
 |  | 
 | 	if ((tp = lookup(keytbl, token)) == NULL)	/* not a keyword */ | 
 | 		return CONTIG; | 
 |  | 
 | 	switch (tp->ival) {		/* some kind of keyword */ | 
 | 	case DEFINE: case TDEFINE: case NDEFINE: | 
 | 		define(tp->ival); | 
 | 		break; | 
 | 	case IFDEF: | 
 | 		ifdef(); | 
 | 		break; | 
 | 	case DELIM: | 
 | 		delim(); | 
 | 		break; | 
 | 	case GSIZE: | 
 | 		globsize(); | 
 | 		break; | 
 | 	case GFONT: | 
 | 		globfont(); | 
 | 		break; | 
 | 	case INCLUDE: | 
 | 		include(); | 
 | 		break; | 
 | 	case SPACE: | 
 | 		space(); | 
 | 		break; | 
 | 	case DOTEQ: | 
 | 			/* .EQ inside equation -- should warn if at bottom level */ | 
 | 		break; | 
 | 	case DOTEN: | 
 | 		if (curfile == infile) | 
 | 			return EOF; | 
 | 		/* else ignore nested .EN */ | 
 | 		break; | 
 | 	default: | 
 | 		return tp->ival; | 
 | 	} | 
 | 	goto begin; | 
 | } | 
 |  | 
 | void getstr(char *s, int n) | 
 | { | 
 | 	register int c; | 
 | 	register char *p; | 
 |  | 
 | 	p = s; | 
 | 	while ((c = input()) == ' ' || c == '\n') | 
 | 		; | 
 | 	if (c == EOF) { | 
 | 		*s = 0; | 
 | 		return; | 
 | 	} | 
 | 	while (c != ' ' && c != '\t' && c != '\n' && c != '{' && c != '}' | 
 | 	    && c != '"' && c != '~' && c != '^') { | 
 | 		if (!display && c == righteq) | 
 | 			break; | 
 | 		if (c == '(' && p > s) {	/* might be defined(...) */ | 
 | 			*p = '\0'; | 
 | 			if (lookup(deftbl, s) != NULL) | 
 | 				break; | 
 | 		} | 
 | 		if (c == '\\') | 
 | 			if ((c = input()) != '"') | 
 | 				*p++ = '\\'; | 
 | 		*p++ = c; | 
 | 		if (--n <= 0) | 
 | 			ERROR "token %.20s... too long", s FATAL; | 
 | 		c = input(); | 
 | 	} | 
 | 	unput(c); | 
 | 	*p = '\0'; | 
 | 	yylval = (uintptr_t)s; | 
 | } | 
 |  | 
 | int | 
 | cstr(char *s, int quote, int maxs) | 
 | { | 
 | 	int del, c, i; | 
 |  | 
 | 	s[0] = 0; | 
 | 	while ((del=input()) == ' ' || del == '\t') | 
 | 		; | 
 | 	if (quote) | 
 | 		for (i=0; (c=input()) != del && c != EOF;) { | 
 | 			s[i++] = c; | 
 | 			if (i >= maxs) | 
 | 				return(1);	/* disaster */ | 
 | 		} | 
 | 	else { | 
 | 		if (del == '\n') | 
 | 			return(1); | 
 | 		s[0] = del; | 
 | 		for (i=1; (c=input())!=' ' && c!= '\t' && c!='\n' && c!=EOF;) { | 
 | 			s[i++] = c; | 
 | 			if (i >= maxs) | 
 | 				return(1);	/* disaster */ | 
 | 		} | 
 | 	} | 
 | 	s[i] = '\0'; | 
 | 	if (c == EOF) | 
 | 		ERROR "Unexpected end of input at %.20s", s FATAL; | 
 | 	return(0); | 
 | } | 
 |  | 
 | void define(int type) | 
 | { | 
 | 	char *p1, *p2; | 
 | 	extern int ftune(char *, char *); | 
 |  | 
 | 	getstr(token, SSIZE);	/* get name */ | 
 | 	if (type != DEFINE) { | 
 | 		cstr(token, 1, SSIZE);	/* skip the definition too */ | 
 | 		return; | 
 | 	} | 
 | 	p1 = strsave(token); | 
 | 	if (cstr(token, 1, SSIZE)) | 
 | 		ERROR "Unterminated definition at %.20s", token FATAL; | 
 | 	if (lookup(ftunetbl, p1) != NULL) {	/* double tuning param */ | 
 | 		dprintf(".\ttune %s %s\n", p1, token); | 
 | 		ftune(p1, token); | 
 | 	} else { | 
 | 		p2 = strsave(token); | 
 | 		install(deftbl, p1, p2, 0); | 
 | 		dprintf(".\tname %s defined as %s\n", p1, p2); | 
 | 	} | 
 | } | 
 |  | 
 | void ifdef(void)		/* do body if name is defined */ | 
 | { | 
 | 	char name[100], *p; | 
 |  | 
 | 	getstr(name, sizeof(name));	/* get name */ | 
 | 	cstr(token, 1, SSIZE);		/* and body */ | 
 | 	if (lookup(deftbl, name) != NULL) {	/* found it */ | 
 | 		p = strsave(token); | 
 | 		pushsrc(Free, p); | 
 | 		pushsrc(String, p); | 
 | 	} | 
 | } | 
 |  | 
 | char	*spaceval	= NULL; | 
 |  | 
 | void space(void)	/* collect line of form "space amt" to replace \x in output */ | 
 | { | 
 | 	getstr(token, SSIZE); | 
 | 	spaceval = strsave(token); | 
 | 	dprintf(".\tsetting spaceval to %s\n", token); | 
 | } | 
 |  | 
 | char *strsave(char *s) | 
 | { | 
 | 	register char *q; | 
 |  | 
 | 	q = malloc(strlen(s)+1); | 
 | 	if (q == NULL) | 
 | 		ERROR "out of space in strsave on %s", s FATAL; | 
 | 	strcpy(q, s); | 
 | 	return(q); | 
 | } | 
 |  | 
 | void include(void) | 
 | { | 
 | 	char name[100]; | 
 | 	FILE *fin; | 
 | 	int c; | 
 |  | 
 | 	while ((c = input()) == ' ') | 
 | 		; | 
 | 	unput(c); | 
 | 	cstr(name, c == '"', sizeof(name));	/* gets it quoted or not */ | 
 | 	if ((fin = fopen(name, "r")) == NULL) | 
 | 		ERROR "can't open file %s", name FATAL; | 
 | 	errno = 0; | 
 | 	curfile++; | 
 | 	curfile->fin = fin; | 
 | 	curfile->fname = strsave(name); | 
 | 	curfile->lineno = 0; | 
 | 	printf(".lf 1 %s\n", curfile->fname); | 
 | 	pushsrc(File, curfile->fname); | 
 | } | 
 |  | 
 | void delim(void) | 
 | { | 
 | 	yyval = eqnreg = 0; | 
 | 	if (cstr(token, 0, SSIZE)) | 
 | 		ERROR "Bizarre delimiters" FATAL; | 
 | 	lefteq = token[0]; | 
 | 	righteq = token[1]; | 
 | 	if (!isprint(lefteq) || !isprint(righteq)) | 
 | 		ERROR "Bizarre delimiters" FATAL; | 
 | 	if (lefteq == 'o' && righteq == 'f') | 
 | 		lefteq = righteq = '\0'; | 
 | } |