fix: squash bugs in cpu intrucion implementation
This commit is contained in:
parent
8a1540c9e9
commit
d7d9fd857f
27
src/cpu.rs
27
src/cpu.rs
|
@ -275,6 +275,33 @@ impl Cpu {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Cpu {
|
||||||
|
fn log_state(&self) -> std::io::Result<()> {
|
||||||
|
use std::io::Write;
|
||||||
|
|
||||||
|
let out = std::io::stdout();
|
||||||
|
let mut handle = out.lock();
|
||||||
|
|
||||||
|
write!(handle, "A: {:02X} ", self.reg.a)?;
|
||||||
|
write!(handle, "F: {:02X} ", u8::from(self.flags))?;
|
||||||
|
write!(handle, "B: {:02X} ", self.reg.b)?;
|
||||||
|
write!(handle, "C: {:02X} ", self.reg.c)?;
|
||||||
|
write!(handle, "D: {:02X} ", self.reg.d)?;
|
||||||
|
write!(handle, "E: {:02X} ", self.reg.e)?;
|
||||||
|
write!(handle, "H: {:02X} ", self.reg.h)?;
|
||||||
|
write!(handle, "L: {:02X} ", self.reg.l)?;
|
||||||
|
write!(handle, "SP: {:04X} ", self.reg.sp)?;
|
||||||
|
write!(handle, "PC: 00:{:04X} ", self.reg.pc)?;
|
||||||
|
write!(handle, "({:02X} ", self.read_byte(self.reg.pc))?;
|
||||||
|
write!(handle, "{:02X} ", self.read_byte(self.reg.pc + 1))?;
|
||||||
|
write!(handle, "{:02X} ", self.read_byte(self.reg.pc + 2))?;
|
||||||
|
write!(handle, "{:02X})\n", self.read_byte(self.reg.pc + 3))?;
|
||||||
|
handle.flush()?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
#[derive(Debug, Copy, Clone)]
|
||||||
pub enum Register {
|
pub enum Register {
|
||||||
A,
|
A,
|
||||||
|
|
|
@ -363,28 +363,23 @@ impl Instruction {
|
||||||
}
|
}
|
||||||
(MATHTarget::Register(InstrRegister::A), MATHTarget::Register(reg)) => {
|
(MATHTarget::Register(InstrRegister::A), MATHTarget::Register(reg)) => {
|
||||||
// ADD A, r[z] | Add (A + r[z]) to register A
|
// ADD A, r[z] | Add (A + r[z]) to register A
|
||||||
|
use InstrRegister::*;
|
||||||
|
|
||||||
let mut flags: Flags = *cpu.flags();
|
let mut flags: Flags = *cpu.flags();
|
||||||
let a_value = cpu.register(Register::A);
|
let a_value = cpu.register(Register::A);
|
||||||
let sum;
|
|
||||||
let cycles: Cycles;
|
let (cycles, sum) = match reg {
|
||||||
match reg {
|
B | C | D | E | H | L | A => {
|
||||||
InstrRegister::B
|
|
||||||
| InstrRegister::C
|
|
||||||
| InstrRegister::D
|
|
||||||
| InstrRegister::E
|
|
||||||
| InstrRegister::H
|
|
||||||
| InstrRegister::L
|
|
||||||
| InstrRegister::A => {
|
|
||||||
let value = cpu.register(Register::try_from(reg).unwrap());
|
let value = cpu.register(Register::try_from(reg).unwrap());
|
||||||
sum = Self::add_u8s(a_value, value, &mut flags);
|
let sum = Self::add_u8s(a_value, value, &mut flags);
|
||||||
cycles = Cycles::new(8);
|
(Cycles::new(4), sum)
|
||||||
}
|
}
|
||||||
InstrRegister::IndirectHL => {
|
IndirectHL => {
|
||||||
let value = cpu.read_byte(cpu.register_pair(RegisterPair::HL));
|
let value = cpu.read_byte(cpu.register_pair(RegisterPair::HL));
|
||||||
sum = Self::add_u8s(a_value, value, &mut flags);
|
let sum = Self::add_u8s(a_value, value, &mut flags);
|
||||||
cycles = Cycles::new(4);
|
(Cycles::new(8), sum)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
cpu.set_register(Register::A, sum);
|
cpu.set_register(Register::A, sum);
|
||||||
cpu.set_flags(flags);
|
cpu.set_flags(flags);
|
||||||
|
@ -851,9 +846,8 @@ impl Instruction {
|
||||||
// CP r[z] | Same behaviour as SUB, except the result is not stored.
|
// CP r[z] | Same behaviour as SUB, except the result is not stored.
|
||||||
let mut flags: Flags = *cpu.flags();
|
let mut flags: Flags = *cpu.flags();
|
||||||
let a_value = cpu.register(Register::A);
|
let a_value = cpu.register(Register::A);
|
||||||
let cycles: Cycles;
|
|
||||||
|
|
||||||
match reg {
|
let cycles = match reg {
|
||||||
InstrRegister::B
|
InstrRegister::B
|
||||||
| InstrRegister::C
|
| InstrRegister::C
|
||||||
| InstrRegister::D
|
| InstrRegister::D
|
||||||
|
@ -863,14 +857,14 @@ impl Instruction {
|
||||||
| InstrRegister::A => {
|
| InstrRegister::A => {
|
||||||
let value = cpu.register(Register::try_from(reg).unwrap());
|
let value = cpu.register(Register::try_from(reg).unwrap());
|
||||||
let _ = Self::sub_u8s(a_value, value, &mut flags);
|
let _ = Self::sub_u8s(a_value, value, &mut flags);
|
||||||
cycles = Cycles::new(4);
|
Cycles::new(4)
|
||||||
}
|
}
|
||||||
InstrRegister::IndirectHL => {
|
InstrRegister::IndirectHL => {
|
||||||
let value = cpu.read_byte(cpu.register_pair(RegisterPair::HL));
|
let value = cpu.read_byte(cpu.register_pair(RegisterPair::HL));
|
||||||
let _ = Self::sub_u8s(a_value, value, &mut flags);
|
let _ = Self::sub_u8s(a_value, value, &mut flags);
|
||||||
cycles = Cycles::new(8);
|
Cycles::new(8)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
cpu.set_flags(flags);
|
cpu.set_flags(flags);
|
||||||
cycles
|
cycles
|
||||||
|
@ -1543,7 +1537,10 @@ impl Instruction {
|
||||||
fn sub_u8s_no_carry(left: u8, right: u8, flags: &mut Flags) -> u8 {
|
fn sub_u8s_no_carry(left: u8, right: u8, flags: &mut Flags) -> u8 {
|
||||||
let diff = left.wrapping_sub(right);
|
let diff = left.wrapping_sub(right);
|
||||||
|
|
||||||
flags.update(diff == 0, true, Self::u8_half_carry(left, right), flags.c());
|
flags.set_z(diff == 0);
|
||||||
|
flags.set_n(true);
|
||||||
|
flags.set_h(Self::sub_u8_half_carry(left, right));
|
||||||
|
|
||||||
diff
|
diff
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1553,7 +1550,7 @@ impl Instruction {
|
||||||
flags.update(
|
flags.update(
|
||||||
diff == 0,
|
diff == 0,
|
||||||
true,
|
true,
|
||||||
Self::u8_half_carry(left, right),
|
Self::sub_u8_half_carry(left, right),
|
||||||
did_overflow,
|
did_overflow,
|
||||||
);
|
);
|
||||||
diff
|
diff
|
||||||
|
@ -1569,7 +1566,7 @@ impl Instruction {
|
||||||
flags.update(
|
flags.update(
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
Self::u16_half_carry(left, right as u16),
|
Self::add_u16_half_carry(left, right as u16),
|
||||||
did_overflow,
|
did_overflow,
|
||||||
);
|
);
|
||||||
sum
|
sum
|
||||||
|
@ -1578,7 +1575,9 @@ impl Instruction {
|
||||||
fn add_u8s_no_carry(left: u8, right: u8, flags: &mut Flags) -> u8 {
|
fn add_u8s_no_carry(left: u8, right: u8, flags: &mut Flags) -> u8 {
|
||||||
let sum = left.wrapping_add(right);
|
let sum = left.wrapping_add(right);
|
||||||
|
|
||||||
flags.update(sum == 0, false, Self::u8_half_carry(left, right), flags.c());
|
flags.set_z(sum == 0);
|
||||||
|
flags.set_n(false);
|
||||||
|
flags.set_h(Self::add_u8_half_carry(left, right));
|
||||||
sum
|
sum
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1588,7 +1587,7 @@ impl Instruction {
|
||||||
flags.update(
|
flags.update(
|
||||||
sum == 0,
|
sum == 0,
|
||||||
false,
|
false,
|
||||||
Self::u8_half_carry(left, right),
|
Self::add_u8_half_carry(left, right),
|
||||||
did_overflow,
|
did_overflow,
|
||||||
);
|
);
|
||||||
sum
|
sum
|
||||||
|
@ -1599,20 +1598,24 @@ impl Instruction {
|
||||||
|
|
||||||
flags.update(
|
flags.update(
|
||||||
false,
|
false,
|
||||||
Self::u16_half_carry(left, right),
|
Self::add_u16_half_carry(left, right),
|
||||||
flags.h(),
|
flags.h(),
|
||||||
did_overflow,
|
did_overflow,
|
||||||
);
|
);
|
||||||
sum
|
sum
|
||||||
}
|
}
|
||||||
|
|
||||||
fn u16_half_carry(left: u16, right: u16) -> bool {
|
fn add_u16_half_carry(left: u16, right: u16) -> bool {
|
||||||
// Self::u8_half_carry((left >> 8) as u8, (right >> 8) as u8)
|
Self::add_u8_half_carry((left >> 8) as u8, (right >> 8) as u8)
|
||||||
left + right > 0xFFF // Thanks @Nectar Boy#1003
|
// left + right > 0xFFF // Thanks @Nectar Boy#1003
|
||||||
}
|
}
|
||||||
|
|
||||||
fn u8_half_carry(left: u8, right: u8) -> bool {
|
fn add_u8_half_carry(left: u8, right: u8) -> bool {
|
||||||
((left & 0xF) + (right & 0xF)) & 0x10 == 0x10
|
(((left & 0xF) + (right & 0xF)) & 0x10) == 0x10
|
||||||
|
}
|
||||||
|
|
||||||
|
fn sub_u8_half_carry(left: u8, right: u8) -> bool {
|
||||||
|
(left & 0xF) < (right & 0xF)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn rl_thru_carry(byte: u8, carry: bool) -> (u8, bool) {
|
fn rl_thru_carry(byte: u8, carry: bool) -> (u8, bool) {
|
||||||
|
|
Loading…
Reference in New Issue