Compare commits
No commits in common. "0339c25877d953d41714dac0bb11182f2a295a8d" and "8325b067b404c7d9fdf060d75649929f4aa81fe3" have entirely different histories.
0339c25877
...
8325b067b4
2
Makefile
2
Makefile
|
@ -1,7 +1,7 @@
|
||||||
FILES = *.c */*.c
|
FILES = *.c */*.c
|
||||||
LIBRARIES = -lpthread
|
LIBRARIES = -lpthread
|
||||||
INCLUDES = -I. -I./parser/ -I./compiler/ -I./vm/ -I./tokenizer/ -I./misc/
|
INCLUDES = -I. -I./parser/ -I./compiler/ -I./vm/ -I./tokenizer/ -I./misc/
|
||||||
CFLAGS = -std=c99 -Wall -O3
|
CFLAGS = -std=c99 -Wall
|
||||||
OUTFILE = jack-compiler
|
OUTFILE = jack-compiler
|
||||||
|
|
||||||
main: ${FILES}
|
main: ${FILES}
|
||||||
|
|
|
@ -15,10 +15,10 @@ char* toascii(char c);
|
||||||
// Dealing with singular terms
|
// Dealing with singular terms
|
||||||
LINEBLOCK* compilestrconst(char* str);
|
LINEBLOCK* compilestrconst(char* str);
|
||||||
LINEBLOCK* compilekeywordconst(SCOPE* s, TERM* t);
|
LINEBLOCK* compilekeywordconst(SCOPE* s, TERM* t);
|
||||||
LINEBLOCK* compilearrayitem(SCOPE* s, DEBUGINFO* d, TERM* t);
|
LINEBLOCK* compilearrayitem(SCOPE* s, TERM* t);
|
||||||
LINEBLOCK* compilecallln(SCOPE* s, SUBROUTDEC* d, SUBROUTCALL* call);
|
LINEBLOCK* compilecallln(SCOPE* s, SUBROUTDEC* d, SUBROUTCALL* call);
|
||||||
LINEBLOCK* pushunaryopterm(SCOPE* s, DEBUGINFO* d, TERM* t);
|
LINEBLOCK* pushunaryopterm(SCOPE* s, TERM* t);
|
||||||
LINEBLOCK* compileterm(SCOPE* s, DEBUGINFO* d, TERM* t);
|
LINEBLOCK* compileterm(SCOPE* s, TERM* t);
|
||||||
|
|
||||||
/* END FORWARD DECLARATIONS */
|
/* END FORWARD DECLARATIONS */
|
||||||
|
|
||||||
|
@ -128,9 +128,9 @@ LINEBLOCK* compilekeywordconst(SCOPE* s, TERM* t) {
|
||||||
return mklnblk(pushconstant(0));
|
return mklnblk(pushconstant(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilearrayitem(SCOPE* s, DEBUGINFO* d, TERM* t) {
|
LINEBLOCK* compilearrayitem(SCOPE* s, TERM* t) {
|
||||||
LINEBLOCK* blk = compileexpression(s, d, t->array->exp);
|
LINEBLOCK* blk = compileexpression(s, t->array->exp);
|
||||||
appendln(blk, pushvar(s, d, t->array->name));
|
appendln(blk, pushvar(s, t->array->name));
|
||||||
|
|
||||||
appendln(blk, onetoken("add"));
|
appendln(blk, onetoken("add"));
|
||||||
|
|
||||||
|
@ -161,7 +161,7 @@ LINEBLOCK* compilesubroutcall(SCOPE* s, SUBROUTCALL* call) {
|
||||||
LINEBLOCK* blk = compilecallln(s, d, call);
|
LINEBLOCK* blk = compilecallln(s, d, call);
|
||||||
|
|
||||||
if(call->parameters != NULL)
|
if(call->parameters != NULL)
|
||||||
blk = mergelnblks(compileexplist(s, call->debug, call->parameters), blk);
|
blk = mergelnblks(compileexplist(s, call->parameters), blk);
|
||||||
|
|
||||||
if(d->subroutclass == method) {
|
if(d->subroutclass == method) {
|
||||||
if(call->parentname == NULL)
|
if(call->parentname == NULL)
|
||||||
|
@ -179,8 +179,8 @@ LINEBLOCK* compilesubroutcall(SCOPE* s, SUBROUTCALL* call) {
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* pushunaryopterm(SCOPE* s, DEBUGINFO* d, TERM* t) {
|
LINEBLOCK* pushunaryopterm(SCOPE* s, TERM* t) {
|
||||||
LINEBLOCK* blk = compileexpression(s, d, t->expression);
|
LINEBLOCK* blk = compileexpression(s, t->expression);
|
||||||
LINE* neg;
|
LINE* neg;
|
||||||
if(t->unaryop == '-')
|
if(t->unaryop == '-')
|
||||||
neg = onetoken("neg");
|
neg = onetoken("neg");
|
||||||
|
@ -190,39 +190,36 @@ LINEBLOCK* pushunaryopterm(SCOPE* s, DEBUGINFO* d, TERM* t) {
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compileterm(SCOPE* s, DEBUGINFO* d, TERM* t) {
|
LINEBLOCK* compileterm(SCOPE* s, TERM* t) {
|
||||||
if(t->type == varname) return mklnblk(pushvar(s, d, t->string));
|
if(t->type == varname) return mklnblk(pushvar(s, t->string));
|
||||||
if(t->type == intconstant) return mklnblk(pushconstant(t->integer));
|
if(t->type == intconstant) return mklnblk(pushconstant(t->integer));
|
||||||
if(t->type == stringconstant) return compilestrconst(t->string);
|
if(t->type == stringconstant) return compilestrconst(t->string);
|
||||||
if(t->type == keywordconstant) return compilekeywordconst(s, t);
|
if(t->type == keywordconstant) return compilekeywordconst(s, t);
|
||||||
if(t->type == arrayitem) return compilearrayitem(s, d, t);
|
if(t->type == arrayitem) return compilearrayitem(s, t);
|
||||||
if(t->type == subroutcall) return compilesubroutcall(s, t->call);
|
if(t->type == subroutcall) return compilesubroutcall(s, t->call);
|
||||||
if(t->type == innerexpression) return compileexpression(s, d, t->expression);
|
if(t->type == innerexpression) return compileexpression(s, t->expression);
|
||||||
return pushunaryopterm(s, d, t);
|
return pushunaryopterm(s, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dealing with whole expressions
|
// Dealing with whole expressions
|
||||||
LINEBLOCK* compileexpression(SCOPE* s, DEBUGINFO* d, TERM* e) {
|
LINEBLOCK* compileexpression(SCOPE* s, TERM* e) {
|
||||||
LINEBLOCK* blk = NULL;
|
LINEBLOCK* blk = NULL;
|
||||||
LINEBLOCK* ops = NULL;
|
|
||||||
if(e != NULL) {
|
if(e != NULL) {
|
||||||
while(true) {
|
while(true) {
|
||||||
blk = mergelnblks(blk, compileterm(s, d, e));
|
blk = mergelnblks(blk, compileterm(s, e));
|
||||||
if(e->next == NULL)
|
if(e->next == NULL)
|
||||||
break;
|
break;
|
||||||
ops = mergelnblks(ops, mklnblk(mathopln(e->op)));
|
appendln(blk, mathopln(e->op));
|
||||||
e = e->next;
|
e = e->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(ops != NULL)
|
|
||||||
blk = mergelnblks(blk, ops);
|
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compileexplist(SCOPE* s, DEBUGINFO* d, EXPRESSIONLIST* explist) {
|
LINEBLOCK* compileexplist(SCOPE* s, EXPRESSIONLIST* explist) {
|
||||||
LINEBLOCK* head = NULL;
|
LINEBLOCK* head = NULL;
|
||||||
while(explist != NULL) {
|
while(explist != NULL) {
|
||||||
head = mergelnblks(head, compileexpression(s, d, explist->expression));
|
head = mergelnblks(head, compileexpression(s, explist->expression));
|
||||||
explist = explist->next;
|
explist = explist->next;
|
||||||
}
|
}
|
||||||
return head;
|
return head;
|
||||||
|
|
|
@ -10,6 +10,6 @@
|
||||||
LINEBLOCK* compilesubroutcall(SCOPE* s, SUBROUTCALL* call);
|
LINEBLOCK* compilesubroutcall(SCOPE* s, SUBROUTCALL* call);
|
||||||
|
|
||||||
// Dealing with whole expressions
|
// Dealing with whole expressions
|
||||||
LINEBLOCK* compileexpression(SCOPE* s, DEBUGINFO* d, TERM* e);
|
LINEBLOCK* compileexpression(SCOPE* s, TERM* e);
|
||||||
LINEBLOCK* compileexplist(SCOPE* s, DEBUGINFO* d, EXPRESSIONLIST* explist);
|
LINEBLOCK* compileexplist(SCOPE* s, EXPRESSIONLIST* explist);
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -11,7 +11,7 @@ char* memsegnames[] = { "local", "static", "argument", "this" };
|
||||||
|
|
||||||
// Error messages
|
// Error messages
|
||||||
void doubledeclaration(const char* name, DEBUGINFO* d1, DEBUGINFO* d2);
|
void doubledeclaration(const char* name, DEBUGINFO* d1, DEBUGINFO* d2);
|
||||||
void ensurenoduplicate(SCOPE* s, DEBUGINFO* d, char* name);
|
void ensurenoduplicate(SCOPE* s, char* name);
|
||||||
|
|
||||||
// Getters
|
// Getters
|
||||||
VAR* getvarinvars(VAR* vars, const char* name);
|
VAR* getvarinvars(VAR* vars, const char* name);
|
||||||
|
@ -49,18 +49,18 @@ void invalidparent(SUBROUTCALL* call) {
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ensurenoduplicate(SCOPE* s, DEBUGINFO* d, char* name) {
|
void ensurenoduplicate(SCOPE* s, char* name) {
|
||||||
VAR* v = getvar(s, name);
|
VAR* v = getvar(s, name);
|
||||||
if(v != NULL)
|
if(v != NULL)
|
||||||
doubledeclaration(name, d, v->debug);
|
doubledeclaration(name, s->currdebug, v->debug);
|
||||||
|
|
||||||
CLASS* c = getclass(s, name);
|
CLASS* c = getclass(s, name);
|
||||||
if(c != NULL)
|
if(c != NULL)
|
||||||
doubledeclaration(name, d, c->debug);
|
doubledeclaration(name, s->currdebug, c->debug);
|
||||||
|
|
||||||
SUBROUTDEC* sr = getsubroutdec(s, name);
|
SUBROUTDEC* sr = getsubroutdec(s, name);
|
||||||
if(sr != NULL)
|
if(sr != NULL)
|
||||||
doubledeclaration(name, d, sr->debug);
|
doubledeclaration(name, s->currdebug, sr->debug);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Scope handling
|
// Scope handling
|
||||||
|
@ -106,13 +106,6 @@ VAR* getvar(SCOPE* s, const char* name) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
VAR* getvarmustexist(SCOPE* s, DEBUGINFO* d, const char* name) {
|
|
||||||
VAR* v = getvar(s, name);
|
|
||||||
if(v == NULL)
|
|
||||||
notdeclared(name, d);
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
|
|
||||||
CLASS* getclass(SCOPE* s, const char* name) {
|
CLASS* getclass(SCOPE* s, const char* name) {
|
||||||
CLASS* curr = s->classes;
|
CLASS* curr = s->classes;
|
||||||
while(curr != NULL) {
|
while(curr != NULL) {
|
||||||
|
@ -220,7 +213,7 @@ VAR* mkvar(char* type, char* name, bool primitive, DEBUGINFO* debug, MEMSEGMENT
|
||||||
}
|
}
|
||||||
|
|
||||||
void addvar(SCOPE* s, VAR** dest, VAR* v) {
|
void addvar(SCOPE* s, VAR** dest, VAR* v) {
|
||||||
ensurenoduplicate(s, v->debug, v->name);
|
ensurenoduplicate(s, v->name);
|
||||||
|
|
||||||
if(!v->primitive) {
|
if(!v->primitive) {
|
||||||
CLASS* type = getclass(s, v->type);
|
CLASS* type = getclass(s, v->type);
|
||||||
|
|
|
@ -22,6 +22,7 @@ typedef struct var {
|
||||||
|
|
||||||
typedef struct scope {
|
typedef struct scope {
|
||||||
struct compiler* compiler;
|
struct compiler* compiler;
|
||||||
|
DEBUGINFO* currdebug;
|
||||||
CLASS* currclass;
|
CLASS* currclass;
|
||||||
|
|
||||||
CLASS* classes;
|
CLASS* classes;
|
||||||
|
@ -51,7 +52,6 @@ CLASS* getclass(SCOPE* s, const char* name);
|
||||||
|
|
||||||
// Generic getters
|
// Generic getters
|
||||||
VAR* getvar(SCOPE* s, const char* name);
|
VAR* getvar(SCOPE* s, const char* name);
|
||||||
VAR* getvarmustexist(SCOPE* s, DEBUGINFO* d, const char* name);
|
|
||||||
|
|
||||||
// Freeing
|
// Freeing
|
||||||
void freescope(SCOPE* s);
|
void freescope(SCOPE* s);
|
||||||
|
|
|
@ -12,10 +12,10 @@ LINE* pushtemp();
|
||||||
char* mkcondlabel(char* name, int count);
|
char* mkcondlabel(char* name, int count);
|
||||||
|
|
||||||
// Handling individual statements
|
// Handling individual statements
|
||||||
LINEBLOCK* compileret(SCOPE* s, STATEMENT* st);
|
LINEBLOCK* compileret(SCOPE* s, TERM* e);
|
||||||
LINEBLOCK* compileif(SCOPE* s, STATEMENT* st);
|
LINEBLOCK* compileif(SCOPE* s, IFSTATEMENT* st);
|
||||||
LINEBLOCK* compilewhile(SCOPE* s, STATEMENT* st);
|
LINEBLOCK* compilewhile(SCOPE* s, CONDSTATEMENT* w);
|
||||||
LINEBLOCK* compilelet(SCOPE* s, STATEMENT* st);
|
LINEBLOCK* compilelet(SCOPE* s, LETSTATEMENT* l);
|
||||||
LINEBLOCK* compilestatement(SCOPE* s, STATEMENT* st);
|
LINEBLOCK* compilestatement(SCOPE* s, STATEMENT* st);
|
||||||
|
|
||||||
/* END FORWARD DECLARATIONS */
|
/* END FORWARD DECLARATIONS */
|
||||||
|
@ -41,8 +41,7 @@ char* mkcondlabel(char* name, int count) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handling individual statements
|
// Handling individual statements
|
||||||
LINEBLOCK* compileret(SCOPE* s, STATEMENT* st) {
|
LINEBLOCK* compileret(SCOPE* s, TERM* e) {
|
||||||
TERM* e = st->retstatement;
|
|
||||||
LINE* ret = onetoken("return");
|
LINE* ret = onetoken("return");
|
||||||
LINEBLOCK* blk = mklnblk(ret);
|
LINEBLOCK* blk = mklnblk(ret);
|
||||||
|
|
||||||
|
@ -51,14 +50,13 @@ LINEBLOCK* compileret(SCOPE* s, STATEMENT* st) {
|
||||||
char* tokens[] = { "push", "constant", "0" };
|
char* tokens[] = { "push", "constant", "0" };
|
||||||
appendlnbefore(blk, mkln(tokens));
|
appendlnbefore(blk, mkln(tokens));
|
||||||
} else
|
} else
|
||||||
blk = mergelnblks(compileexpression(s, st->debug, e), blk);
|
blk = mergelnblks(compileexpression(s, e), blk);
|
||||||
|
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compileif(SCOPE* s, STATEMENT* st) {
|
LINEBLOCK* compileif(SCOPE* s, IFSTATEMENT* st) {
|
||||||
IFSTATEMENT* ifst = st->ifstatement;
|
LINEBLOCK* blk = compileexpression(s, st->base->expression);
|
||||||
LINEBLOCK* blk = compileexpression(s, st->debug, ifst->base->expression);
|
|
||||||
|
|
||||||
pthread_mutex_lock(&(s->compiler->ifmutex));
|
pthread_mutex_lock(&(s->compiler->ifmutex));
|
||||||
static int ifcount = 0;
|
static int ifcount = 0;
|
||||||
|
@ -77,10 +75,10 @@ LINEBLOCK* compileif(SCOPE* s, STATEMENT* st) {
|
||||||
char* truelabelln[] = { "label", truelabel };
|
char* truelabelln[] = { "label", truelabel };
|
||||||
appendln(blk, mkln(truelabelln));
|
appendln(blk, mkln(truelabelln));
|
||||||
|
|
||||||
blk = mergelnblks(blk, compilestatements(s, ifst->base->statements));
|
blk = mergelnblks(blk, compilestatements(s, st->base->statements));
|
||||||
|
|
||||||
char* endlabel;
|
char* endlabel;
|
||||||
bool haselse = ifst->elsestatements != NULL;
|
bool haselse = st->elsestatements != NULL;
|
||||||
if(haselse) {
|
if(haselse) {
|
||||||
endlabel = mkcondlabel("IF_END", mycount);
|
endlabel = mkcondlabel("IF_END", mycount);
|
||||||
char* endgoto[] = { "goto", endlabel };
|
char* endgoto[] = { "goto", endlabel };
|
||||||
|
@ -91,7 +89,7 @@ LINEBLOCK* compileif(SCOPE* s, STATEMENT* st) {
|
||||||
appendln(blk, mkln(falselabelln));
|
appendln(blk, mkln(falselabelln));
|
||||||
|
|
||||||
if(haselse) {
|
if(haselse) {
|
||||||
blk = mergelnblks(blk, compilestatements(s, ifst->elsestatements));
|
blk = mergelnblks(blk, compilestatements(s, st->elsestatements));
|
||||||
char* endlabelln[] = { "label", endlabel };
|
char* endlabelln[] = { "label", endlabel };
|
||||||
appendln(blk, mkln(endlabelln));
|
appendln(blk, mkln(endlabelln));
|
||||||
free(endlabel);
|
free(endlabel);
|
||||||
|
@ -103,9 +101,8 @@ LINEBLOCK* compileif(SCOPE* s, STATEMENT* st) {
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilewhile(SCOPE* s, STATEMENT* st) {
|
LINEBLOCK* compilewhile(SCOPE* s, CONDSTATEMENT* w) {
|
||||||
CONDSTATEMENT* w = st->whilestatement;
|
LINEBLOCK* blk = compileexpression(s, w->expression);
|
||||||
LINEBLOCK* blk = compileexpression(s, st->debug, w->expression);
|
|
||||||
|
|
||||||
pthread_mutex_lock(&(s->compiler->whilemutex));
|
pthread_mutex_lock(&(s->compiler->whilemutex));
|
||||||
static int whilecount = 0;
|
static int whilecount = 0;
|
||||||
|
@ -137,14 +134,13 @@ LINEBLOCK* compilewhile(SCOPE* s, STATEMENT* st) {
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilelet(SCOPE* s, STATEMENT* st) {
|
LINEBLOCK* compilelet(SCOPE* s, LETSTATEMENT* l) {
|
||||||
LETSTATEMENT* l = st->letstatement;
|
LINEBLOCK* blk = compileexpression(s, l->expression);
|
||||||
LINEBLOCK* blk = compileexpression(s, st->debug, l->expression);
|
|
||||||
|
|
||||||
if(l->arrayind != NULL) {
|
if(l->arrayind != NULL) {
|
||||||
appendlnbefore(blk, onetoken("add"));
|
appendlnbefore(blk, onetoken("add"));
|
||||||
appendlnbefore(blk, pushvar(s, st->debug, l->varname));
|
appendlnbefore(blk, pushvar(s, l->varname));
|
||||||
blk = mergelnblks(compileexpression(s, st->debug, l->arrayind), blk);
|
blk = mergelnblks(compileexpression(s, l->arrayind), blk);
|
||||||
|
|
||||||
appendln(blk, poptemp());
|
appendln(blk, poptemp());
|
||||||
appendln(blk, popthatadd());
|
appendln(blk, popthatadd());
|
||||||
|
@ -152,16 +148,17 @@ LINEBLOCK* compilelet(SCOPE* s, STATEMENT* st) {
|
||||||
appendln(blk, popthat());
|
appendln(blk, popthat());
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
appendln(blk, popvar(s, st->debug, l->varname));
|
appendln(blk, popvar(s, l->varname));
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilestatement(SCOPE* s, STATEMENT* st) {
|
LINEBLOCK* compilestatement(SCOPE* s, STATEMENT* st) {
|
||||||
|
s->currdebug = st->debug;
|
||||||
if(st->type == dostatement) return compilesubroutcall(s, st->dostatement);
|
if(st->type == dostatement) return compilesubroutcall(s, st->dostatement);
|
||||||
if(st->type == returnstatement) return compileret(s, st);
|
if(st->type == returnstatement) return compileret(s, st->retstatement);
|
||||||
if(st->type == ifstatement) return compileif(s, st);
|
if(st->type == ifstatement) return compileif(s, st->ifstatement);
|
||||||
if(st->type == whilestatement) return compilewhile(s, st);
|
if(st->type == whilestatement) return compilewhile(s, st->whilestatement);
|
||||||
return compilelet(s, st);
|
return compilelet(s, st->letstatement);
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilestatements(SCOPE* s, STATEMENT* sts) {
|
LINEBLOCK* compilestatements(SCOPE* s, STATEMENT* sts) {
|
||||||
|
@ -172,15 +169,3 @@ LINEBLOCK* compilestatements(SCOPE* s, STATEMENT* sts) {
|
||||||
}
|
}
|
||||||
return head;
|
return head;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compilestatementsretlast(SCOPE* s, STATEMENT* sts, STATEMENT** retlast) {
|
|
||||||
LINEBLOCK* head = NULL;
|
|
||||||
STATEMENT* last = NULL;
|
|
||||||
while(sts != NULL) {
|
|
||||||
head = mergelnblks(head, compilestatement(s, sts));
|
|
||||||
last = sts;
|
|
||||||
sts = sts->next;
|
|
||||||
}
|
|
||||||
*retlast = last;
|
|
||||||
return head;
|
|
||||||
}
|
|
||||||
|
|
|
@ -6,5 +6,4 @@
|
||||||
* Single function for compiling statements */
|
* Single function for compiling statements */
|
||||||
|
|
||||||
LINEBLOCK* compilestatements(SCOPE* s, STATEMENT* sts);
|
LINEBLOCK* compilestatements(SCOPE* s, STATEMENT* sts);
|
||||||
LINEBLOCK* compilestatementsretlast(SCOPE* s, STATEMENT* sts, STATEMENT** retlast);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -12,7 +12,7 @@ int getobjsize(CLASS* c);
|
||||||
LINE* mksubdeclabel(CLASS* c, SUBROUTDEC* sd);
|
LINE* mksubdeclabel(CLASS* c, SUBROUTDEC* sd);
|
||||||
|
|
||||||
// Compiling methods
|
// Compiling methods
|
||||||
LINEBLOCK* compilefunbody(SCOPE* s, CLASS* cl, SUBROUTDEC* d);
|
LINEBLOCK* compilefunbody(SCOPE* s, CLASS* cl, SUBROUTBODY* b);
|
||||||
LINEBLOCK* compilefundec(SCOPE* s, CLASS* cl, SUBROUTDEC* f);
|
LINEBLOCK* compilefundec(SCOPE* s, CLASS* cl, SUBROUTDEC* f);
|
||||||
LINEBLOCK* compileconstructor(SCOPE* s, CLASS* cl, SUBROUTDEC* con);
|
LINEBLOCK* compileconstructor(SCOPE* s, CLASS* cl, SUBROUTDEC* con);
|
||||||
LINEBLOCK* compilemethod(SCOPE* s, CLASS* cl, SUBROUTDEC* m);
|
LINEBLOCK* compilemethod(SCOPE* s, CLASS* cl, SUBROUTDEC* m);
|
||||||
|
@ -64,27 +64,12 @@ LINE* mksubdeclabel(CLASS* c, SUBROUTDEC* sd) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compiling methods
|
// Compiling methods
|
||||||
LINEBLOCK* compilefunbody(SCOPE* s, CLASS* cl, SUBROUTDEC* d) {
|
LINEBLOCK* compilefunbody(SCOPE* s, CLASS* cl, SUBROUTBODY* b) {
|
||||||
SUBROUTBODY* b = d->body;
|
|
||||||
SCOPE* myscope = mkscope(s);
|
SCOPE* myscope = mkscope(s);
|
||||||
myscope->currclass = cl;
|
myscope->currclass = cl;
|
||||||
if(b->vardecs != NULL)
|
if(b->vardecs != NULL)
|
||||||
addlocalvars(myscope, b->vardecs);
|
addlocalvars(myscope, b->vardecs);
|
||||||
|
LINEBLOCK* head = compilestatements(myscope, b->statements);
|
||||||
if(b->statements == NULL) {
|
|
||||||
eprintf("Subroutine body has no statements; file '%s', line %i\n",
|
|
||||||
d->debug->file, d->debug->definedat);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATEMENT* last;
|
|
||||||
LINEBLOCK* head = compilestatementsretlast(myscope, b->statements, &last);
|
|
||||||
if(last->type != returnstatement) {
|
|
||||||
eprintf("Subroutine must end with a return statement; file '%s', line %i\n",
|
|
||||||
last->debug->file, last->debug->definedat);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
freescope(myscope);
|
freescope(myscope);
|
||||||
return head;
|
return head;
|
||||||
}
|
}
|
||||||
|
@ -92,9 +77,13 @@ LINEBLOCK* compilefunbody(SCOPE* s, CLASS* cl, SUBROUTDEC* d) {
|
||||||
LINEBLOCK* compilefundec(SCOPE* s, CLASS* cl, SUBROUTDEC* f) {
|
LINEBLOCK* compilefundec(SCOPE* s, CLASS* cl, SUBROUTDEC* f) {
|
||||||
LINE* label = mksubdeclabel(cl, f);
|
LINE* label = mksubdeclabel(cl, f);
|
||||||
|
|
||||||
LINEBLOCK* body = compilefunbody(s, cl, f);
|
if(f->body->statements != NULL) {
|
||||||
appendlnbefore(body, label);
|
LINEBLOCK* body = compilefunbody(s, cl, f->body);
|
||||||
return body;
|
appendlnbefore(body, label);
|
||||||
|
return body;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return mklnblk(label);
|
||||||
}
|
}
|
||||||
|
|
||||||
LINEBLOCK* compileconstructor(SCOPE* s, CLASS* cl, SUBROUTDEC* con) {
|
LINEBLOCK* compileconstructor(SCOPE* s, CLASS* cl, SUBROUTDEC* con) {
|
||||||
|
@ -110,7 +99,7 @@ LINEBLOCK* compileconstructor(SCOPE* s, CLASS* cl, SUBROUTDEC* con) {
|
||||||
free(size[2]);
|
free(size[2]);
|
||||||
|
|
||||||
if(con->body != NULL)
|
if(con->body != NULL)
|
||||||
return mergelnblks(blk, compilefunbody(s, cl, con));
|
return mergelnblks(blk, compilefunbody(s, cl, con->body));
|
||||||
else
|
else
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
@ -125,7 +114,7 @@ LINEBLOCK* compilemethod(SCOPE* s, CLASS* cl, SUBROUTDEC* m) {
|
||||||
appendln(blk, mkln(poppointer));
|
appendln(blk, mkln(poppointer));
|
||||||
|
|
||||||
if(m->body != NULL)
|
if(m->body != NULL)
|
||||||
return mergelnblks(blk, compilefunbody(s, cl, m));
|
return mergelnblks(blk, compilefunbody(s, cl, m->body));
|
||||||
else
|
else
|
||||||
return blk;
|
return blk;
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,16 +8,21 @@ LINE* opvarraw(SCOPE* s, char* op, VAR* v) {
|
||||||
return ln;
|
return ln;
|
||||||
}
|
}
|
||||||
|
|
||||||
LINE* pushvarraw(SCOPE* s, VAR* v) {
|
LINE* opvar(SCOPE* s, char* op, const char* name) {
|
||||||
|
VAR* v = getvar(s, name);
|
||||||
|
return opvarraw(s, op, v);
|
||||||
|
}
|
||||||
|
|
||||||
|
LINE* pushvarraw(SCOPE*s, VAR* v) {
|
||||||
return opvarraw(s, "push", v);
|
return opvarraw(s, "push", v);
|
||||||
}
|
}
|
||||||
|
|
||||||
LINE* pushvar(SCOPE* s, DEBUGINFO* d, const char* name) {
|
LINE* pushvar(SCOPE* s, const char* name) {
|
||||||
return opvarraw(s, "push", getvarmustexist(s, d, name));
|
return opvar(s, "push", name);
|
||||||
}
|
}
|
||||||
|
|
||||||
LINE* popvar(SCOPE* s, DEBUGINFO* d, const char* name) {
|
LINE* popvar(SCOPE* s, const char* name) {
|
||||||
return opvarraw(s, "pop", getvarmustexist(s, d, name));
|
return opvar(s, "pop", name);
|
||||||
}
|
}
|
||||||
|
|
||||||
LINE* poptemp() {
|
LINE* poptemp() {
|
||||||
|
|
|
@ -12,9 +12,9 @@
|
||||||
LINE* onetoken(char* str);
|
LINE* onetoken(char* str);
|
||||||
LINE* mksimpleln(char** tokens, int count);
|
LINE* mksimpleln(char** tokens, int count);
|
||||||
|
|
||||||
LINE* pushvarraw(SCOPE* s, VAR* v);
|
LINE* pushvarraw(SCOPE*s, VAR* v);
|
||||||
LINE* pushvar(SCOPE* s, DEBUGINFO* d, const char* name);
|
LINE* pushvar(SCOPE* s, const char* name);
|
||||||
LINE* popvar(SCOPE* s, DEBUGINFO* d, const char* name);
|
LINE* popvar(SCOPE* s, const char* name);
|
||||||
LINE* poptemp();
|
LINE* poptemp();
|
||||||
LINE* popthatadd();
|
LINE* popthatadd();
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue