const std = @import("std"); const shifter = @import("../barrel_shifter.zig"); const Bus = @import("../../Bus.zig"); const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi; const InstrFn = @import("../../cpu.zig").ArmInstrFn; pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4) InstrFn { return struct { fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { const rd = @truncate(u4, opcode >> 12 & 0xF); const rn = opcode >> 16 & 0xF; const old_carry = @boolToInt(cpu.cpsr.c.read()); // If certain conditions are met, PC is 12 ahead instead of 8 if (!I and opcode >> 4 & 1 == 1) cpu.r[15] += 4; const op1 = if (rn == 0xF) cpu.fakePC() else cpu.r[rn]; var op2: u32 = undefined; if (I) { const amount = @truncate(u8, (opcode >> 8 & 0xF) << 1); op2 = shifter.rotateRight(S, &cpu.cpsr, opcode & 0xFF, amount); } else { op2 = shifter.execute(S, cpu, opcode); } // Undo special condition from above if (!I and opcode >> 4 & 1 == 1) cpu.r[15] -= 4; switch (instrKind) { 0x0 => { // AND const result = op1 & op2; cpu.r[rd] = result; setArmLogicOpFlags(S, cpu, rd, result); }, 0x1 => { // EOR const result = op1 ^ op2; cpu.r[rd] = result; setArmLogicOpFlags(S, cpu, rd, result); }, 0x2 => { // SUB cpu.r[rd] = armSub(S, cpu, rd, op1, op2); }, 0x3 => { // RSB cpu.r[rd] = armSub(S, cpu, rd, op2, op1); }, 0x4 => { // ADD cpu.r[rd] = armAdd(S, cpu, rd, op1, op2); }, 0x5 => { // ADC cpu.r[rd] = armAdc(S, cpu, rd, op1, op2, old_carry); }, 0x6 => { // SBC cpu.r[rd] = armSbc(S, cpu, rd, op1, op2, old_carry); }, 0x7 => { // RSC cpu.r[rd] = armSbc(S, cpu, rd, op2, op1, old_carry); }, 0x8 => { // TST if (rd == 0xF) { undefinedTestBehaviour(cpu); return; } const result = op1 & op2; setTestOpFlags(S, cpu, opcode, result); }, 0x9 => { // TEQ if (rd == 0xF) { undefinedTestBehaviour(cpu); return; } const result = op1 ^ op2; setTestOpFlags(S, cpu, opcode, result); }, 0xA => { // CMP if (rd == 0xF) { undefinedTestBehaviour(cpu); return; } cmp(cpu, op1, op2); }, 0xB => { // CMN if (rd == 0xF) { undefinedTestBehaviour(cpu); return; } cmn(cpu, op1, op2); }, 0xC => { // ORR const result = op1 | op2; cpu.r[rd] = result; setArmLogicOpFlags(S, cpu, rd, result); }, 0xD => { // MOV cpu.r[rd] = op2; setArmLogicOpFlags(S, cpu, rd, op2); }, 0xE => { // BIC const result = op1 & ~op2; cpu.r[rd] = result; setArmLogicOpFlags(S, cpu, rd, result); }, 0xF => { // MVN const result = ~op2; cpu.r[rd] = result; setArmLogicOpFlags(S, cpu, rd, result); }, } } }.inner; } fn armSbc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_carry: u1) u32 { var result: u32 = undefined; if (S and rd == 0xF) { result = sbc(false, cpu, left, right, old_carry); cpu.setCpsr(cpu.spsr.raw); } else { result = sbc(S, cpu, left, right, old_carry); } return result; } pub fn sbc(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32, old_carry: u1) u32 { // TODO: Make your own version (thanks peach.bot) const subtrahend = @as(u64, right) -% old_carry +% 1; const result = @truncate(u32, left -% subtrahend); if (S) { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(subtrahend <= left); cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); } return result; } fn armSub(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32) u32 { var result: u32 = undefined; if (S and rd == 0xF) { result = sub(false, cpu, left, right); cpu.setCpsr(cpu.spsr.raw); } else { result = sub(S, cpu, left, right); } return result; } pub fn sub(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 { const result = left -% right; if (S) { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(right <= left); cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); } return result; } fn armAdd(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32) u32 { var result: u32 = undefined; if (S and rd == 0xF) { result = add(false, cpu, left, right); cpu.setCpsr(cpu.spsr.raw); } else { result = add(S, cpu, left, right); } return result; } pub fn add(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 { var result: u32 = undefined; const didOverflow = @addWithOverflow(u32, left, right, &result); if (S) { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(didOverflow); cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); } return result; } fn armAdc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_carry: u1) u32 { var result: u32 = undefined; if (S and rd == 0xF) { result = adc(false, cpu, left, right, old_carry); cpu.setCpsr(cpu.spsr.raw); } else { result = adc(S, cpu, left, right, old_carry); } return result; } pub fn adc(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32, old_carry: u1) u32 { var result: u32 = undefined; const did = @addWithOverflow(u32, left, right, &result); const overflow = @addWithOverflow(u32, result, old_carry, &result); if (S) { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(did or overflow); cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); } return result; } pub fn cmp(cpu: *Arm7tdmi, left: u32, right: u32) void { const result = left -% right; cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(right <= left); cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1); } pub fn cmn(cpu: *Arm7tdmi, left: u32, right: u32) void { var result: u32 = undefined; const didOverflow = @addWithOverflow(u32, left, right, &result); cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); cpu.cpsr.c.write(didOverflow); cpu.cpsr.v.write(((left ^ result) & (right ^ result)) >> 31 & 1 == 1); } fn setArmLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, rd: u4, result: u32) void { if (S and rd == 0xF) { cpu.setCpsr(cpu.spsr.raw); } else { setLogicOpFlags(S, cpu, result); } } pub fn setLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, result: u32) void { if (S) { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); // C set by Barrel Shifter, V is unaffected } } fn setTestOpFlags(comptime S: bool, cpu: *Arm7tdmi, opcode: u32, result: u32) void { cpu.cpsr.n.write(result >> 31 & 1 == 1); cpu.cpsr.z.write(result == 0); // Barrel Shifter should always calc CPSR C in TST if (!S) _ = shifter.execute(true, cpu, opcode); } fn undefinedTestBehaviour(cpu: *Arm7tdmi) void { @setCold(true); cpu.setCpsr(cpu.spsr.raw); cpu.r[15] += 4; // FIXME: This is objectively wrong I think }