njvm/njvm.c
2023-10-26 14:03:31 +02:00

194 lines
4.1 KiB
C

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
//Comment to disable debug
#define DEBUG
#define IMMEDIATE(x) ((x) & 0x00FFFFFF)
#define HALT 0
#define PUSHC 1
#define ADD 2
#define SUB 3
#define MUL 4
#define DIV 5
#define MOD 6
#define RDINT 7
#define WRINT 8
#define RDCHR 9
#define WRCHR 10
unsigned int programmSpeicher[1000];
unsigned int code1[] = {
(PUSHC << 24) | IMMEDIATE(3),
(PUSHC << 24) | IMMEDIATE(4),
(ADD << 24),
(PUSHC << 24) | IMMEDIATE(10),
(PUSHC << 24) | IMMEDIATE(6),
(SUB << 24),
(MUL << 24),
(WRINT << 24),
(PUSHC << 24) | IMMEDIATE(10),
(WRCHR << 24),
(HALT)
};
unsigned int code2[] = {
(PUSHC << 24) | IMMEDIATE(-2),
(RDINT << 24),
(MUL << 24),
(PUSHC << 24) | IMMEDIATE(3),
(ADD << 24),
(WRINT << 24),
(PUSHC << 24) | IMMEDIATE('\n'),
(WRCHR << 24),
(HALT << 24)
};
unsigned int code3[] = {
(RDCHR << 24),
(WRINT << 24),
(PUSHC << 24) | IMMEDIATE('\n'),
(WRCHR << 24),
(HALT << 24)
};
void copyToProgramm(unsigned int codeToCopy[]){
int size = sizeof(codeToCopy)/sizeof(codeToCopy[0]);
for (int i = 0; i < size; i++) {
printf("Test\n");
}
printf("%i",size);
}
// Stack
#define maxValues 1000
unsigned int stack[maxValues];
unsigned int current = 0;
void stackPush(unsigned int value) {
if (current > (maxValues-1)) {
fprintf(stderr, "stack overflow\n");
exit(EXIT_FAILURE);
}
stack[current] = value;
current++;
}
unsigned int stackPop(void) {
if (current < 1) {
fprintf(stderr, "stack underflow\n");
exit(EXIT_FAILURE);
}
return stack[--current];
}
void version(void) {
printf("Ninja Virtual Machine version %i (compiled %s, %s)\n", 0, __DATE__, __TIME__);
}
void help(void) {
printf("usage: ./njvm [option] [option] ...\n\t--version\tshow version and exit\n\t--help\t\tshow this help and exit\n");
}
void useption(int argc, char *argv[]) {
if (argc > 1) {
if (strcmp(argv[1], "--version") == 0) {
version();
} else if (strcmp(argv[1], "--help") == 0) {
help();
} else {
printf("unknown command line argument '%s', try './njvm --help'", argv[1]);
}
}
}
void execute(void) {
int i = 0;
int intInput;
int temp;
char charInput;
while (programmSpeicher[i] != 0) {
switch (programmSpeicher[i] >> 24) {
case HALT:
goto end;
break;
case PUSHC:
stackPush(IMMEDIATE(programmSpeicher[i]));
break;
case ADD:
stackPush(stackPop()+stackPop());
break;
case SUB:
temp = stackPop();
stackPush(stackPop() - temp);
break;
case MUL:
stackPush(stackPop()*stackPop());
break;
case DIV:
temp = stackPop();
stackPush(stackPop()/temp);
break;
case MOD:
temp = stackPop();
stackPush(stackPop()%temp);
break;
case RDINT:
scanf("%i",&intInput);
stackPush(intInput);
break;
case WRINT:
printf("%i\n",stackPop());
break;
case RDCHR:
scanf("%c",&charInput);
stackPush(charInput);
break;
case WRCHR:
printf("%c\n",stackPop());
break;
}
i++;
}
end:
printf("Finished\n");
}
#ifdef DEBUG
void printStack(void){
if(current >= 0) return;
printf("----\n");
for (int i = 0; i < current; i++) {
printf("|%i| \n",stack[i]);
}
printf("----\n");
}
void tests(void){
printf("Runnig debug mode\n");
copyToProgramm(code1);
}
#endif /* ifdef DEBUG */
int main(int argc, char *argv[]) {
#ifdef DEBUG
tests();
#endif /* ifdef DEBUG */
if (argc > 1) useption(argc, argv);
else {
// Started
printf("Ninja Virtual Machine started\n");
// Stopped
printf("Ninja Virtual Machine stopped\n");
return 0;
}
}