diff --git a/emu/cpu/instruction.c b/emu/cpu/instruction.c index 4ebf50a..12fa1b5 100644 --- a/emu/cpu/instruction.c +++ b/emu/cpu/instruction.c @@ -1,6 +1,64 @@ #include "./instruction.h" +#include "../cpu_human.h" #include #include +#include + +void LogInstructionData(CpuInstructions instruction, ArgumentInfo arg1Info, + ArgumentInfo arg2Info, r16_int arg1, r16_int arg2, + r16_int arg1Val, r16_int arg2Val) +{ + char *hReadIns = CpuInstructionsHumanReadable[instruction], + *hReadA1I = ArgumentInfoHumanReadable[arg1Info], + *hReadA2I = ArgumentInfoHumanReadable[arg1Info], + *actArg1 = malloc(sizeof(char) * 32), + *actArg2 = malloc(sizeof(char) * 32), + *hReadA1 = malloc(sizeof(char) * 32), + *hReadA2 = malloc(sizeof(char) * 32); + + sprintf(actArg1, "0x%X", arg2Val.u); + sprintf(actArg2, "0x%X", arg2Val.u); + + switch (arg1Info) + { + case value: + hReadA1 = "~"; + break; + case valueInRam: + sprintf(hReadA1, "0x%X", arg1.u); + break; + case valueInRegister: + hReadA1 = CpuRegistersHumanReadable[arg1.u]; + break; + case pointerInRegister: + sprintf(hReadA1, "%s ~ 0x%X", CpuRegistersHumanReadable[arg1.u], GetRegister(arg1.u).u); + break; + } + + switch (arg2Info) + { + case value: + hReadA2 = "~"; + break; + case valueInRam: + sprintf(hReadA2, "0x%X", arg2.u); + break; + case valueInRegister: + hReadA2 = CpuRegistersHumanReadable[arg2.u]; + break; + case pointerInRegister: + sprintf(hReadA2, "%s ~ 0x%X", CpuRegistersHumanReadable[arg2.u], GetRegister(arg2.u).u); + break; + } + + printf("%s :: (%s) %s -> %s, (%s) %s -> %s\n", + hReadIns, hReadA1I, hReadA1, actArg1, hReadA2I, hReadA2, actArg2); + + free(actArg1); + free(actArg2); + free(hReadA1); + free(hReadA2); +} r16_int GetArgVal(ArgumentInfo argInfo, r16_int arg) { @@ -34,71 +92,91 @@ void ExecuteInstruction(CpuInstructions instruction, ArgumentInfo arg1Info, { case HLT: printf("\nprogram HALTED!\n"); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, (r16_int){ 0 }, (r16_int){ 0 }); exit(0); case ADD: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s + actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case SUB: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s - actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case MUL: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s * actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case DIV: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s / actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case LBS: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s << actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case RBS: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.s >> actArg2.s }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case BAN: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.u & actArg2.u }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case BOR: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.u | actArg2.u }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case BXO: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetRegister(ACU, (r16_int){ actArg1.u ^ actArg2.u }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case BNO: - SetRegister(ACU, (r16_int){ ~ GetArgVal(arg1Info, arg1).u }); + actArg1 = GetArgVal(arg1Info, arg1); + SetRegister(ACU, (r16_int){ ~ actArg1.u }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, (r16_int){ 0 }); break; case PUS: - StackPush(GetArgVal(arg1Info, arg1)); + actArg1 = GetArgVal(arg1Info, arg1); + StackPush(actArg1); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, (r16_int){ 0 }); break; case POP: SetRegister(ACU, StackPop()); break; case JMP: SetRegister(PC, GetArgVal(arg1Info, arg1)); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, (r16_int){ 0 }, (r16_int){ 0 }); break; case JEQ: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); if (actArg2.u == GetRegister(ACU).u) SetRegister(PC, actArg1); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case JNZ: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); if (actArg2.s != 0) SetRegister(PC, actArg1); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; case CAL: + actArg1 = GetArgVal(arg1Info, arg1); StackPush(GetRegister(PC)); - SetRegister(PC, GetArgVal(arg1Info, arg1)); + SetRegister(PC, actArg1); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, (r16_int){ 0 }); break; case RET: SetRegister(PC, StackPop()); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, (r16_int){ 0 }, (r16_int){ 0 }); break; case REG: // TODO case INT: // TODO @@ -106,9 +184,12 @@ void ExecuteInstruction(CpuInstructions instruction, ArgumentInfo arg1Info, exit(-1); case INP: SetRegister(ACU, (r16_int){ getchar() }); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, (r16_int){ 0 }, (r16_int){ 0 }); break; case OUT: + actArg1 = GetArgVal(arg1Info, arg1); putchar((char)(GetArgVal(arg1Info, arg1).u)); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, (r16_int){ 0 }); break; case DIN: // TODO case DOT: // TODO @@ -117,6 +198,7 @@ void ExecuteInstruction(CpuInstructions instruction, ArgumentInfo arg1Info, case MOV: GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2); SetValue(actArg1, actArg2); + LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2); break; } } \ No newline at end of file diff --git a/emu/cpu/instruction.h b/emu/cpu/instruction.h index a3900a5..ae28004 100644 --- a/emu/cpu/instruction.h +++ b/emu/cpu/instruction.h @@ -8,5 +8,8 @@ void ExecuteInstruction(CpuInstructions instruction, ArgumentInfo arg1Info, ArgumentInfo arg2Info, r16_int arg1, r16_int arg2); +void LogInstructionData(CpuInstructions instruction, ArgumentInfo arg1Info, + ArgumentInfo arg2Info, r16_int arg1, r16_int arg2, + r16_int arg1Val, r16_int arg2Val); #endif \ No newline at end of file diff --git a/emu/cpu/register.c b/emu/cpu/register.c index 911da2d..2711855 100644 --- a/emu/cpu/register.c +++ b/emu/cpu/register.c @@ -1,8 +1,15 @@ #include "./register.h" #include +#include r16_int r0, r1, r2, r3, acu, pc, sp; +void LogRegisters() +{ + printf(" r0: 0x%X\n r1: 0x%X\n r2: 0x%X\n r3: 0x%X\nACU: 0x%X\n PC: 0x%X\n SP: 0x%X\n", + r0.u, r1.u, r2.u, r3.u, acu.u, pc.u, sp.u); +} + r16_int *GetRegPoint(CpuRegisters reg) { switch (reg) diff --git a/emu/cpu/register.h b/emu/cpu/register.h index bc213db..cc524ea 100644 --- a/emu/cpu/register.h +++ b/emu/cpu/register.h @@ -4,6 +4,7 @@ #include "../cpu_const.h" #include "./ram.h" +void LogRegisters(); r16_int GetRegister(CpuRegisters reg); void SetRegister(CpuRegisters reg, r16_int value); diff --git a/emu/cpu/stack.c b/emu/cpu/stack.c index cd378cd..9e9ffa0 100644 --- a/emu/cpu/stack.c +++ b/emu/cpu/stack.c @@ -1,6 +1,20 @@ #include "./stack.h" +#include "./register.h" #include #include +#include + +void LogStack(bool onlyUpToSP, bool pointOutSP) +{ + char *pointer = "->"; + char *nointer = " "; + + for (int i = 0; i < (onlyUpToSP ? GetRegister(SP).u : STACK_SIZE); i++) + { + printf("%s 0x%x: 0x%x\n", ((pointOutSP && i == GetRegister(SP).u) ? pointer : nointer), + i, GetValue((r16_int){ STACK_LOCATION + i }).u); + } +} r16_int StackPop() { diff --git a/emu/cpu_const.h b/emu/cpu_const.h index 6975821..95d4923 100644 --- a/emu/cpu_const.h +++ b/emu/cpu_const.h @@ -55,4 +55,8 @@ typedef enum #define INSTRUCTION_ARG_1_INFO_MASK 0b0011000000000000 #define INSTRUCTION_ARG_2_INFO_MASK 0b1100000000000000 +#define R16 "Reduced 16 bit CPU" +#define R16_AUTHOR "Rose Apollo" +#define R16_STANDARD "r16_0 version 1" + #endif \ No newline at end of file diff --git a/emu/cpu_human.h b/emu/cpu_human.h new file mode 100644 index 0000000..0d96e58 --- /dev/null +++ b/emu/cpu_human.h @@ -0,0 +1,53 @@ +#ifndef EMU_CPU_HUMAN_H +#define EMU_CPU_HUMAN_H + +char CpuRegistersHumanReadable[9][4] = +{ + "R0", + "R1", + "R2", + "R3", + "R4", + "ACU", + "PC", + "SP" +}; + +char ArgumentInfoHumanReadable[4][21] = +{ + "Value", + "Value in register", + "Value in ram", + "Pointer in register" +}; + +char CpuInstructionsHumanReadable[25][4] = +{ + "HLT", + "ADD", + "SUB", + "MUL", + "DIV", + "LBS", + "RBS", + "BAN", + "BOR", + "BXO", + "BNO", + "PUS", + "POP", + "JMP", + "JEQ", + "JNZ", + "CAL", + "RET", + "REG", + "INT", + "INP", + "OUT", + "DIN", + "DOT", + "MOV" +}; + +#endif \ No newline at end of file diff --git a/emu/main.c b/emu/main.c index c05f1ed..0e804e9 100644 --- a/emu/main.c +++ b/emu/main.c @@ -8,7 +8,11 @@ int main(int argc, char **argv) { + printf("R16 " R16 "\n" R16_STANDARD "\n(c) " R16_AUTHOR "\n"); + carg_parse_data* args = carg_parse(argc, argv); carg_parse_free(args); + + return 0; }