mirror of
https://github.com/SEPPDROID/Digital-Research-Source-Code.git
synced 2025-10-24 00:44:23 +00:00
508 lines
12 KiB
C
508 lines
12 KiB
C
/*
|
|
Copyright 1982
|
|
Alcyon Corporation
|
|
8716 Production Ave.
|
|
San Diego, Ca. 92121
|
|
*/
|
|
|
|
#include "parser.h"
|
|
#define labgen(l,sl) sl=l;l=nextlabel++;
|
|
int swp -1;
|
|
|
|
|
|
/* stmt - process a single statement*/
|
|
stmt() /* returns - none*/
|
|
{
|
|
register int token, lab, i;
|
|
register struct tnode *tp;
|
|
register char *p;
|
|
|
|
while( 1 ) {
|
|
switch(token=gettok()) {
|
|
|
|
case LCURBR: /*handle { ... }*/
|
|
while( !next(EOF) ) {
|
|
if( next(RCURBR) )
|
|
return;
|
|
stmt();
|
|
}
|
|
case EOF:
|
|
error("{ not matched by }");
|
|
case SEMI: /*null statement*/
|
|
return;
|
|
|
|
case RCURBR:
|
|
pbtok(token);
|
|
return;
|
|
|
|
case SYMBOL: /*symbol: statement*/
|
|
if( peekc(':') ) {
|
|
dolabel();
|
|
continue;
|
|
}
|
|
default: /*anything else...*/
|
|
pbtok(token);
|
|
outexpr(expr());
|
|
break;
|
|
|
|
case RESWORD:
|
|
switch(cvalue) {
|
|
|
|
case R_BREAK:
|
|
lab = brklabel();
|
|
outgoto(lab); /*branch to break label*/
|
|
break;
|
|
|
|
case R_CASE:
|
|
docase();
|
|
continue;
|
|
|
|
case R_CONTINUE:
|
|
lab = contlabel(); /*branch to continue label*/
|
|
outgoto(lab);
|
|
break;
|
|
|
|
case R_DEFAULT:
|
|
dodefault();
|
|
continue;
|
|
|
|
case R_DO:
|
|
dodo();
|
|
break;
|
|
|
|
case R_FOR:
|
|
dofor();
|
|
return;
|
|
|
|
case R_GOTO:
|
|
lab = gotolabel();
|
|
outgoto(lab);
|
|
break;
|
|
|
|
case R_IF:
|
|
doif();
|
|
return;
|
|
|
|
case R_RETURN:
|
|
doreturn();
|
|
break;
|
|
|
|
case R_SWITCH:
|
|
doswitch();
|
|
return;
|
|
|
|
case R_WHILE:
|
|
dowhile();
|
|
return;
|
|
|
|
default:
|
|
synerr("invalid keyword");
|
|
return;
|
|
}
|
|
}
|
|
if( !next(SEMI) )
|
|
synerr("missing semicolon");
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* balpar - handle expression within parenthesis for while and if*/
|
|
/* Merely checks for left and right parens and builds expression.*/
|
|
char *balpar() /* returns pointer to expression*/
|
|
{
|
|
register struct tnode *tp;
|
|
|
|
if( next(LPAREN) ) {
|
|
reducep = 1;
|
|
tp = expr();
|
|
reducep = 0;
|
|
if( next(RPAREN) )
|
|
return(tp);
|
|
}
|
|
synerr("parenthesized expression syntax");
|
|
return(0);
|
|
}
|
|
|
|
/* synerr - syntax error*/
|
|
/* Outputs error message and tries to resyncronize input.*/
|
|
synerr(s,x1,x2,x3,x4,x5,x6) /* returns - none*/
|
|
char *s; /* printf format string*/
|
|
int x1, x2, x3, x4, x5, x6; /* printf arguments*/
|
|
{
|
|
register int token;
|
|
|
|
error(s,x1,x2,x3,x4,x5,x6);
|
|
while( (token=gettok()) != SEMI && token != EOF && token != LCURBR &&
|
|
token != RCURBR )
|
|
;
|
|
pbtok(token);
|
|
}
|
|
|
|
/* gotolabel - gets label id for goto*/
|
|
/* This is used for both: goto symbol and if(...)goto symbol*/
|
|
gotolabel() /* returns 0 if not, else label id*/
|
|
{
|
|
register struct symbol *sp;
|
|
|
|
if( !next(SYMBOL) )
|
|
synerr("expected label");
|
|
else {
|
|
sp = csp;
|
|
if( !(sp->s_sc) ) {
|
|
sp->s_type = LLABEL;
|
|
if( !sp->s_offset )
|
|
sp->s_offset = nextlabel++;
|
|
}
|
|
if( (!sp->s_sc || sp->s_sc == STATIC ) && sp->s_type == LLABEL )
|
|
return(sp->s_offset);
|
|
synerr("invalid label");
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/* dolabel - do statement label*/
|
|
/* Checks current symbol for already being defined, then sets*/
|
|
/* symbol attributes for label.*/
|
|
dolabel() /* returns - none*/
|
|
{
|
|
register struct symbol *sp;
|
|
|
|
sp = csp;
|
|
if( sp->s_sc )
|
|
error("label redeclaration: %.8s",sp->s_symbol);
|
|
else {
|
|
sp->s_attrib =| SDEFINED;
|
|
sp->s_sc = STATIC;
|
|
sp->s_type = LLABEL;
|
|
if( !sp->s_offset )
|
|
sp->s_offset = nextlabel++;
|
|
outlab(sp->s_offset);
|
|
}
|
|
}
|
|
|
|
/* brklabel - generate break label*/
|
|
/* Checks if break label is undefined, and if so, generates message*/
|
|
brklabel() /* returns label number*/
|
|
{
|
|
if( !blabel )
|
|
error("invalid break statement");
|
|
return(blabel);
|
|
}
|
|
|
|
/* contlabel - generate continue label*/
|
|
/* Checks if continue label is undefined, and if so, generates message*/
|
|
contlabel() /* returns label number*/
|
|
{
|
|
if( !clabel )
|
|
error("invalid continue statement");
|
|
return(clabel);
|
|
}
|
|
|
|
/* docase - handles: case constant : statement*/
|
|
/* Checks for being in a switch statement, adds entry to switch table*/
|
|
docase() /* returns - none*/
|
|
{
|
|
register int value, lab;
|
|
|
|
colonstop++;
|
|
value = cexpr(); /*get case value*/
|
|
colonstop--;
|
|
if( !next(COLON) ) /*check for colon*/
|
|
synerr("missing colon");
|
|
if( swp < 0 )
|
|
error("case not inside a switch block");
|
|
else if( swp >= (SWSIZE-1) )
|
|
error("too many cases in switch");
|
|
else {
|
|
addswitch(&swtab[cswp],swp-cswp,value,lab=nextlabel++);
|
|
outlab(lab);
|
|
swp++;
|
|
}
|
|
}
|
|
|
|
/* dodefault - handles: default : statement*/
|
|
/* Checks for colon and being in a switch statement*/
|
|
dodefault() /* returns - none*/
|
|
{
|
|
if( !next(COLON) )
|
|
error("missing colon");
|
|
if( swp < 0 )
|
|
error("default not inside a switch block");
|
|
else {
|
|
dlabel = nextlabel++; /*allocate default label*/
|
|
outlab(dlabel); /*output default label*/
|
|
}
|
|
}
|
|
|
|
/* dodo - handles: do statement while ( expression )*/
|
|
dodo() /* returns - none*/
|
|
{
|
|
register int lab, saveblab, saveclab;
|
|
|
|
labgen(blabel,saveblab);
|
|
labgen(clabel,saveclab);
|
|
lab = nextlabel++;
|
|
outlab(lab); /*branch back to here*/
|
|
stmt(); /*do statement*/
|
|
outlab(clabel); /*continue label*/
|
|
if( !nextrw(R_WHILE) ) {
|
|
error("missing while"); /*only advisory...*/
|
|
outgoto(lab);
|
|
}
|
|
else
|
|
outifgoto(balpar(),TRUE,lab); /*while expression*/
|
|
outlab(blabel); /*break label*/
|
|
blabel = saveblab; /*restore labels*/
|
|
clabel = saveclab;
|
|
}
|
|
|
|
/*
|
|
* dofor - handle: for ( expression ; expression ; expression ) statement
|
|
* Hard part is handling re-initialization expression, which is
|
|
* parsed and saved, then the statement is parsed, then the reinit
|
|
* clause expression tree is output.
|
|
*/
|
|
dofor() /* returns - none*/
|
|
{
|
|
register int lab, saveblab, saveclab, reinit, clineno;
|
|
register char *savep;
|
|
register struct tnode *tp;
|
|
|
|
labgen(blabel,saveblab);
|
|
labgen(clabel,saveclab);
|
|
if( !next(LPAREN) ) {
|
|
forerr:
|
|
synerr("invalid for statement");
|
|
return;
|
|
}
|
|
if( !next(SEMI) ) { /*do init expression*/
|
|
outexpr(expr());
|
|
if( !next(SEMI) )
|
|
goto forerr;
|
|
}
|
|
outlab(clabel); /*branch back to here*/
|
|
if( !next(SEMI) ) { /*do for condition*/
|
|
outifgoto(expr(),FALSE,blabel);
|
|
if( !next(SEMI) )
|
|
goto forerr;
|
|
}
|
|
if( next(RPAREN) ) { /*no re-init - easy case*/
|
|
stmt(); /*output statement*/
|
|
outgoto(clabel); /*output continue label*/
|
|
}
|
|
else { /*there is a re-init clause*/
|
|
labgen(clabel,lab);
|
|
savep = exprp;
|
|
tp = expr(); /*save re-init tree until done*/
|
|
exprp = opap; /*remember until reinit is output*/
|
|
reinit = lineno;
|
|
if( !next(RPAREN) )
|
|
goto forerr;
|
|
stmt(); /*do statment*/
|
|
clineno = lineno;
|
|
lineno = reinit;
|
|
outlab(clabel); /*we branch to here for reinit*/
|
|
outexpr(tp); /*output re-init clause*/
|
|
exprp = savep;
|
|
lineno = clineno;
|
|
outgoto(lab); /*branch back to top of loop*/
|
|
}
|
|
outlab(blabel); /*break to here*/
|
|
blabel = saveblab;
|
|
clabel = saveclab; /*restore labels*/
|
|
}
|
|
|
|
/* doif - handles: if ( expression ) statement [ else statement ]*/
|
|
/* Handles special cases for goto, break, continue and return.*/
|
|
doif() /* returns - none*/
|
|
{
|
|
register struct tnode *tp;
|
|
register int elselab, exitlab;
|
|
|
|
tp = balpar(); /*if( expr )...*/
|
|
exitlab = 0;
|
|
if( nextrw(R_GOTO) )
|
|
exitlab = gotolabel();
|
|
else if( nextrw(R_BREAK) )
|
|
exitlab = brklabel();
|
|
else if( nextrw(R_CONTINUE) )
|
|
exitlab = contlabel();
|
|
else if( nextrw(R_RETURN) ) {
|
|
if( peekc(';') ) {
|
|
exitlab = rlabel;
|
|
putback(';');
|
|
}
|
|
else
|
|
pbtok(RESWORD);
|
|
}
|
|
if( exitlab ) { /*easy goto, do branch if true*/
|
|
outifgoto(tp,TRUE,exitlab);
|
|
if( !next(SEMI) )
|
|
synerr("missing semicolon");
|
|
if( nextrw(R_ELSE) ) /*else clause, just output it*/
|
|
stmt();
|
|
}
|
|
else { /*hard goto, branch over statement*/
|
|
elselab = nextlabel++;
|
|
outifgoto(tp,FALSE,elselab);
|
|
stmt();
|
|
if( nextrw(R_ELSE) ) {
|
|
exitlab = nextlabel++; /*branches over else clause*/
|
|
outgoto(exitlab); /*branch out of then clause*/
|
|
outlab(elselab); /*label to start else clause*/
|
|
stmt(); /*else statement*/
|
|
outlab(exitlab);
|
|
}
|
|
else
|
|
outlab(elselab); /*no else, just branch out*/
|
|
}
|
|
}
|
|
|
|
/*
|
|
* doreturn - handles: return [ expression ] ;
|
|
* Expression is the hard part, must create an assignment expression
|
|
* to assign expression to the type of the function, then get it
|
|
* loaded into a specific register.
|
|
*/
|
|
doreturn() /* returns - none*/
|
|
{
|
|
register struct tnode *tp;
|
|
|
|
if( !peekc(';') ) /*need to compute return?*/
|
|
outforreg(FRETURN,frp,expr());
|
|
else
|
|
putback(';');
|
|
outgoto(rlabel); /*branch to the return label*/
|
|
}
|
|
|
|
/*
|
|
* doswitch - handles: switch ( expression ) statement
|
|
* Evaluates the expression, forces the result into a known register
|
|
* collects the case statements in swtab, then outputs the switch
|
|
* operator and switch cases.
|
|
*/
|
|
doswitch() /* returns - none*/
|
|
{
|
|
register int saveblab, swlab, savedlab, saveswp, i;
|
|
register struct tnode *tp;
|
|
|
|
labgen(blabel,saveblab);
|
|
tp = balpar();
|
|
integral(tp,-1); /*must be integral type result*/
|
|
outforreg(ASSIGN,snalloc(INT,AUTO,0,0,0),tp);
|
|
saveswp = swp; /*remember old switch pointer*/
|
|
if( saveswp < 0 )
|
|
swp++;
|
|
i = cswp;
|
|
cswp = swp; /*remember real first entry*/
|
|
swlab = nextlabel++;
|
|
outgoto(swlab); /*branch to switch code*/
|
|
savedlab = dlabel;
|
|
dlabel = 0;
|
|
stmt(); /*do switch statement*/
|
|
outgoto(blabel); /*output branch just in case*/
|
|
outlab(swlab); /*here we now do the switch code*/
|
|
if( !dlabel )
|
|
dlabel = blabel;
|
|
outswitch(swp-cswp,dlabel,&swtab[cswp]);
|
|
outlab(blabel); /*break to here*/
|
|
cswp = i;
|
|
swp = saveswp;
|
|
blabel = saveblab;
|
|
dlabel = savedlab;
|
|
}
|
|
|
|
/* dowhile - handles: while ( expression ) statement*/
|
|
/* This is fairly straight-forward.*/
|
|
dowhile() /* returns - none*/
|
|
{
|
|
register int saveclab, saveblab;
|
|
|
|
labgen(blabel,saveblab);
|
|
labgen(clabel,saveclab);
|
|
outlab(clabel); /*continue label*/
|
|
outifgoto(balpar(),FALSE,blabel); /*condition clause*/
|
|
stmt(); /*statement*/
|
|
outgoto(clabel); /*branch back to top of loop*/
|
|
outlab(blabel); /*break to here*/
|
|
blabel = saveblab;
|
|
clabel = saveclab; /*restore labels*/
|
|
}
|
|
|
|
/* nextrw - is next token the specified reserved word?*/
|
|
nextrw(rw) /* returns 1 if match, 0 otherwise*/
|
|
int rw; /* reserved word to match*/
|
|
{
|
|
register int token;
|
|
|
|
if( (token=gettok()) != RESWORD || cvalue != rw ) {
|
|
pbtok(token);
|
|
return(0);
|
|
}
|
|
return(1);
|
|
}
|
|
|
|
/*
|
|
* addswitch - add an entry into current switch table, bubble sorting
|
|
* This makes it easier on the code generator and also checks for
|
|
* duplicate labels at the "right" time.
|
|
*/
|
|
addswitch(sp,ncases,nval,nlab) /* returns - none*/
|
|
struct swtch *sp; /* switch table pointer*/
|
|
int ncases; /* number of cases in switch*/
|
|
int nval; /* new value*/
|
|
int nlab; /* new label*/
|
|
{
|
|
register struct swtch *nswp, *s;
|
|
register int temp, i;
|
|
|
|
nswp = &sp[ncases];
|
|
nswp->sw_value = nval;
|
|
nswp->sw_label = nlab;
|
|
s = nswp--;
|
|
for( ; --ncases >= 0; s--, nswp-- ) {
|
|
if( s->sw_value == nswp->sw_value )
|
|
error("duplicate case value");
|
|
if( s->sw_value < nswp->sw_value ) {
|
|
temp = s->sw_value;
|
|
s->sw_value = nswp->sw_value;
|
|
nswp->sw_value = temp;
|
|
temp = s->sw_label;
|
|
s->sw_label = nswp->sw_label;
|
|
nswp->sw_label = temp;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* outforreg - generate assignment for switch and return*/
|
|
outforreg(op,ltp,rtp) /*returns - none*/
|
|
int op; /*operator for build tree*/
|
|
struct tnode *ltp; /*left expression tree*/
|
|
struct tnode *rtp; /*right expression tree*/
|
|
{
|
|
register struct tnode *tp;
|
|
|
|
opp = opstack;
|
|
opdp = opdstack;
|
|
pushopd(ltp);
|
|
pushopd(rtp);
|
|
maketree(op);
|
|
if( tp = popopd() )
|
|
outcforreg(tp->t_right);
|
|
opp = opdp = 0;
|
|
}
|
|
|
|
/* outassign - generate assignment for function args*/
|
|
outassign(ltp,rtp) /*returns - none*/
|
|
struct tnode *ltp; /*left expression tree*/
|
|
struct tnode *rtp; /*right expression tree*/
|
|
{
|
|
opp = opstack;
|
|
opdp = opdstack;
|
|
pushopd(ltp);
|
|
pushopd(rtp);
|
|
maketree(ASSIGN);
|
|
outexpr(popopd());
|
|
opp = opdp = 0;
|
|
}
|