fix: don't flush pipeline when reloading CPSR in ARM Data Processing
This commit is contained in:
parent
a948c6f900
commit
a3996cbc58
|
@ -34,8 +34,8 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) Ins
|
||||||
0x3 => result = op2 -% op1, // RSB
|
0x3 => result = op2 -% op1, // RSB
|
||||||
0x4 => result = newAdd(&didOverflow, op1, op2), // ADD
|
0x4 => result = newAdd(&didOverflow, op1, op2), // ADD
|
||||||
0x5 => result = newAdc(&didOverflow, op1, op2, old_carry), // ADC
|
0x5 => result = newAdc(&didOverflow, op1, op2, old_carry), // ADC
|
||||||
0x6 => result = newSbc(op1, op2, old_carry), // SBC
|
0x6 => result = sbc(op1, op2, old_carry), // SBC
|
||||||
0x7 => result = newSbc(op2, op1, old_carry), // RSC
|
0x7 => result = sbc(op2, op1, old_carry), // RSC
|
||||||
0x8 => {
|
0x8 => {
|
||||||
// TST
|
// TST
|
||||||
if (rd == 0xF)
|
if (rd == 0xF)
|
||||||
|
@ -76,7 +76,7 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) Ins
|
||||||
else => {
|
else => {
|
||||||
cpu.r[rd] = result;
|
cpu.r[rd] = result;
|
||||||
if (rd == 0xF) {
|
if (rd == 0xF) {
|
||||||
if (S) cpu.setCpsr(cpu.spsr.raw);
|
if (S) cpu.setCpsrNoFlush(cpu.spsr.raw);
|
||||||
|
|
||||||
cpu.pipe.reload(u32, cpu);
|
cpu.pipe.reload(u32, cpu);
|
||||||
}
|
}
|
||||||
|
@ -145,7 +145,7 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) Ins
|
||||||
cpu.cpsr.c.write(didOverflow);
|
cpu.cpsr.c.write(didOverflow);
|
||||||
cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1);
|
cpu.cpsr.v.write(((op1 ^ result) & (op2 ^ result)) >> 31 & 1 == 1);
|
||||||
} else {
|
} else {
|
||||||
// TEST, TEQ specific
|
// TST, TEQ specific
|
||||||
// Barrel Shifter should always calc CPSR C in TST
|
// Barrel Shifter should always calc CPSR C in TST
|
||||||
if (!S) _ = execute(true, cpu, opcode);
|
if (!S) _ = execute(true, cpu, opcode);
|
||||||
}
|
}
|
||||||
|
@ -155,134 +155,7 @@ pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime kind: u4) Ins
|
||||||
}.inner;
|
}.inner;
|
||||||
}
|
}
|
||||||
|
|
||||||
// pub fn dataProcessing(comptime I: bool, comptime S: bool, comptime instrKind: u4) InstrFn {
|
pub fn sbc(left: u32, right: u32, old_carry: u1) u32 {
|
||||||
// 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) {
|
|
||||||
result = sbc(false, cpu, left, right, old_carry);
|
|
||||||
cpu.setCpsr(cpu.spsr.raw);
|
|
||||||
} else {
|
|
||||||
result = sbc(S, cpu, left, right, old_carry);
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn newSbc(left: u32, right: u32, old_carry: u1) u32 {
|
|
||||||
// TODO: Make your own version (thanks peach.bot)
|
// TODO: Make your own version (thanks peach.bot)
|
||||||
const subtrahend = @as(u64, right) -% old_carry +% 1;
|
const subtrahend = @as(u64, right) -% old_carry +% 1;
|
||||||
const ret = @truncate(u32, left -% subtrahend);
|
const ret = @truncate(u32, left -% subtrahend);
|
||||||
|
@ -290,33 +163,6 @@ fn newSbc(left: u32, right: u32, old_carry: u1) u32 {
|
||||||
return ret;
|
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;
|
|
||||||
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 {
|
pub fn sub(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 {
|
||||||
const result = left -% right;
|
const result = left -% right;
|
||||||
|
|
||||||
|
@ -330,18 +176,6 @@ pub fn sub(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 {
|
||||||
return result;
|
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn newAdd(didOverflow: *bool, left: u32, right: u32) u32 {
|
fn newAdd(didOverflow: *bool, left: u32, right: u32) u32 {
|
||||||
var ret: u32 = undefined;
|
var ret: u32 = undefined;
|
||||||
didOverflow.* = @addWithOverflow(u32, left, right, &ret);
|
didOverflow.* = @addWithOverflow(u32, left, right, &ret);
|
||||||
|
@ -362,19 +196,7 @@ pub fn add(comptime S: bool, cpu: *Arm7tdmi, left: u32, right: u32) u32 {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn armAdc(comptime S: bool, cpu: *Arm7tdmi, rd: u4, left: u32, right: u32, old_carry: u1) u32 {
|
pub fn newAdc(didOverflow: *bool, 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;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn newAdc(didOverflow: *bool, left: u32, right: u32, old_carry: u1) u32 {
|
|
||||||
var ret: u32 = undefined;
|
var ret: u32 = undefined;
|
||||||
const did = @addWithOverflow(u32, left, right, &ret);
|
const did = @addWithOverflow(u32, left, right, &ret);
|
||||||
const overflow = @addWithOverflow(u32, ret, old_carry, &ret);
|
const overflow = @addWithOverflow(u32, ret, old_carry, &ret);
|
||||||
|
@ -383,21 +205,6 @@ fn newAdc(didOverflow: *bool, left: u32, right: u32, old_carry: u1) u32 {
|
||||||
return ret;
|
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);
|
|
||||||
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 {
|
pub fn cmp(cpu: *Arm7tdmi, left: u32, right: u32) void {
|
||||||
const result = left -% right;
|
const result = left -% right;
|
||||||
|
|
||||||
|
@ -407,24 +214,6 @@ pub fn cmp(cpu: *Arm7tdmi, left: u32, right: u32) void {
|
||||||
cpu.cpsr.v.write(((left ^ result) & (~right ^ result)) >> 31 & 1 == 1);
|
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 {
|
pub fn setLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, result: u32) void {
|
||||||
if (S) {
|
if (S) {
|
||||||
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
|
@ -433,13 +222,6 @@ pub fn setLogicOpFlags(comptime S: bool, cpu: *Arm7tdmi, result: u32) void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
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) _ = execute(true, cpu, opcode);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn undefinedTestBehaviour(cpu: *Arm7tdmi) void {
|
fn undefinedTestBehaviour(cpu: *Arm7tdmi) void {
|
||||||
@setCold(true);
|
@setCold(true);
|
||||||
cpu.setCpsrNoFlush(cpu.spsr.raw);
|
cpu.setCpsrNoFlush(cpu.spsr.raw);
|
||||||
|
|
|
@ -2,18 +2,13 @@ const Bus = @import("../../Bus.zig");
|
||||||
const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi;
|
const Arm7tdmi = @import("../../cpu.zig").Arm7tdmi;
|
||||||
const InstrFn = @import("../../cpu.zig").thumb.InstrFn;
|
const InstrFn = @import("../../cpu.zig").thumb.InstrFn;
|
||||||
|
|
||||||
const adc = @import("../arm/data_processing.zig").adc;
|
const adc = @import("../arm/data_processing.zig").newAdc;
|
||||||
const sbc = @import("../arm/data_processing.zig").sbc;
|
const sbc = @import("../arm/data_processing.zig").sbc;
|
||||||
const sub = @import("../arm/data_processing.zig").sub;
|
|
||||||
const cmp = @import("../arm/data_processing.zig").cmp;
|
|
||||||
const cmn = @import("../arm/data_processing.zig").cmn;
|
|
||||||
const setTestOpFlags = @import("../arm/data_processing.zig").setTestOpFlags;
|
|
||||||
const setLogicOpFlags = @import("../arm/data_processing.zig").setLogicOpFlags;
|
|
||||||
|
|
||||||
const logicalLeft = @import("../barrel_shifter.zig").logicalLeft;
|
const lsl = @import("../barrel_shifter.zig").logicalLeft;
|
||||||
const logicalRight = @import("../barrel_shifter.zig").logicalRight;
|
const lsr = @import("../barrel_shifter.zig").logicalRight;
|
||||||
const arithmeticRight = @import("../barrel_shifter.zig").arithmeticRight;
|
const asr = @import("../barrel_shifter.zig").arithmeticRight;
|
||||||
const rotateRight = @import("../barrel_shifter.zig").rotateRight;
|
const ror = @import("../barrel_shifter.zig").rotateRight;
|
||||||
|
|
||||||
pub fn fmt4(comptime op: u4) InstrFn {
|
pub fn fmt4(comptime op: u4) InstrFn {
|
||||||
return struct {
|
return struct {
|
||||||
|
@ -22,96 +17,84 @@ pub fn fmt4(comptime op: u4) InstrFn {
|
||||||
const rd = opcode & 0x7;
|
const rd = opcode & 0x7;
|
||||||
const carry = @boolToInt(cpu.cpsr.c.read());
|
const carry = @boolToInt(cpu.cpsr.c.read());
|
||||||
|
|
||||||
|
var result: u32 = undefined;
|
||||||
|
var didOverflow: bool = undefined;
|
||||||
switch (op) {
|
switch (op) {
|
||||||
0x0 => {
|
0x0 => result = cpu.r[rd] & cpu.r[rs], // AND
|
||||||
// AND
|
0x1 => result = cpu.r[rd] ^ cpu.r[rs], // EOR
|
||||||
const result = cpu.r[rd] & cpu.r[rs];
|
0x2 => result = lsl(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])), // LSL
|
||||||
cpu.r[rd] = result;
|
0x3 => result = lsr(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])), // LSR
|
||||||
setLogicOpFlags(true, cpu, result);
|
0x4 => result = asr(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])), // ASR
|
||||||
|
0x5 => result = adc(&didOverflow, cpu.r[rd], cpu.r[rs], carry), // ADC
|
||||||
|
0x6 => result = sbc(cpu.r[rd], cpu.r[rs], carry), // SBC
|
||||||
|
0x7 => result = ror(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs])), // ROR
|
||||||
|
0x8 => result = cpu.r[rd] & cpu.r[rs], // TST
|
||||||
|
0x9 => result = 0 -% cpu.r[rs], // NEG
|
||||||
|
0xA => result = cpu.r[rd] -% cpu.r[rs], // CMP
|
||||||
|
0xB => didOverflow = @addWithOverflow(u32, cpu.r[rd], cpu.r[rs], &result), // CMN
|
||||||
|
0xC => result = cpu.r[rd] | cpu.r[rs], // ORR
|
||||||
|
0xD => result = @truncate(u32, @as(u64, cpu.r[rs]) * @as(u64, cpu.r[rd])),
|
||||||
|
0xE => result = cpu.r[rd] & ~cpu.r[rs],
|
||||||
|
0xF => result = ~cpu.r[rs],
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write to Destination Register
|
||||||
|
switch (op) {
|
||||||
|
0x8, 0xA, 0xB => {},
|
||||||
|
else => cpu.r[rd] = result,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write Flags
|
||||||
|
switch (op) {
|
||||||
|
0x0, 0x1, 0x2, 0x3, 0x4, 0x7, 0xC, 0xE, 0xF => {
|
||||||
|
// Logic Operations
|
||||||
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
|
cpu.cpsr.z.write(result == 0);
|
||||||
|
// C set by Barrel Shifter, V is unaffected
|
||||||
},
|
},
|
||||||
0x1 => {
|
0x8, 0xA => {
|
||||||
// EOR
|
// Test Flags
|
||||||
const result = cpu.r[rd] ^ cpu.r[rs];
|
// CMN (0xB) is handled with ADC
|
||||||
cpu.r[rd] = result;
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
setLogicOpFlags(true, cpu, result);
|
cpu.cpsr.z.write(result == 0);
|
||||||
|
|
||||||
|
if (op == 0xA) {
|
||||||
|
// CMP specific
|
||||||
|
cpu.cpsr.c.write(cpu.r[rs] <= cpu.r[rd]);
|
||||||
|
cpu.cpsr.v.write(((cpu.r[rd] ^ result) & (~cpu.r[rs] ^ result)) >> 31 & 1 == 1);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
0x2 => {
|
0x5, 0xB => {
|
||||||
// LSL
|
// ADC, CMN
|
||||||
const result = logicalLeft(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs]));
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
cpu.r[rd] = result;
|
cpu.cpsr.z.write(result == 0);
|
||||||
setLogicOpFlags(true, cpu, result);
|
cpu.cpsr.c.write(didOverflow);
|
||||||
},
|
cpu.cpsr.v.write(((cpu.r[rd] ^ result) & (cpu.r[rs] ^ result)) >> 31 & 1 == 1);
|
||||||
0x3 => {
|
|
||||||
// LSR
|
// FIXME: Pretty sure CMN Is the same
|
||||||
const result = logicalRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs]));
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
|
||||||
0x4 => {
|
|
||||||
// ASR
|
|
||||||
const result = arithmeticRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs]));
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
|
||||||
0x5 => {
|
|
||||||
// ADC
|
|
||||||
cpu.r[rd] = adc(true, cpu, cpu.r[rd], cpu.r[rs], carry);
|
|
||||||
},
|
},
|
||||||
0x6 => {
|
0x6 => {
|
||||||
// SBC
|
// SBC
|
||||||
cpu.r[rd] = sbc(true, cpu, cpu.r[rd], cpu.r[rs], carry);
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
},
|
cpu.cpsr.z.write(result == 0);
|
||||||
0x7 => {
|
|
||||||
// ROR
|
const subtrahend = @as(u64, cpu.r[rs]) -% carry +% 1;
|
||||||
const result = rotateRight(true, &cpu.cpsr, cpu.r[rd], @truncate(u8, cpu.r[rs]));
|
cpu.cpsr.c.write(subtrahend <= cpu.r[rd]);
|
||||||
cpu.r[rd] = result;
|
cpu.cpsr.v.write(((cpu.r[rd] ^ result) & (~cpu.r[rs] ^ result)) >> 31 & 1 == 1);
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
|
||||||
0x8 => {
|
|
||||||
// TST
|
|
||||||
const result = cpu.r[rd] & cpu.r[rs];
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
},
|
||||||
0x9 => {
|
0x9 => {
|
||||||
// NEG
|
// NEG
|
||||||
cpu.r[rd] = sub(true, cpu, 0, cpu.r[rs]);
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
},
|
cpu.cpsr.z.write(result == 0);
|
||||||
0xA => {
|
cpu.cpsr.c.write(cpu.r[rs] <= 0);
|
||||||
// CMP
|
cpu.cpsr.v.write(((0 ^ result) & (~cpu.r[rs] ^ result)) >> 31 & 1 == 1);
|
||||||
cmp(cpu, cpu.r[rd], cpu.r[rs]);
|
|
||||||
},
|
|
||||||
0xB => {
|
|
||||||
// CMN
|
|
||||||
cmn(cpu, cpu.r[rd], cpu.r[rs]);
|
|
||||||
},
|
|
||||||
0xC => {
|
|
||||||
// ORR
|
|
||||||
const result = cpu.r[rd] | cpu.r[rs];
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
},
|
||||||
0xD => {
|
0xD => {
|
||||||
// MUL
|
// Multiplication
|
||||||
const temp = @as(u64, cpu.r[rs]) * @as(u64, cpu.r[rd]);
|
|
||||||
const result = @truncate(u32, temp);
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
|
|
||||||
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
cpu.cpsr.n.write(result >> 31 & 1 == 1);
|
||||||
cpu.cpsr.z.write(result == 0);
|
cpu.cpsr.z.write(result == 0);
|
||||||
// V is unaffected, assuming similar behaviour to ARMv4 MUL C is undefined
|
// V is unaffected, assuming similar behaviour to ARMv4 MUL C is undefined
|
||||||
},
|
},
|
||||||
0xE => {
|
|
||||||
// BIC
|
|
||||||
const result = cpu.r[rd] & ~cpu.r[rs];
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
|
||||||
0xF => {
|
|
||||||
// MVN
|
|
||||||
const result = ~cpu.r[rs];
|
|
||||||
cpu.r[rd] = result;
|
|
||||||
setLogicOpFlags(true, cpu, result);
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}.inner;
|
}.inner;
|
||||||
|
|
Loading…
Reference in New Issue