2024-04-19 10:08:32 +00:00
|
|
|
#include "./instruction.h"
|
2024-04-22 11:08:57 +00:00
|
|
|
#include "../cpu_human.h"
|
2024-04-19 12:11:28 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2024-04-22 11:08:57 +00:00
|
|
|
#include <inttypes.h>
|
2024-04-25 09:47:01 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdbool.h>
|
2024-04-22 11:08:57 +00:00
|
|
|
|
2024-04-25 09:47:01 +00:00
|
|
|
bool LogInstructionData(CpuInstructions instruction, ArgumentInfo arg1Info,
|
2024-04-22 11:08:57 +00:00
|
|
|
ArgumentInfo arg2Info, r16_int arg1, r16_int arg2,
|
|
|
|
r16_int arg1Val, r16_int arg2Val)
|
|
|
|
{
|
2024-04-25 09:47:01 +00:00
|
|
|
#ifdef LOG_INSTRUCTION_DATA
|
2024-04-22 11:08:57 +00:00
|
|
|
char *hReadIns = CpuInstructionsHumanReadable[instruction],
|
|
|
|
*hReadA1I = ArgumentInfoHumanReadable[arg1Info],
|
2024-04-25 09:47:01 +00:00
|
|
|
*hReadA2I = ArgumentInfoHumanReadable[arg2Info],
|
|
|
|
actArg1[32] = { 0 }, actArg2[32] = { 0 },
|
|
|
|
hReadA1[32] = { 0 }, hReadA2[32] = { 0 };
|
|
|
|
|
|
|
|
|
2024-04-22 11:08:57 +00:00
|
|
|
|
2024-04-24 12:11:00 +00:00
|
|
|
sprintf(actArg1, "0x%X", arg1Val.u);
|
2024-04-22 11:08:57 +00:00
|
|
|
sprintf(actArg2, "0x%X", arg2Val.u);
|
|
|
|
|
|
|
|
switch (arg1Info)
|
|
|
|
{
|
|
|
|
case value:
|
2024-04-25 09:47:01 +00:00
|
|
|
hReadA1[0] = '~';
|
2024-04-22 11:08:57 +00:00
|
|
|
break;
|
|
|
|
case valueInRam:
|
|
|
|
sprintf(hReadA1, "0x%X", arg1.u);
|
|
|
|
break;
|
|
|
|
case valueInRegister:
|
2024-04-25 09:47:01 +00:00
|
|
|
strncpy(hReadA1, CpuRegistersHumanReadable[arg1.u], 31);
|
2024-04-22 11:08:57 +00:00
|
|
|
break;
|
|
|
|
case pointerInRegister:
|
|
|
|
sprintf(hReadA1, "%s ~ 0x%X", CpuRegistersHumanReadable[arg1.u], GetRegister(arg1.u).u);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (arg2Info)
|
|
|
|
{
|
|
|
|
case value:
|
2024-04-25 09:47:01 +00:00
|
|
|
hReadA2[0] = '~';
|
2024-04-22 11:08:57 +00:00
|
|
|
break;
|
|
|
|
case valueInRam:
|
|
|
|
sprintf(hReadA2, "0x%X", arg2.u);
|
|
|
|
break;
|
|
|
|
case valueInRegister:
|
2024-04-25 09:47:01 +00:00
|
|
|
strncpy(hReadA2, CpuRegistersHumanReadable[arg2.u], 31);
|
2024-04-22 11:08:57 +00:00
|
|
|
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);
|
2024-04-25 09:47:01 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2024-04-22 11:08:57 +00:00
|
|
|
}
|
2024-04-19 10:08:32 +00:00
|
|
|
|
|
|
|
r16_int GetArgVal(ArgumentInfo argInfo, r16_int arg)
|
|
|
|
{
|
2024-04-19 12:11:28 +00:00
|
|
|
switch (argInfo)
|
|
|
|
{
|
|
|
|
case value:
|
|
|
|
return arg;
|
|
|
|
case valueInRegister:
|
|
|
|
return GetRegister((CpuRegisters)arg.u);
|
|
|
|
case valueInRam:
|
|
|
|
return GetValue(arg);
|
|
|
|
case pointerInRegister:
|
|
|
|
return GetValue(GetRegister((CpuRegisters)arg.u));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GetAllArgVal(ArgumentInfo arg1Info, ArgumentInfo arg2Info,
|
|
|
|
r16_int arg1, r16_int arg2,
|
|
|
|
r16_int *arg1Out, r16_int *arg2Out)
|
|
|
|
{
|
|
|
|
*arg1Out = GetArgVal(arg1Info, arg1);
|
|
|
|
*arg2Out = GetArgVal(arg2Info, arg2);
|
2024-04-19 10:08:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ExecuteInstruction(CpuInstructions instruction, ArgumentInfo arg1Info,
|
|
|
|
ArgumentInfo arg2Info, r16_int arg1, r16_int arg2)
|
|
|
|
{
|
2024-04-19 12:11:28 +00:00
|
|
|
r16_int actArg1, actArg2;
|
2024-04-25 09:47:01 +00:00
|
|
|
GetAllArgVal(arg1Info, arg2Info, arg1, arg2, &actArg1, &actArg2);
|
|
|
|
LogInstructionData(instruction, arg1Info, arg2Info, arg1, arg2, actArg1, actArg2);
|
2024-04-19 12:11:28 +00:00
|
|
|
|
2024-04-19 10:08:32 +00:00
|
|
|
switch (instruction)
|
|
|
|
{
|
|
|
|
case HLT:
|
2024-04-19 12:11:28 +00:00
|
|
|
printf("\nprogram HALTED!\n");
|
2024-04-19 10:08:32 +00:00
|
|
|
exit(0);
|
|
|
|
case ADD:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s + actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case SUB:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s - actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case MUL:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s * actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case DIV:
|
2024-04-24 12:11:00 +00:00
|
|
|
if (actArg2.s == 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "DIV by 0\n");
|
|
|
|
exit(-1);
|
|
|
|
}
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s / actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case LBS:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s << actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case RBS:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.s >> actArg2.s });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case BAN:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.u & actArg2.u });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case BOR:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.u | actArg2.u });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case BXO:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ actArg1.u ^ actArg2.u });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case BNO:
|
2024-04-22 11:08:57 +00:00
|
|
|
SetRegister(ACU, (r16_int){ ~ actArg1.u });
|
2024-04-19 12:11:28 +00:00
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case PUS:
|
2024-04-22 11:08:57 +00:00
|
|
|
StackPush(actArg1);
|
2024-04-19 12:11:28 +00:00
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case POP:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, StackPop());
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case JMP:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(PC, GetArgVal(arg1Info, arg1));
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case JEQ:
|
2024-04-19 12:11:28 +00:00
|
|
|
if (actArg2.u == GetRegister(ACU).u)
|
|
|
|
SetRegister(PC, actArg1);
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case JNZ:
|
2024-04-19 12:11:28 +00:00
|
|
|
if (actArg2.s != 0)
|
|
|
|
SetRegister(PC, actArg1);
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case CAL:
|
2024-04-19 12:11:28 +00:00
|
|
|
StackPush(GetRegister(PC));
|
2024-04-22 11:08:57 +00:00
|
|
|
SetRegister(PC, actArg1);
|
2024-04-19 12:11:28 +00:00
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case RET:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(PC, StackPop());
|
|
|
|
break;
|
|
|
|
case REG: // TODO
|
|
|
|
case INT: // TODO
|
|
|
|
fprintf(stderr, "INTS NOT IMPLEMENTED YET\n");
|
|
|
|
exit(-1);
|
2024-04-19 10:08:32 +00:00
|
|
|
case INP:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetRegister(ACU, (r16_int){ getchar() });
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
case OUT:
|
2024-04-19 12:11:28 +00:00
|
|
|
putchar((char)(GetArgVal(arg1Info, arg1).u));
|
|
|
|
break;
|
|
|
|
case DIN: // TODO
|
|
|
|
case DOT: // TODO
|
|
|
|
fprintf(stderr, "DISK NOT IMPLEMENTED YET\n");
|
|
|
|
exit(-1);
|
2024-04-19 10:08:32 +00:00
|
|
|
case MOV:
|
2024-04-19 12:11:28 +00:00
|
|
|
SetValue(actArg1, actArg2);
|
|
|
|
break;
|
2024-04-19 10:08:32 +00:00
|
|
|
}
|
|
|
|
}
|