2025-01-30 17:03:56 -07:00
|
|
|
#include <cpu.h>
|
|
|
|
#include <emu.h>
|
2025-01-30 21:59:05 -07:00
|
|
|
#include <bus.h>
|
2025-01-30 22:54:33 -07:00
|
|
|
#include <stack.h>
|
2025-01-30 17:03:56 -07:00
|
|
|
|
|
|
|
//process CPU instructions...
|
|
|
|
|
2025-02-01 00:48:49 -07:00
|
|
|
void cpu_set_flags(cpu_context *ctx, int8_t z, int8_t n, int8_t h, int8_t c){
|
2025-01-30 18:38:29 -07:00
|
|
|
if (z != -1){
|
|
|
|
BIT_SET(ctx->regs.f, 7, z)
|
|
|
|
}
|
|
|
|
if (n != -1){
|
|
|
|
BIT_SET(ctx->regs.f, 6, n)
|
|
|
|
}
|
|
|
|
if (h != -1){
|
|
|
|
BIT_SET(ctx->regs.f, 5, h)
|
|
|
|
}
|
|
|
|
if (c != -1){
|
|
|
|
BIT_SET(ctx->regs.f, 4, c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-31 12:24:55 -07:00
|
|
|
static bool is_16_bit(reg_type rt){
|
|
|
|
return rt >= RT_AF;
|
|
|
|
}
|
|
|
|
|
2025-01-30 21:59:05 -07:00
|
|
|
static void proc_none(cpu_context *ctx) {
|
|
|
|
printf("INVALID INSTRUCTION!\n");
|
|
|
|
exit(-7);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_nop(cpu_context *ctx) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2025-01-31 14:39:38 -07:00
|
|
|
static void proc_daa(cpu_context *ctx) {
|
|
|
|
u8 u = 0;
|
|
|
|
int fc = 0;
|
|
|
|
|
2025-01-31 17:07:09 -07:00
|
|
|
if(CPU_FLAG_H || (!CPU_FLAG_N && (ctx->regs.a & 0xF) > 9)) {
|
2025-01-31 14:39:38 -07:00
|
|
|
u = 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(CPU_FLAG_C || (!CPU_FLAG_N && ctx->regs.a > 0x99)) {
|
|
|
|
u |= 0x60;
|
|
|
|
fc = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx->regs.a += CPU_FLAG_N ? -u : u;
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, ctx->regs.a == 0, -1, 0, fc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_cpl(cpu_context *ctx) {
|
2025-02-01 00:48:49 -07:00
|
|
|
ctx->regs.a = ~ctx->regs.a;
|
2025-01-31 14:39:38 -07:00
|
|
|
cpu_set_flags(ctx, -1, 1, 1, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_scf(cpu_context *ctx) {
|
|
|
|
cpu_set_flags(ctx, -1, 0, 0, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_ccf(cpu_context *ctx) {
|
|
|
|
cpu_set_flags(ctx, -1, 0, 0, CPU_FLAG_C ^ 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_halt(cpu_context *ctx) {
|
|
|
|
ctx->halted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_stop(cpu_context *ctx) {
|
|
|
|
printf("CPU STOP!\n");
|
|
|
|
NO_IMPL
|
|
|
|
}
|
|
|
|
|
2025-01-31 12:24:55 -07:00
|
|
|
static void proc_rlca(cpu_context *ctx) {
|
|
|
|
u8 u = ctx->regs.a;
|
|
|
|
bool c = (u >> 7) & 1;
|
|
|
|
u = (u << 1) | c;
|
|
|
|
ctx->regs.a = u;
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, 0, 0, 0, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_rrca(cpu_context *ctx) {
|
|
|
|
u8 b = ctx->regs.a & 1;
|
|
|
|
ctx->regs.a >>= 1;
|
|
|
|
ctx->regs.a |= (b << 7);
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, 0, 0, 0, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_rla(cpu_context *ctx) {
|
2025-01-31 14:39:38 -07:00
|
|
|
u8 u = ctx->regs.a;
|
|
|
|
u8 cf = CPU_FLAG_C;
|
|
|
|
u8 c = (u >> 7) & 1;
|
|
|
|
|
|
|
|
ctx->regs.a = (u << 1) | cf;
|
|
|
|
cpu_set_flags(ctx, 0, 0, 0, c);
|
2025-01-31 12:24:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_rra(cpu_context *ctx) {
|
2025-01-31 14:39:38 -07:00
|
|
|
u8 carry = CPU_FLAG_C;
|
|
|
|
u8 new_c = ctx->regs.a & 1;
|
|
|
|
|
|
|
|
ctx->regs.a >>= 1;
|
|
|
|
ctx->regs.a |= (carry << 7);
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, 0, 0, 0, new_c);
|
2025-01-31 12:24:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_cb(cpu_context *ctx) {
|
|
|
|
u8 op = ctx->fetched_data;
|
|
|
|
reg_type reg = decode_reg(op & 0b111);
|
2025-02-01 00:48:49 -07:00
|
|
|
u8 bit = (op >> 3) & 0b111;
|
2025-01-31 12:24:55 -07:00
|
|
|
u8 bit_op = (op >> 6) & 0b11;
|
|
|
|
u8 reg_val = cpu_read_reg8(reg);
|
|
|
|
emu_cycles(1);
|
|
|
|
|
|
|
|
if(reg == RT_HL) {
|
|
|
|
emu_cycles(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(bit_op) {
|
|
|
|
case 1:
|
|
|
|
//BIT
|
|
|
|
cpu_set_flags(ctx, !(reg_val & (1 << bit)), 0, 1, -1);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
//RST
|
|
|
|
reg_val &= ~(1 << bit);
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
//set
|
|
|
|
reg_val |= (1 << bit);
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
return;
|
2025-02-01 00:48:49 -07:00
|
|
|
default: break;
|
2025-01-31 12:24:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool flagC = CPU_FLAG_C;
|
|
|
|
|
|
|
|
switch(bit) {
|
|
|
|
case 0: {
|
|
|
|
//RLC
|
|
|
|
bool setC = false;
|
|
|
|
u8 result = (reg_val << 1) & 0xFF;
|
|
|
|
|
2025-02-01 00:48:49 -07:00
|
|
|
if((reg_val & (1 << 7)) != 0) {
|
2025-01-31 12:24:55 -07:00
|
|
|
result |= 1;
|
|
|
|
setC = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
cpu_set_reg8(reg, result);
|
|
|
|
cpu_set_flags(ctx, result == 0, 0, 0, setC);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 1: {
|
|
|
|
//RRC
|
|
|
|
u8 old = reg_val;
|
|
|
|
reg_val >>= 1;
|
|
|
|
reg_val |= (old << 7);
|
|
|
|
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
cpu_set_flags(ctx, !reg_val, 0, 0, old & 1);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 2: {
|
|
|
|
//RL
|
|
|
|
u8 old = reg_val;
|
|
|
|
reg_val <<= 1;
|
|
|
|
reg_val |= flagC;
|
|
|
|
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
cpu_set_flags(ctx, !reg_val, 0, 0, !!(old & 0x80));
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 3: {
|
|
|
|
//RR
|
|
|
|
u8 old = reg_val;
|
|
|
|
reg_val >>= 1;
|
|
|
|
|
|
|
|
reg_val |= (flagC << 7);
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
cpu_set_flags(ctx, !reg_val, 0, 0, old & 1);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 4: {
|
|
|
|
//SLA
|
|
|
|
u8 old = reg_val;
|
|
|
|
reg_val <<= 1;
|
|
|
|
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
cpu_set_flags(ctx, !reg_val, 0, 0, old & 0x80);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 5: {
|
|
|
|
//SRA
|
|
|
|
u8 u = (int8_t)reg_val >> 1;
|
|
|
|
cpu_set_reg8(reg, u);
|
|
|
|
cpu_set_flags(ctx, !u, 0, 0, reg_val & 1);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 6: {
|
|
|
|
//SWAP
|
|
|
|
reg_val = ((reg_val & 0xF0) >> 4) | ((reg_val & 0xF) << 4);
|
|
|
|
cpu_set_reg8(reg, reg_val);
|
|
|
|
cpu_set_flags(ctx, reg_val == 0, 0, 0, 0);
|
|
|
|
} return;
|
|
|
|
|
|
|
|
case 7: {
|
|
|
|
//SRL
|
|
|
|
u8 u = reg_val >> 1;
|
|
|
|
cpu_set_reg8(reg, u);
|
|
|
|
cpu_set_flags(ctx, !u, 0, 0, reg_val & 1);
|
|
|
|
} return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "ERROR: INVALID CB: %02X\n", op);
|
|
|
|
NO_IMPL
|
|
|
|
}
|
|
|
|
|
|
|
|
reg_type rt_lookup[] = {
|
|
|
|
RT_B,
|
|
|
|
RT_C,
|
|
|
|
RT_D,
|
|
|
|
RT_E,
|
|
|
|
RT_H,
|
|
|
|
RT_L,
|
|
|
|
RT_HL,
|
|
|
|
RT_A
|
|
|
|
};
|
|
|
|
|
|
|
|
reg_type decode_reg(u8 reg) {
|
|
|
|
if (reg > 0b111) {
|
|
|
|
return RT_NONE;
|
|
|
|
}
|
|
|
|
return rt_lookup[reg];
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_and(cpu_context *ctx) {
|
|
|
|
ctx->regs.a &= ctx->fetched_data;
|
|
|
|
cpu_set_flags(ctx, ctx->regs.a == 0, 0, 1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_xor(cpu_context *ctx) {
|
|
|
|
ctx->regs.a ^= ctx->fetched_data & 0xFF;
|
|
|
|
cpu_set_flags(ctx, ctx->regs.a == 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_or(cpu_context *ctx) {
|
|
|
|
ctx->regs.a |= ctx->fetched_data & 0xFF;
|
|
|
|
cpu_set_flags(ctx, ctx->regs.a == 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_cp(cpu_context *ctx) {
|
|
|
|
int n = (int)ctx->regs.a - (int)ctx->fetched_data;
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, n == 0, 1, ((int)ctx->regs.a & 0x0F) - ((int)ctx->fetched_data & 0x0f) < 0, n < 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_sub(cpu_context *ctx) {
|
|
|
|
u16 val = cpu_read_reg(ctx->cur_inst->reg_1) - ctx->fetched_data;
|
|
|
|
|
|
|
|
int z = val == 0;
|
|
|
|
int h = ((int)cpu_read_reg(ctx->cur_inst->reg_1) & 0xF) - ((int)ctx->fetched_data & 0xF) < 0;
|
|
|
|
int c = ((int)cpu_read_reg(ctx->cur_inst->reg_1)) - ((int)ctx->fetched_data) < 0;
|
|
|
|
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, val);
|
|
|
|
cpu_set_flags(ctx, z, 1, h, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_sbc(cpu_context *ctx) {
|
|
|
|
u8 val = ctx->fetched_data + CPU_FLAG_C;
|
|
|
|
|
|
|
|
int z = cpu_read_reg(ctx->cur_inst->reg_1) - val == 0;
|
|
|
|
int h = ((int)cpu_read_reg(ctx->cur_inst->reg_1) & 0xF)
|
|
|
|
- ((int)ctx->fetched_data & 0xF) - ((int)CPU_FLAG_C) < 0;
|
|
|
|
int c = ((int)cpu_read_reg(ctx->cur_inst->reg_1))
|
|
|
|
- ((int)ctx->fetched_data) - ((int)CPU_FLAG_C) < 0;
|
|
|
|
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, cpu_read_reg(ctx->cur_inst->reg_1) - val);
|
|
|
|
cpu_set_flags(ctx, z, 1, h, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_adc(cpu_context *ctx) {
|
|
|
|
u16 u = ctx->fetched_data;
|
|
|
|
u16 a = ctx->regs.a;
|
|
|
|
u16 c = CPU_FLAG_C;
|
|
|
|
|
|
|
|
ctx->regs.a = (a + u + c) & 0xFF;
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, ctx->regs.a == 0, 0, (a & 0xF) + (u & 0xF) + c > 0xF, a + u + c > 0xFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_add(cpu_context *ctx) {
|
|
|
|
u32 val = cpu_read_reg(ctx->cur_inst->reg_1) + ctx->fetched_data;
|
|
|
|
|
|
|
|
bool is_16bit = is_16_bit(ctx->cur_inst->reg_1);
|
|
|
|
|
|
|
|
if(is_16bit) {
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ctx->cur_inst->reg_1 == RT_SP) {
|
|
|
|
val = cpu_read_reg(ctx->cur_inst->reg_1) + (char)ctx->fetched_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
int z = (val & 0xFF) == 0;
|
|
|
|
int h = (cpu_read_reg(ctx->cur_inst->reg_1) & 0xF) + (ctx->fetched_data & 0xF) >= 0x10;
|
|
|
|
int c = (int)(cpu_read_reg(ctx->cur_inst->reg_1) & 0xFF) + (int)(ctx->fetched_data & 0xFF) >= 0x100;
|
|
|
|
|
|
|
|
if (is_16bit) {
|
|
|
|
z = -1;
|
|
|
|
h = (cpu_read_reg(ctx->cur_inst->reg_1) & 0xFFF) + (ctx->fetched_data & 0xFFF) >= 0x1000;
|
|
|
|
u32 n = ((u32)cpu_read_reg(ctx->cur_inst->reg_1)) + ((u32)ctx->fetched_data);
|
|
|
|
c = n >= 0x10000;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ctx->cur_inst->reg_1 == RT_SP) {
|
|
|
|
z = 0;
|
|
|
|
h = (cpu_read_reg(ctx->cur_inst->reg_1) & 0xF) + (ctx->fetched_data & 0xF) >= 0x10;
|
2025-02-01 00:48:49 -07:00
|
|
|
c = (int)(cpu_read_reg(ctx->cur_inst->reg_1) & 0xFF) + (int)(ctx->fetched_data & 0xFF) >= 0x100;
|
2025-01-31 12:24:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, val & 0xFFFF);
|
|
|
|
cpu_set_flags(ctx, z, 0, h, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_inc(cpu_context *ctx) {
|
|
|
|
u16 val = cpu_read_reg(ctx->cur_inst->reg_1) + 1;
|
|
|
|
|
|
|
|
if(is_16_bit(ctx->cur_inst->reg_1)) {
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->cur_inst->reg_1 == RT_HL && ctx->dest_is_mem) {
|
|
|
|
val = bus_read(cpu_read_reg(RT_HL)) + 1;
|
|
|
|
val &= 0xFF;
|
|
|
|
bus_write(cpu_read_reg(RT_HL), val);
|
|
|
|
} else {
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, val);
|
|
|
|
val = cpu_read_reg(ctx->cur_inst->reg_1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if((ctx->cur_opcode & 0x03) == 0x03) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-02-01 00:48:49 -07:00
|
|
|
cpu_set_flags(ctx, val == 0, 0, (val & 0x0F) == 0, -1);
|
2025-01-31 12:24:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_dec(cpu_context *ctx) {
|
|
|
|
u16 val = cpu_read_reg(ctx->cur_inst->reg_1) - 1;
|
|
|
|
|
|
|
|
if(is_16_bit(ctx->cur_inst->reg_1)) {
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->cur_inst->reg_1 == RT_HL && ctx->dest_is_mem) {
|
|
|
|
val = bus_read(cpu_read_reg(RT_HL)) - 1;
|
|
|
|
val &= 0xFF;
|
|
|
|
bus_write(cpu_read_reg(RT_HL), val);
|
|
|
|
} else {
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, val);
|
|
|
|
val = cpu_read_reg(ctx->cur_inst->reg_1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if((ctx->cur_opcode & 0x0B) == 0x0B) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cpu_set_flags(ctx, val == 0, 1, (val & 0x0F) == 0x0F, -1);
|
|
|
|
}
|
|
|
|
|
2025-01-30 22:54:33 -07:00
|
|
|
static void proc_pop(cpu_context *ctx) {
|
2025-01-31 12:24:55 -07:00
|
|
|
u16 lo = stack_pop();
|
|
|
|
emu_cycles(1);
|
|
|
|
u16 hi = stack_pop();
|
|
|
|
emu_cycles(1);
|
|
|
|
|
|
|
|
u16 n = (hi << 8) | lo;
|
2025-01-30 22:54:33 -07:00
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, n);
|
|
|
|
emu_cycles(2);
|
|
|
|
|
|
|
|
if (ctx->cur_inst->reg_1 == RT_AF) {
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, n & 0xFFF0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_push(cpu_context *ctx) {
|
2025-01-31 12:24:55 -07:00
|
|
|
u16 val = cpu_read_reg(ctx->cur_inst->reg_1);
|
|
|
|
stack_push((val >> 8) & 0xFF);
|
|
|
|
emu_cycles(1);
|
|
|
|
stack_push((val & 0xFF));
|
|
|
|
emu_cycles(1);
|
2025-01-30 22:54:33 -07:00
|
|
|
}
|
|
|
|
|
2025-01-30 21:59:05 -07:00
|
|
|
static void proc_ldh(cpu_context *ctx) {
|
|
|
|
if (!ctx->dest_is_mem) {
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, bus_read(0XFF00 | ctx->fetched_data));
|
|
|
|
} else {
|
2025-01-31 17:07:09 -07:00
|
|
|
bus_write(0xFF00 | ctx->mem_dest, cpu_read_reg(ctx->cur_inst->reg_2));
|
2025-01-30 21:59:05 -07:00
|
|
|
}
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
|
2025-01-30 18:38:29 -07:00
|
|
|
static void proc_di(cpu_context *ctx) {
|
|
|
|
ctx->int_master_enabled = false;
|
|
|
|
}
|
|
|
|
|
2025-01-31 14:39:38 -07:00
|
|
|
static void proc_ei(cpu_context *ctx) {
|
|
|
|
ctx->enabling_ime = true;
|
|
|
|
}
|
|
|
|
|
2025-01-30 17:03:56 -07:00
|
|
|
static void proc_ld(cpu_context *ctx) {
|
2025-01-30 21:59:05 -07:00
|
|
|
if(ctx->dest_is_mem) {
|
2025-01-31 12:24:55 -07:00
|
|
|
if(is_16_bit(ctx->cur_inst->reg_2)) {
|
2025-01-30 21:59:05 -07:00
|
|
|
bus_write16(ctx->mem_dest, ctx->fetched_data);
|
|
|
|
emu_cycles(1);
|
|
|
|
} else {
|
|
|
|
bus_write(ctx->mem_dest, ctx->fetched_data);
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->cur_inst->mode == AM_HL_SPR) {
|
|
|
|
u8 hflag = (cpu_read_reg(ctx->cur_inst->reg_2) & 0xF) + (ctx->fetched_data & 0xF) >= 0x10;
|
|
|
|
u8 cflag = (cpu_read_reg(ctx->cur_inst->reg_2) & 0xFF) + (ctx->fetched_data & 0xFF) >= 0x100;
|
|
|
|
cpu_set_flags(ctx, 0, 0, hflag, cflag);
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, cpu_read_reg(ctx->cur_inst->reg_2) + (char)ctx->fetched_data);
|
2025-02-01 00:48:49 -07:00
|
|
|
return;
|
2025-01-30 21:59:05 -07:00
|
|
|
}
|
|
|
|
cpu_set_reg(ctx->cur_inst->reg_1, ctx->fetched_data);
|
2025-01-30 17:03:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_condition(cpu_context *ctx) {
|
|
|
|
bool z = CPU_FLAG_Z;
|
|
|
|
bool c = CPU_FLAG_C;
|
|
|
|
|
|
|
|
switch(ctx->cur_inst->cond) {
|
|
|
|
case CT_NONE: return true;
|
|
|
|
case CT_C: return c;
|
|
|
|
case CT_NC: return !c;
|
|
|
|
case CT_Z: return z;
|
|
|
|
case CT_NZ: return !z;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2025-01-30 22:54:33 -07:00
|
|
|
static void goto_addr(cpu_context *ctx, u16 addr, bool pushpc){
|
2025-01-30 17:03:56 -07:00
|
|
|
if (check_condition(ctx)) {
|
2025-01-30 22:54:33 -07:00
|
|
|
if(pushpc) {
|
|
|
|
stack_push16(ctx->regs.pc);
|
|
|
|
emu_cycles(2);
|
|
|
|
}
|
|
|
|
ctx->regs.pc = addr;
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_jp(cpu_context *ctx) {
|
|
|
|
goto_addr(ctx, ctx->fetched_data, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_call(cpu_context *ctx) {
|
|
|
|
goto_addr(ctx, ctx->fetched_data, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_rst(cpu_context *ctx) {
|
|
|
|
goto_addr(ctx, ctx->cur_inst->param, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_ret(cpu_context *ctx) {
|
|
|
|
if (ctx->cur_inst->cond != CT_NONE) {
|
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
if(check_condition(ctx)){
|
|
|
|
u16 lo = stack_pop();
|
|
|
|
emu_cycles(1);
|
|
|
|
u16 hi = stack_pop();
|
|
|
|
emu_cycles(1);
|
|
|
|
|
|
|
|
u16 n = (hi << 8) | lo;
|
|
|
|
ctx->regs.pc = n;
|
2025-01-30 17:03:56 -07:00
|
|
|
emu_cycles(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-30 22:54:33 -07:00
|
|
|
static void proc_reti(cpu_context *ctx) {
|
|
|
|
ctx->int_master_enabled = true;
|
|
|
|
proc_ret(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void proc_jr(cpu_context *ctx) {
|
2025-01-31 12:24:55 -07:00
|
|
|
int8_t rel = (int8_t)(ctx->fetched_data & 0XFF);
|
2025-01-30 22:54:33 -07:00
|
|
|
u16 addr = ctx->regs.pc + rel;
|
|
|
|
goto_addr(ctx, addr, false);
|
|
|
|
}
|
|
|
|
|
2025-01-30 17:03:56 -07:00
|
|
|
IN_PROC processors[] = {
|
|
|
|
[IN_NONE] = proc_none,
|
|
|
|
[IN_NOP] = proc_nop,
|
|
|
|
[IN_LD] = proc_ld,
|
|
|
|
[IN_JP] = proc_jp,
|
2025-01-30 18:38:29 -07:00
|
|
|
[IN_DI] = proc_di,
|
2025-01-30 22:54:33 -07:00
|
|
|
[IN_LDH] = proc_ldh,
|
|
|
|
[IN_POP] = proc_pop,
|
|
|
|
[IN_PUSH] = proc_push,
|
|
|
|
[IN_CALL] = proc_call,
|
|
|
|
[IN_JR] = proc_jr,
|
|
|
|
[IN_RET] = proc_ret,
|
|
|
|
[IN_RETI] = proc_reti,
|
2025-01-31 12:24:55 -07:00
|
|
|
[IN_RST] = proc_rst,
|
|
|
|
[IN_INC] = proc_inc,
|
|
|
|
[IN_DEC] = proc_dec,
|
|
|
|
[IN_ADD] = proc_add,
|
|
|
|
[IN_ADC] = proc_adc,
|
|
|
|
[IN_SBC] = proc_sbc,
|
|
|
|
[IN_SUB] = proc_sub,
|
|
|
|
[IN_AND] = proc_and,
|
|
|
|
[IN_XOR] = proc_xor,
|
|
|
|
[IN_OR] = proc_or,
|
|
|
|
[IN_CP] = proc_cp,
|
2025-01-31 14:39:38 -07:00
|
|
|
[IN_RLCA] = proc_rlca,
|
|
|
|
[IN_RRCA] = proc_rrca,
|
|
|
|
[IN_RRA] = proc_rra,
|
|
|
|
[IN_RLA] = proc_rla,
|
|
|
|
[IN_STOP] = proc_stop,
|
|
|
|
[IN_HALT] = proc_halt,
|
|
|
|
[IN_DAA] = proc_daa,
|
|
|
|
[IN_CPL] = proc_cpl,
|
|
|
|
[IN_SCF] = proc_scf,
|
|
|
|
[IN_CCF] = proc_ccf,
|
|
|
|
[IN_CB] = proc_cb,
|
|
|
|
[IN_EI] = proc_ei
|
2025-01-30 17:03:56 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
IN_PROC inst_get_processor(in_type type) {
|
|
|
|
return processors[type];
|
|
|
|
}
|