#include #include #include "instruktion.c" #include "code.c" #include "stack.c" #include "program.c" #include "codeReader.c" #include "SDA.c" //Comment to disable debug #define DEBUG // Stack struct stack stack; #define SIZE 1000 // Program struct program program; // SDA struct sda sda; unsigned fp; void version(void) { printf("Ninja Virtual Machine version %i (compiled %s, %s)\n", 0, __DATE__, __TIME__); } void help(void) { printf("Usage: ./njvm [options] \n\t--debug\tstart virtual machine in debug mode\n\t--version\tshow version and exit\n\t--help\t\tshow this help and exit\n"); } void execute(struct program program) { int i; int intInput; unsigned int temp; char charInput; for (i = 0; i < *program.size; ++i) { switch (program.program[i] >> 24) { case HALT: goto end; case PUSHC: push(stack, SIGN_EXTEND(IMMEDIATE(program.program[i]))); break; case ADD: push(stack, pop(stack) + pop(stack)); break; case SUB: temp = pop(stack); push(stack, pop(stack) - temp); break; case MUL: push(stack, pop(stack) * pop(stack)); break; case DIV: temp = pop(stack); push(stack, pop(stack) / temp); break; case MOD: temp = pop(stack); push(stack, pop(stack) % temp); break; case RDINT: scanf("%i", &intInput); push(stack, intInput); break; case WRINT: printf("%i", pop(stack)); break; case RDCHR: scanf("%c", &charInput); push(stack, charInput); break; case WRCHR: printf("%c", pop(stack)); break; case PUSHG: push(stack, getSDA(SIGN_EXTEND(IMMEDIATE(program.program[i])), sda)); break; case POPG: setSDA(SIGN_EXTEND(IMMEDIATE(program.program[i])), pop(stack), sda); break; case ASF: push(stack, *stack.current); fp = *stack.current; *stack.current = *stack.current + SIGN_EXTEND(IMMEDIATE(program.program[i])); break; case RSF: *stack.current = fp; fp = pop(stack); case PUSHL: stack.stack[fp + SIGN_EXTEND(IMMEDIATE(program.program[i]))] = pop(stack); break; case POPL: push(stack, stack.stack[fp + SIGN_EXTEND(IMMEDIATE(program.program[i]))]); break; } } end: return; } #ifdef DEBUG void tests(void) { printf("Runnig debug mode\n"); int temp = fromFile("./prog1.bin", program); int sizeSDA = temp; unsigned int s[sizeSDA]; sda.size = &temp; sda.sda = s; printProgram(program); } #endif /* ifdef DEBUG */ int main(int argc, char *argv[]) { // Initialize the Stack int size = SIZE; int current = 0; unsigned int s[SIZE]; stack.size = &size; stack.current = ¤t; stack.stack = s; // Initialize ProgrammSpeicher int psize = 1000; int saveProgram = 0; unsigned int p[1000]; program.size = &psize; program.program = p; program.saveProgram = &saveProgram; #ifdef DEBUG tests(); #endif /* ifdef DEBUG */ if (argc > 1) { if (strcmp(argv[1], "--version") == 0) { version(); } else if (strcmp(argv[1], "--help") == 0) { help(); } else if (strcmp(argv[1], "--prog1") == 0) { copyToProgram(code1, sizeof(code1) / sizeof(code1[0]), program); goto run; } else if (strcmp(argv[1], "--prog2") == 0) { copyToProgram(code2, sizeof(code2) / sizeof(code2[0]), program); goto run; } else if (strcmp(argv[1], "--prog3") == 0) { copyToProgram(code3, sizeof(code3) / sizeof(code3[0]), program); goto run; } else { printf("unknown command line argument '%s', try './njvm --help'", argv[1]); } } else { run: // Started if (*program.saveProgram == 1) { printf("Ninja Virtual Machine started\n"); printProgram(program); execute(program); printSDA(sda); } else { printf("Error: no code file specified\n"); return 1; } // Stopped printf("Ninja Virtual Machine stopped\n"); return 0; } }