123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344 |
- #include <setjmp.h>
- #include <stdarg.h>
- #include <stdio.h>
- #include "Compiler.h"
- #include "FunctionMap.h"
- #include "Operation.h"
- #include "StringIntMap.h"
- #include "Tokenizer.h"
- #define ERROR_LENGTH 256
- #define RETURN_BUFFER 16
- static jmp_buf errorJump;
- static char error[ERROR_LENGTH] = {'\0'};
- static ByteCode* code;
- static int16 line = 1;
- static int varIndex = 0;
- static StringIntMap vars[2];
- static FunctionMap functions;
- static int returns[RETURN_BUFFER];
- static int returnIndex = 0;
- static void cError(const char* format, ...) {
- va_list args;
- va_start(args, format);
- vsnprintf(error, ERROR_LENGTH, format, args);
- va_end(args);
- longjmp(errorJump, 0);
- }
- static int cAddVar(const char* var) {
- int index = vars[varIndex].entries;
- simAdd(vars + varIndex, var, &index);
- return index;
- }
- static void cUnexpectedToken(Token t) {
- cError("unexpected token on line %d: %s", line, tGetTokenName(t));
- }
- static void cAddOperation(Operation token) {
- unsigned char c = token;
- bcAddBytes(code, &c, 1);
- }
- static int cReserveInt() {
- return bcReserveBytes(code, sizeof(int));
- }
- static void cSetInt(int p, int i) {
- bcSetBytes(code, p, &i, sizeof(int));
- }
- static void cAddInt(int i) {
- bcAddBytes(code, &i, sizeof(int));
- }
- static void cAddInt16(int16 i) {
- bcAddBytes(code, &i, sizeof(int16));
- }
- static void cAddFloat(float f) {
- bcAddBytes(code, &f, sizeof(float));
- }
- static int cAddPush(int offset) {
- cAddOperation(OP_PUSH);
- int p = cReserveInt();
- cAddInt(offset);
- return p;
- }
- static void cAddPop(int p, int vars) {
- cAddOperation(OP_POP);
- cAddInt(vars);
- cSetInt(p, vars);
- }
- static Token cReadTokenAndLine() {
- Token t = tReadToken();
- if(tReadInt16(&line)) {
- return t;
- }
- return T_END;
- }
- static void cConsumeToken(Token wanted) {
- Token t = cReadTokenAndLine();
- if(wanted != t) {
- cError("unexpected token on line %d: expected '%s' got '%s'", line, tGetTokenName(wanted), tGetTokenName(t));
- }
- }
- static bool cConsumeTokenIf(Token t) {
- if(tPeekToken() == t) {
- cReadTokenAndLine();
- return true;
- }
- return false;
- }
- static void cConstantInt() {
- int value;
- if(!tReadInt(&value)) {
- cError("int token without an int on line %d", line);
- }
- cAddOperation(OP_PUSH_INT);
- cAddInt(value);
- }
- static void cConstantFloat() {
- float value;
- if(!tReadFloat(&value)) {
- cError("float token without a float on line %d", line);
- }
- cAddOperation(OP_PUSH_FLOAT);
- cAddFloat(value);
- }
- static const char* cReadString() {
- const char* literal = tReadString();
- if(literal == NULL) {
- cError("literal without string on line %d", line);
- }
- return literal;
- }
- static void cGetVar() {
- cAddOperation(OP_GET);
- cAddInt(cAddVar(cReadString()));
- }
- static void cExpression();
- static void cPrimary() {
- Token t = cReadTokenAndLine();
- switch(t) {
- case T_INT: cConstantInt(); break;
- case T_FLOAT: cConstantFloat(); break;
- case T_NULL: cAddOperation(OP_PUSH_NULL); break;
- case T_TRUE: cAddOperation(OP_PUSH_TRUE); break;
- case T_FALSE: cAddOperation(OP_PUSH_FALSE); break;
- case T_OPEN_BRACKET:
- cExpression();
- cConsumeToken(T_CLOSE_BRACKET);
- break;
- case T_LITERAL: cGetVar(); break;
- default: cUnexpectedToken(t); break;
- }
- }
- static void cMul() {
- cPrimary();
- while(cConsumeTokenIf(T_MUL)) {
- cPrimary();
- cAddOperation(OP_MUL);
- }
- }
- static void cAdd() {
- cMul();
- while(cConsumeTokenIf(T_ADD)) {
- cMul();
- cAddOperation(OP_ADD);
- }
- }
- static void cExpression() {
- cAdd();
- }
- static void cSetVar(const char* literal) {
- cExpression();
- cConsumeToken(T_SEMICOLON);
- cAddOperation(OP_SET);
- cAddInt(cAddVar(literal));
- }
- static int cCallFunctionArguments() {
- int arguments = 0;
- while(!cConsumeTokenIf(T_CLOSE_BRACKET)) {
- arguments++;
- cExpression();
- if(cConsumeTokenIf(T_COMMA) && tPeekToken() == T_CLOSE_BRACKET) {
- cUnexpectedToken(tPeekToken());
- }
- }
- return arguments;
- }
- static void cCallFunction(const char* literal) {
- cAddOperation(OP_PUSH_INT);
- cAddInt(0);
- int arguments = cCallFunctionArguments();
- int address = fmSearchAddress(&functions, literal, arguments);
- if(address == -1) {
- cError("unknown function on line %d", line);
- }
- cAddOperation(OP_GOSUB);
- cAddInt(address);
- cAddInt(arguments);
- cConsumeToken(T_SEMICOLON);
- }
- static void cLiteral() {
- const char* literal = cReadString();
- Token t = cReadTokenAndLine();
- switch(t) {
- case T_SET: cSetVar(literal); break;
- case T_OPEN_BRACKET: cCallFunction(literal); break;
- default: cUnexpectedToken(t);
- }
- }
- static int cFunctionArguments() {
- int arguments = 0;
- while(!cConsumeTokenIf(T_CLOSE_BRACKET)) {
- cConsumeToken(T_LITERAL);
- arguments++;
- cAddVar(cReadString());
- if(cConsumeTokenIf(T_COMMA) && tPeekToken() != T_LITERAL) {
- cUnexpectedToken(tPeekToken());
- }
- }
- return arguments;
- }
- static void cLine(Token t);
- static void cFunctionInnerBody(int arguments) {
- int p = cAddPush(arguments);
- int oldLine = line;
- while(!cConsumeTokenIf(T_CLOSE_CURVED_BRACKET)) {
- Token t = cReadTokenAndLine();
- if(t == T_END) {
- cError("unexpected end of file: function not closed on line %d", oldLine);
- }
- cLine(t);
- }
- cAddPop(p, vars[1].entries);
- for(int i = 0; i < returnIndex; i++) {
- cSetInt(returns[i], vars[1].entries);
- }
- returnIndex = 0;
- }
- static void cFunctionBody(const char* name, int arguments) {
- cConsumeToken(T_OPEN_CURVED_BRACKET);
- cAddOperation(OP_GOTO);
- int gotoIndex = cReserveInt();
- if(!fmAdd(&functions, name, arguments, code->length)) {
- cError("function registered twice on line %d", line);
- }
- cFunctionInnerBody(arguments);
- cAddOperation(OP_RETURN);
- cSetInt(gotoIndex, code->length);
- }
- static void cFunction() {
- if(varIndex == 1) {
- cError("function inside function on line %d", line);
- }
- cConsumeToken(T_LITERAL);
- const char* name = cReadString();
- cConsumeToken(T_OPEN_BRACKET);
- varIndex = 1;
- vars[1].entries = 0;
- cFunctionBody(name, cFunctionArguments());
- varIndex = 0;
- }
- static void cReturn() {
- if(returnIndex >= RETURN_BUFFER) {
- cError("too much returns in function around line %d", line);
- }
- cAddOperation(OP_POP);
- returns[returnIndex++] = cReserveInt(vars);
- cAddOperation(OP_RETURN);
- cConsumeToken(T_SEMICOLON);
- }
- static void cPrint() {
- cExpression();
- cConsumeToken(T_SEMICOLON);
- cAddOperation(OP_PRINT);
- }
- static void cLine(Token t) {
- cAddOperation(OP_LINE);
- cAddInt16(line);
- switch(t) {
- case T_PRINT: cPrint(); break;
- case T_LITERAL: cLiteral(); break;
- case T_FUNCTION: cFunction(); break;
- case T_RETURN: cReturn(); break;
- default: cUnexpectedToken(t);
- }
- }
- static void cForEachLine() {
- Token t = cReadTokenAndLine();
- while(t != T_END) {
- cLine(t);
- t = cReadTokenAndLine();
- }
- }
- static void cAllocAndCompile() {
- varIndex = 0;
- returnIndex = 0;
- simInit(vars);
- simInit(vars + 1);
- fmInit(&functions);
- if(!setjmp(errorJump)) {
- int p = cAddPush(0);
- cForEachLine();
- cAddPop(p, vars[varIndex].entries);
- }
- fmDelete(&functions);
- simDelete(vars + 1);
- simDelete(vars);
- }
- ByteCode* cCompile() {
- error[0] = '\0';
- code = bcInit();
- cAllocAndCompile();
- if(error[0] != '\0') {
- bcDelete(code);
- return NULL;
- }
- return code;
- }
- const char* cGetError() {
- return error;
- }
|