chore: instantly refill the pipeline on flush
I believe this to be necessary in order to get hardware interrupts working. thumb.gba test 108 fails but I'm committing anyways (despite the regression) because this is kind of rebase/merge hell and I have something that at least sort of works rn
This commit is contained in:
		| @@ -5,7 +5,7 @@ const InstrFn = @import("../../cpu.zig").arm.InstrFn; | ||||
| const rotateRight = @import("../barrel_shifter.zig").rotateRight; | ||||
| const execute = @import("../barrel_shifter.zig").execute; | ||||
|  | ||||
| pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4) InstrFn { | ||||
| pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) InstrFn { | ||||
|     return struct { | ||||
|         fn inner(cpu: *Arm7tdmi, _: *Bus, opcode: u32) void { | ||||
|             const rd = @truncate(u4, opcode >> 12 & 0xF); | ||||
| @@ -13,7 +13,7 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4 | ||||
|             const old_carry = @boolToInt(cpu.cpsr.c.read()); | ||||
|  | ||||
|             // If certain conditions are met, PC is 12 ahead instead of 8 | ||||
|             // TODO: What are these conditions? I can't remember | ||||
|             // TODO: Why these conditions? | ||||
|             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] += 4; | ||||
|             const op1 = cpu.r[rn]; | ||||
|  | ||||
| @@ -23,103 +23,266 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4 | ||||
|             // 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); | ||||
|                 }, | ||||
|             var result: u32 = undefined; | ||||
|             var didOverflow: bool = undefined; | ||||
|  | ||||
|             // Perform Data Processing Logic | ||||
|             switch (kind) { | ||||
|                 0x0 => result = op1 & op2, // AND | ||||
|                 0x1 => result = op1 ^ op2, // EOR | ||||
|                 0x2 => result = op1 -% op2, // SUB | ||||
|                 0x3 => result = op2 -% op1, // RSB | ||||
|                 0x4 => result = newAdd(&didOverflow, op1, op2), // ADD | ||||
|                 0x5 => result = newAdc(&didOverflow, op1, op2, old_carry), // ADC | ||||
|                 0x6 => result = newSbc(op1, op2, old_carry), // SBC | ||||
|                 0x7 => result = newSbc(op2, op1, old_carry), // RSC | ||||
|                 0x8 => { | ||||
|                     // TST | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     const result = op1 & op2; | ||||
|                     setTestOpFlags(S, cpu, opcode, result); | ||||
|                     result = op1 & op2; | ||||
|                 }, | ||||
|                 0x9 => { | ||||
|                     // TEQ | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     const result = op1 ^ op2; | ||||
|                     setTestOpFlags(S, cpu, opcode, result); | ||||
|                     result = op1 ^ op2; | ||||
|                 }, | ||||
|                 0xA => { | ||||
|                     // CMP | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     cmp(cpu, op1, op2); | ||||
|                     result = op1 -% op2; | ||||
|                 }, | ||||
|                 0xB => { | ||||
|                     // CMN | ||||
|                     if (rd == 0xF) | ||||
|                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
|                     cmn(cpu, op1, op2); | ||||
|                     didOverflow = @addWithOverflow(u32, op1, op2, &result); | ||||
|                 }, | ||||
|                 0xC => { | ||||
|                     // ORR | ||||
|                     const result = op1 | op2; | ||||
|                 0xC => result = op1 | op2, // ORR | ||||
|                 0xD => result = op2, // MOV | ||||
|                 0xE => result = op1 & ~op2, // BIC | ||||
|                 0xF => result = ~op2, // MVN | ||||
|             } | ||||
|  | ||||
|             // Write to Destination Register | ||||
|             switch (kind) { | ||||
|                 0x8, 0x9, 0xA, 0xB => {}, // Test Operations | ||||
|                 else => { | ||||
|                     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); | ||||
|                     if (rd == 0xF) cpu.pipe.reload(u32, cpu); | ||||
|                 }, | ||||
|             } | ||||
|  | ||||
|             if (rd == 0xF) cpu.pipe.flush(); | ||||
|             // Write Flags | ||||
|             switch (kind) { | ||||
|                 0x0, 0x1, 0xC, 0xD, 0xE, 0xF => { | ||||
|                     // Logic Operation Flags | ||||
|                     if (S) { | ||||
|                         if (rd == 0xF) { | ||||
|                             cpu.setCpsr(cpu.spsr.raw); | ||||
|                         } else { | ||||
|                             cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                             cpu.cpsr.z.write(result == 0); | ||||
|                             // C set by Barrel Shifter, V is unaffected | ||||
|                         } | ||||
|                     } | ||||
|                 }, | ||||
|                 0x2, 0x3 => { | ||||
|                     // SUB, RSB Flags | ||||
|                     if (S) { | ||||
|                         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                         cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                         if (kind == 0x2) { | ||||
|                             // SUB specific | ||||
|                             cpu.cpsr.c.write(op2 <= op1); | ||||
|                             cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                         } else { | ||||
|                             // RSB Specific | ||||
|                             cpu.cpsr.c.write(op1 <= op2); | ||||
|                             cpu.cpsr.v.write(((op2 ^ result) & (~op1 ^ result)) >> 31 & 1 == 1); | ||||
|                         } | ||||
|  | ||||
|                         if (rd == 0xF) cpu.setCpsr(cpu.spsr.raw); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x4, 0x5 => { | ||||
|                     // ADD, ADC Flags | ||||
|                     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(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|  | ||||
|                         if (rd == 0xF) cpu.setCpsr(cpu.spsr.raw); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x6, 0x7 => { | ||||
|                     // SBC, RSC Flags | ||||
|                     if (S) { | ||||
|                         cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                         cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                         if (kind == 0x6) { | ||||
|                             // SBC specific | ||||
|                             const subtrahend = @as(u64, op2) -% old_carry +% 1; | ||||
|                             cpu.cpsr.c.write(subtrahend <= op1); | ||||
|                             cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                         } else { | ||||
|                             // RSC Specific | ||||
|                             const subtrahend = @as(u64, op1) -% old_carry +% 1; | ||||
|                             cpu.cpsr.c.write(subtrahend <= op2); | ||||
|                             cpu.cpsr.v.write(((op2 ^ result) & (~op1 ^ result)) >> 31 & 1 == 1); | ||||
|                         } | ||||
|  | ||||
|                         if (rd == 0xF) cpu.setCpsr(cpu.spsr.raw); | ||||
|                     } | ||||
|                 }, | ||||
|                 0x8, 0x9, 0xA, 0xB => { | ||||
|                     // Test Operation Flags | ||||
|                     cpu.cpsr.n.write(result >> 31 & 1 == 1); | ||||
|                     cpu.cpsr.z.write(result == 0); | ||||
|  | ||||
|                     if (kind == 0xA) { | ||||
|                         // CMP specific | ||||
|                         cpu.cpsr.c.write(op2 <= op1); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (~op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else if (kind == 0xB) { | ||||
|                         // CMN specific | ||||
|                         cpu.cpsr.c.write(didOverflow); | ||||
|                         cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1); | ||||
|                     } else { | ||||
|                         // TEST, TEQ specific | ||||
|                         // Barrel Shifter should always calc CPSR C in TST | ||||
|                         if (!S) _ = execute(true, cpu, opcode); | ||||
|                     } | ||||
|                 }, | ||||
|             } | ||||
|         } | ||||
|     }.inner; | ||||
| } | ||||
|  | ||||
| // 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 | ||||
| //             // TODO: What are these conditions? I can't remember | ||||
| //             if (!I and opcode >> 4 & 1 == 1) cpu.r[15] += 4; | ||||
| //             const op1 = cpu.r[rn]; | ||||
|  | ||||
| //             const amount = @truncate(u8, (opcode >> 8 & 0xF) << 1); | ||||
| //             const op2 = if (I) rotateRight(S, &cpu.cpsr, opcode & 0xFF, amount) else 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) | ||||
| //                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
| //                     const result = op1 & op2; | ||||
| //                     setTestOpFlags(S, cpu, opcode, result); | ||||
| //                 }, | ||||
| //                 0x9 => { | ||||
| //                     // TEQ | ||||
| //                     if (rd == 0xF) | ||||
| //                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
| //                     const result = op1 ^ op2; | ||||
| //                     setTestOpFlags(S, cpu, opcode, result); | ||||
| //                 }, | ||||
| //                 0xA => { | ||||
| //                     // CMP | ||||
| //                     if (rd == 0xF) | ||||
| //                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
| //                     cmp(cpu, op1, op2); | ||||
| //                 }, | ||||
| //                 0xB => { | ||||
| //                     // CMN | ||||
| //                     if (rd == 0xF) | ||||
| //                         return undefinedTestBehaviour(cpu); | ||||
|  | ||||
| //                     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); | ||||
| //                 }, | ||||
| //             } | ||||
|  | ||||
| //             if (rd == 0xF) cpu.pipe.reload(u32, cpu); | ||||
| //         } | ||||
| //     }.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) { | ||||
| @@ -132,6 +295,14 @@ fn armSbc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_c | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn newSbc(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 ret = @truncate(u32, left -% subtrahend); | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| 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; | ||||
| @@ -184,6 +355,12 @@ fn armAdd(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32) u32 { | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn newAdd(didOverflow: *bool, left: u32, right: u32) u32 { | ||||
|     var ret: u32 = undefined; | ||||
|     didOverflow.* = @addWithOverflow(u32, left, right, &ret); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| pub fn add(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 { | ||||
|     var result: u32 = undefined; | ||||
|     const didOverflow = @addWithOverflow(u32, left, right, &result); | ||||
| @@ -210,6 +387,15 @@ fn armAdc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_c | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| fn newAdc(didOverflow: *bool, left: u32, right: u32, old_carry: u1) u32 { | ||||
|     var ret: u32 = undefined; | ||||
|     const did = @addWithOverflow(u32, left, right, &ret); | ||||
|     const overflow = @addWithOverflow(u32, ret, old_carry, &ret); | ||||
|  | ||||
|     didOverflow.* = did or overflow; | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| 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); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user