2021-11-02 05:06:40 +00:00
|
|
|
|
|
|
|
use crate::system::System;
|
|
|
|
use crate::error::{ErrorType, Error};
|
2021-11-03 03:58:03 +00:00
|
|
|
use crate::devices::{ClockElapsed, Address, Steppable, Addressable, Interruptable, Debuggable, Transmutable, read_beu16, write_beu16};
|
2021-11-02 05:06:40 +00:00
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
use super::decode::{Condition, Instruction, LoadTarget, Target, OptionalSource, RegisterPair, IndexRegister, IndexRegisterHalf, Size};
|
2021-11-03 03:58:03 +00:00
|
|
|
use super::state::{Z80, Status, Flags, Register};
|
2021-11-02 05:06:40 +00:00
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
|
2021-11-07 04:18:45 +00:00
|
|
|
const DEV_NAME: &'static str = "z80-cpu";
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
const FLAGS_ALL: u8 = 0xFF;
|
|
|
|
const FLAGS_ALL_BUT_CARRY: u8 = 0xFE;
|
|
|
|
const FLAGS_NUMERIC: u8 = 0xC0;
|
|
|
|
const FLAGS_ARITHMETIC: u8 = 0x17;
|
|
|
|
const FLAGS_CARRY_HALF_CARRY: u8 = 0x11;
|
|
|
|
|
2021-11-07 04:18:45 +00:00
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
enum RotateType {
|
|
|
|
Bit8,
|
|
|
|
Bit9,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-02 05:06:40 +00:00
|
|
|
impl Steppable for Z80 {
|
|
|
|
fn step(&mut self, system: &System) -> Result<ClockElapsed, Error> {
|
|
|
|
self.step_internal(system)?;
|
|
|
|
Ok((1_000_000_000 / self.frequency as u64) * 4)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn on_error(&mut self, system: &System) {
|
2021-11-07 04:18:45 +00:00
|
|
|
self.dump_state(system);
|
2021-11-02 05:06:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Interruptable for Z80 { }
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
|
2021-11-02 05:06:40 +00:00
|
|
|
impl Transmutable for Z80 {
|
|
|
|
fn as_steppable(&mut self) -> Option<&mut dyn Steppable> {
|
|
|
|
Some(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn as_interruptable(&mut self) -> Option<&mut dyn Interruptable> {
|
|
|
|
Some(self)
|
|
|
|
}
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
fn as_debuggable(&mut self) -> Option<&mut dyn Debuggable> {
|
|
|
|
Some(self)
|
|
|
|
}
|
2021-11-02 05:06:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Z80 {
|
|
|
|
pub fn step_internal(&mut self, system: &System) -> Result<(), Error> {
|
|
|
|
match self.state.status {
|
|
|
|
Status::Init => self.init(system),
|
|
|
|
Status::Halted => Err(Error::new("CPU stopped")),
|
|
|
|
Status::Running => {
|
|
|
|
match self.cycle_one(system) {
|
|
|
|
Ok(()) => Ok(()),
|
|
|
|
//Err(Error { err: ErrorType::Processor, native, .. }) => {
|
|
|
|
Err(Error { err: ErrorType::Processor, native, .. }) => {
|
|
|
|
//self.exception(system, native as u8, false)?;
|
|
|
|
Ok(())
|
|
|
|
},
|
|
|
|
Err(err) => Err(err),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn init(&mut self, system: &System) -> Result<(), Error> {
|
|
|
|
//self.state.msp = self.port.read_beu32(0)?;
|
|
|
|
//self.state.pc = self.port.read_beu32(4)?;
|
|
|
|
self.state.status = Status::Running;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn cycle_one(&mut self, system: &System) -> Result<(), Error> {
|
|
|
|
//self.timer.cycle.start();
|
|
|
|
self.decode_next(system)?;
|
2021-11-03 03:58:03 +00:00
|
|
|
self.execute_current(system)?;
|
2021-11-02 05:06:40 +00:00
|
|
|
//self.timer.cycle.end();
|
|
|
|
|
|
|
|
//if (self.timer.cycle.events % 500) == 0 {
|
|
|
|
// println!("{}", self.timer);
|
|
|
|
//}
|
|
|
|
|
|
|
|
//self.check_pending_interrupts(system)?;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.check_breakpoints(system);
|
2021-11-02 05:06:40 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn decode_next(&mut self, system: &System) -> Result<(), Error> {
|
|
|
|
//self.timer.decode.start();
|
|
|
|
self.decoder.decode_at(&mut self.port, self.state.pc)?;
|
|
|
|
//self.timer.decode.end();
|
|
|
|
|
|
|
|
//if self.debugger.use_tracing {
|
2021-11-07 04:18:45 +00:00
|
|
|
//self.decoder.dump_decoded(&mut self.port);
|
2021-11-06 22:08:03 +00:00
|
|
|
//self.dump_state(system);
|
2021-11-02 05:06:40 +00:00
|
|
|
//}
|
|
|
|
|
|
|
|
self.state.pc = self.decoder.end;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn execute_current(&mut self, system: &System) -> Result<(), Error> {
|
2021-11-03 03:58:03 +00:00
|
|
|
match self.decoder.instruction {
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::ADCa(target) => {
|
|
|
|
let src = self.get_target_value(target)?;
|
|
|
|
let acc = self.get_register_value(Register::A);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result1, carry1, overflow1) = add_bytes(acc, self.get_flag(Flags::Carry) as u8);
|
|
|
|
let (result2, carry2, overflow2) = add_bytes(result1, src);
|
|
|
|
self.set_arithmetic_op_flags(result2 as u16, Size::Byte, false, carry1 | carry2, overflow1 | overflow2, (result2 & 0x10) != 0);
|
|
|
|
|
|
|
|
self.set_register_value(Register::A, result2);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
|
|
|
Instruction::ADC16(dest_pair, src_pair) => {
|
|
|
|
let src = self.get_register_pair_value(src_pair);
|
2021-11-08 00:28:44 +00:00
|
|
|
let dest = self.get_register_pair_value(dest_pair);
|
|
|
|
|
|
|
|
let (result1, carry1, overflow1) = add_words(dest, self.get_flag(Flags::Carry) as u16);
|
|
|
|
let (result2, carry2, overflow2) = add_words(result1, src);
|
|
|
|
self.set_arithmetic_op_flags(result2, Size::Word, false, carry1 | carry2, overflow1 | overflow2, (result2 & 0x10) != 0);
|
|
|
|
|
|
|
|
self.set_register_pair_value(dest_pair, result2);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
2021-11-05 04:30:33 +00:00
|
|
|
Instruction::ADDa(target) => {
|
|
|
|
let src = self.get_target_value(target)?;
|
|
|
|
let acc = self.get_register_value(Register::A);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = add_bytes(acc, src);
|
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, false, carry, overflow, (result & 0x10) != 0);
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::ADD16(dest_pair, src_pair) => {
|
|
|
|
let src = self.get_register_pair_value(src_pair);
|
2021-11-08 00:28:44 +00:00
|
|
|
let dest = self.get_register_pair_value(dest_pair);
|
|
|
|
|
|
|
|
let (result, carry, _) = add_words(dest, src);
|
|
|
|
self.set_flag(Flags::AddSubtract, false);
|
|
|
|
self.set_flag(Flags::Carry, carry);
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_pair_value(dest_pair, result);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
|
|
|
Instruction::AND(target) => {
|
|
|
|
let acc = self.get_register_value(Register::A);
|
|
|
|
let value = self.get_target_value(target)?;
|
|
|
|
let result = acc & value;
|
|
|
|
self.set_register_value(Register::A, result);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, false, true);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::BIT(bit, target) => {
|
|
|
|
let value = self.get_target_value(target)?;
|
2021-11-09 19:03:57 +00:00
|
|
|
let result = value & (1 << bit);
|
2021-11-08 06:44:40 +00:00
|
|
|
self.set_flag(Flags::Zero, result == 0);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_flag(Flags::AddSubtract, false);
|
2021-11-05 04:30:33 +00:00
|
|
|
self.set_flag(Flags::HalfCarry, true);
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::CALL(addr) => {
|
|
|
|
self.push_word(self.decoder.end)?;
|
|
|
|
self.state.pc = addr;
|
2021-11-03 03:58:03 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::CALLcc(cond, addr) => {
|
|
|
|
if self.get_current_condition(cond) {
|
|
|
|
self.push_word(self.decoder.end)?;
|
|
|
|
self.state.pc = addr;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Instruction::CCF => {
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_flag(Flags::AddSubtract, false);
|
2021-11-08 06:44:40 +00:00
|
|
|
self.set_flag(Flags::HalfCarry, self.get_flag(Flags::Carry));
|
|
|
|
self.set_flag(Flags::Carry, !self.get_flag(Flags::Carry));
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
|
|
|
Instruction::CP(target) => {
|
2021-11-05 04:30:33 +00:00
|
|
|
let src = self.get_target_value(target)?;
|
|
|
|
let acc = self.get_register_value(Register::A);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = sub_bytes(acc, src);
|
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, true, carry, overflow, (result & 0x10) != 0);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::CPD => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::CPDR => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::CPI => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::CPIR => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::CPL => {
|
|
|
|
let value = self.get_register_value(Register::A);
|
|
|
|
self.set_register_value(Register::A, !value);
|
|
|
|
self.set_flag(Flags::HalfCarry, true);
|
|
|
|
self.set_flag(Flags::AddSubtract, true);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::DAA => {
|
|
|
|
//},
|
2021-11-05 04:30:33 +00:00
|
|
|
Instruction::DEC16(regpair) => {
|
|
|
|
let value = self.get_register_pair_value(regpair);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = sub_words(value, 1);
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
self.set_register_pair_value(regpair, result);
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::DEC8(target) => {
|
2021-11-05 04:30:33 +00:00
|
|
|
let value = self.get_target_value(target)?;
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = sub_bytes(value, 1);
|
2021-11-09 19:03:57 +00:00
|
|
|
let carry = self.get_flag(Flags::Carry); // Preserve the carry bit, according to Z80 reference
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, true, carry, overflow, (result & 0x10) != 0);
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::DI => {
|
|
|
|
self.state.interrupts_enabled = false;
|
2021-11-03 03:58:03 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::DJNZ(offset) => {
|
|
|
|
let value = self.get_register_value(Register::B);
|
|
|
|
let result = value.wrapping_sub(1);
|
|
|
|
self.set_register_value(Register::B, result);
|
2021-11-03 03:58:03 +00:00
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
if result != 0 {
|
|
|
|
self.state.pc = ((self.state.pc as i16) + (offset as i16)) as u16;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Instruction::EI => {
|
|
|
|
self.state.interrupts_enabled = true;
|
|
|
|
},
|
2021-11-05 04:30:33 +00:00
|
|
|
Instruction::EXX => {
|
|
|
|
for i in 0..6 {
|
|
|
|
let (normal, shadow) = (self.state.reg[i], self.state.shadow_reg[i]);
|
|
|
|
self.state.reg[i] = shadow;
|
|
|
|
self.state.shadow_reg[i] = normal;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Instruction::EXafaf => {
|
|
|
|
for i in 6..8 {
|
|
|
|
let (normal, shadow) = (self.state.reg[i], self.state.shadow_reg[i]);
|
|
|
|
self.state.reg[i] = shadow;
|
|
|
|
self.state.shadow_reg[i] = normal;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Instruction::EXhlde => {
|
|
|
|
let (hl, de) = (self.get_register_pair_value(RegisterPair::HL), self.get_register_pair_value(RegisterPair::DE));
|
|
|
|
self.set_register_pair_value(RegisterPair::DE, hl);
|
|
|
|
self.set_register_pair_value(RegisterPair::HL, de);
|
|
|
|
},
|
2021-11-08 06:44:40 +00:00
|
|
|
Instruction::EXsp(regpair) => {
|
|
|
|
let reg_value = self.get_register_pair_value(regpair);
|
|
|
|
let sp = self.get_register_pair_value(RegisterPair::SP);
|
|
|
|
let sp_value = self.port.read_leu16(sp as Address)?;
|
|
|
|
self.set_register_pair_value(regpair, sp_value);
|
|
|
|
self.port.write_leu16(sp as Address, reg_value)?;
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::HALT => {
|
|
|
|
self.state.status = Status::Halted;
|
2021-11-03 03:58:03 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::IM(u8) => {
|
|
|
|
//},
|
|
|
|
Instruction::INC16(regpair) => {
|
2021-11-03 03:58:03 +00:00
|
|
|
let value = self.get_register_pair_value(regpair);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = add_words(value, 1);
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_pair_value(regpair, result);
|
2021-11-03 03:58:03 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::INC8(target) => {
|
|
|
|
let value = self.get_target_value(target)?;
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = add_bytes(value, 1);
|
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, false, carry, overflow, (result & 0x10) != 0);
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
//Instruction::IND => {
|
|
|
|
//},
|
|
|
|
//Instruction::INDR => {
|
|
|
|
//},
|
|
|
|
//Instruction::INI => {
|
|
|
|
//},
|
|
|
|
//Instruction::INIR => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
|
|
|
//Instruction::INic(reg) => {
|
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
//Instruction::INx(u8) => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
|
|
|
Instruction::JP(addr) => {
|
|
|
|
self.state.pc = addr;
|
|
|
|
},
|
2021-11-08 06:44:40 +00:00
|
|
|
Instruction::JPIndirect(regpair) => {
|
|
|
|
let value = self.get_register_pair_value(regpair);
|
|
|
|
//let addr = self.port.read_leu16(value as Address)?;
|
|
|
|
self.state.pc = value;
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
Instruction::JPcc(cond, addr) => {
|
|
|
|
if self.get_current_condition(cond) {
|
|
|
|
self.state.pc = addr;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Instruction::JR(offset) => {
|
|
|
|
self.state.pc = ((self.state.pc as i16) + (offset as i16)) as u16;
|
|
|
|
},
|
|
|
|
Instruction::JRcc(cond, offset) => {
|
|
|
|
if self.get_current_condition(cond) {
|
|
|
|
self.state.pc = ((self.state.pc as i16) + (offset as i16)) as u16;
|
|
|
|
}
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::LD(dest, src) => {
|
|
|
|
let src_value = self.get_load_target_value(src)?;
|
2021-11-08 00:28:44 +00:00
|
|
|
if let LoadTarget::DirectSpecialRegByte(_) = src {
|
|
|
|
self.set_logic_op_flags(src_value as u8, self.get_flag(Flags::Carry), false);
|
|
|
|
}
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_load_target_value(dest, src_value)?;
|
|
|
|
},
|
|
|
|
//Instruction::LDD => {
|
|
|
|
//},
|
|
|
|
//Instruction::LDDR => {
|
|
|
|
//},
|
|
|
|
//Instruction::LDI => {
|
|
|
|
//},
|
|
|
|
Instruction::LDIR => {
|
|
|
|
let src_value = self.get_load_target_value(LoadTarget::IndirectRegByte(RegisterPair::HL))?;
|
|
|
|
self.set_load_target_value(LoadTarget::IndirectRegByte(RegisterPair::DE), src_value)?;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.add_to_regpair(RegisterPair::DE, 1);
|
|
|
|
self.add_to_regpair(RegisterPair::HL, 1);
|
|
|
|
let count = self.add_to_regpair(RegisterPair::BC, -1);
|
2021-11-06 22:08:03 +00:00
|
|
|
if count != 0 {
|
|
|
|
self.state.pc -= 2;
|
|
|
|
}
|
2021-11-08 00:28:44 +00:00
|
|
|
let mask = (Flags::AddSubtract as u8) | (Flags::HalfCarry as u8) | (Flags::Parity as u8);
|
|
|
|
let parity = if count != 0 { Flags::Parity as u8 } else { 0 };
|
|
|
|
self.set_flags(mask, parity);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
2021-11-09 19:03:57 +00:00
|
|
|
Instruction::NEG => {
|
|
|
|
let acc = self.get_register_value(Register::A);
|
|
|
|
|
|
|
|
let (result, carry, overflow) = sub_bytes(0, acc);
|
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, true, carry, overflow, (result & 0x10) != 0);
|
|
|
|
|
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::NOP => { },
|
|
|
|
Instruction::OR(target) => {
|
|
|
|
let acc = self.get_register_value(Register::A);
|
|
|
|
let value = self.get_target_value(target)?;
|
|
|
|
let result = acc | value;
|
|
|
|
self.set_register_value(Register::A, result);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, false, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
|
|
|
//Instruction::OTDR => {
|
|
|
|
//},
|
|
|
|
//Instruction::OTIR => {
|
|
|
|
//},
|
|
|
|
//Instruction::OUTD => {
|
|
|
|
//},
|
|
|
|
//Instruction::OUTI => {
|
|
|
|
//},
|
|
|
|
//Instruction::OUTic(reg) => {
|
|
|
|
//},
|
|
|
|
Instruction::OUTx(port) => {
|
|
|
|
// TODO this needs to be fixed
|
2021-11-07 04:18:45 +00:00
|
|
|
println!("OUT ({:x}), {:x}", port, self.state.reg[Register::A as usize]);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
|
|
|
Instruction::POP(regpair) => {
|
|
|
|
let value = self.pop_word()?;
|
|
|
|
self.set_register_pair_value(regpair, value);
|
|
|
|
},
|
|
|
|
Instruction::PUSH(regpair) => {
|
|
|
|
let value = self.get_register_pair_value(regpair);
|
|
|
|
self.push_word(value)?;
|
|
|
|
},
|
|
|
|
Instruction::RES(bit, target, opt_src) => {
|
|
|
|
let mut value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
value = value & !(1 << bit);
|
|
|
|
self.set_target_value(target, value);
|
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
Instruction::RET => {
|
|
|
|
self.state.pc = self.pop_word()?;
|
|
|
|
},
|
|
|
|
//Instruction::RETI => {
|
|
|
|
//},
|
|
|
|
//Instruction::RETN => {
|
|
|
|
//},
|
2021-11-05 04:30:33 +00:00
|
|
|
Instruction::RETcc(cond) => {
|
|
|
|
if self.get_current_condition(cond) {
|
|
|
|
self.state.pc = self.pop_word()?;
|
|
|
|
}
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::RL(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_left(value, RotateType::Bit9);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_target_value(target, result)?;
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::RLA => {
|
|
|
|
let value = self.get_register_value(Register::A);
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_left(value, RotateType::Bit9);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::RLC(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_left(value, RotateType::Bit8);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
Instruction::RLCA => {
|
|
|
|
let value = self.get_register_value(Register::A);
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_left(value, RotateType::Bit8);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
|
|
|
//Instruction::RLD => {
|
2021-11-03 03:58:03 +00:00
|
|
|
//},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::RR(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_right(value, RotateType::Bit9);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
Instruction::RRA => {
|
|
|
|
let value = self.get_register_value(Register::A);
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_right(value, RotateType::Bit9);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
|
|
|
Instruction::RRC(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_right(value, RotateType::Bit8);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_target_value(target, result)?;
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::RRCA => {
|
|
|
|
let value = self.get_register_value(Register::A);
|
2021-11-07 04:18:45 +00:00
|
|
|
let (result, out_bit) = self.rotate_right(value, RotateType::Bit8);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
|
|
|
//Instruction::RRD => {
|
|
|
|
//},
|
2021-11-05 04:30:33 +00:00
|
|
|
Instruction::RST(addr) => {
|
|
|
|
self.push_word(self.decoder.end)?;
|
|
|
|
self.state.pc = addr as u16;
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::SBCa(target) => {
|
|
|
|
let src = self.get_target_value(target)?;
|
|
|
|
let acc = self.get_register_value(Register::A);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result1, carry1, overflow1) = sub_bytes(acc, self.get_flag(Flags::Carry) as u8);
|
|
|
|
let (result2, carry2, overflow2) = sub_bytes(result1, src);
|
|
|
|
self.set_arithmetic_op_flags(result2 as u16, Size::Byte, true, carry1 | carry2, overflow1 | overflow2, (result2 & 0x10) != 0);
|
|
|
|
|
|
|
|
self.set_register_value(Register::A, result2);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
|
|
|
Instruction::SBC16(dest_pair, src_pair) => {
|
|
|
|
let src = self.get_register_pair_value(src_pair);
|
2021-11-08 00:28:44 +00:00
|
|
|
let dest = self.get_register_pair_value(dest_pair);
|
|
|
|
|
|
|
|
let (result1, carry1, overflow1) = sub_words(dest, self.get_flag(Flags::Carry) as u16);
|
|
|
|
let (result2, carry2, overflow2) = sub_words(result1, src);
|
|
|
|
self.set_arithmetic_op_flags(result2, Size::Word, true, carry1 | carry2, overflow1 | overflow2, (result2 & 0x10) != 0);
|
|
|
|
|
|
|
|
self.set_register_pair_value(dest_pair, result2);
|
2021-11-05 04:30:33 +00:00
|
|
|
},
|
|
|
|
Instruction::SCF => {
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_flag(Flags::AddSubtract, false);
|
|
|
|
self.set_flag(Flags::HalfCarry, false);
|
2021-11-05 04:30:33 +00:00
|
|
|
self.set_flag(Flags::Carry, true);
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::SET(bit, target, opt_src) => {
|
|
|
|
let mut value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
value = value | (1 << bit);
|
2021-11-09 19:03:57 +00:00
|
|
|
self.set_target_value(target, value)?;
|
2021-11-03 03:58:03 +00:00
|
|
|
},
|
2021-11-07 04:18:45 +00:00
|
|
|
Instruction::SLA(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
let out_bit = get_msb(value as u16, Size::Byte);
|
|
|
|
let result = (value << 1) | 0x01;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-07 04:18:45 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
Instruction::SLL(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
let out_bit = get_msb(value as u16, Size::Byte);
|
|
|
|
let result = value << 1;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-07 04:18:45 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
Instruction::SRA(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
let out_bit = (value & 0x01) != 0;
|
|
|
|
let msb_mask = if get_msb(value as u16, Size::Byte) { 0x80 } else { 0 };
|
|
|
|
let result = (value >> 1) | msb_mask;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-07 04:18:45 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
|
|
|
Instruction::SRL(target, opt_src) => {
|
|
|
|
let value = self.get_opt_src_target_value(opt_src, target)?;
|
|
|
|
let out_bit = (value & 0x01) != 0;
|
|
|
|
let result = value >> 1;
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, out_bit, false);
|
2021-11-07 04:18:45 +00:00
|
|
|
self.set_target_value(target, result)?;
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Instruction::SUB(target) => {
|
|
|
|
let src = self.get_target_value(target)?;
|
|
|
|
let acc = self.get_register_value(Register::A);
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
let (result, carry, overflow) = sub_bytes(acc, src);
|
|
|
|
self.set_arithmetic_op_flags(result as u16, Size::Byte, true, carry, overflow, (result & 0x10) != 0);
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
self.set_register_value(Register::A, result);
|
|
|
|
},
|
|
|
|
Instruction::XOR(target) => {
|
|
|
|
let acc = self.get_register_value(Register::A);
|
|
|
|
let value = self.get_target_value(target)?;
|
|
|
|
let result = acc ^ value;
|
|
|
|
self.set_register_value(Register::A, result);
|
2021-11-08 00:28:44 +00:00
|
|
|
self.set_logic_op_flags(result, false, false);
|
2021-11-06 22:08:03 +00:00
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
_ => {
|
2021-11-07 04:18:45 +00:00
|
|
|
return Err(Error::new(&format!("{}: unimplemented instruction: {:?}", DEV_NAME, self.decoder.instruction)));
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-11-07 04:18:45 +00:00
|
|
|
fn rotate_left(&mut self, mut value: u8, rtype: RotateType) -> (u8, bool) {
|
2021-11-06 22:08:03 +00:00
|
|
|
let out_bit = get_msb(value as u16, Size::Byte);
|
|
|
|
|
|
|
|
let in_bit = match rtype {
|
|
|
|
RotateType::Bit9 => self.get_flag(Flags::Carry),
|
|
|
|
RotateType::Bit8 => out_bit,
|
|
|
|
};
|
|
|
|
|
|
|
|
value <<= 1;
|
|
|
|
value |= in_bit as u8;
|
2021-11-07 04:18:45 +00:00
|
|
|
(value, out_bit)
|
2021-11-06 22:08:03 +00:00
|
|
|
}
|
|
|
|
|
2021-11-07 04:18:45 +00:00
|
|
|
fn rotate_right(&mut self, mut value: u8, rtype: RotateType) -> (u8, bool) {
|
2021-11-06 22:08:03 +00:00
|
|
|
let out_bit = (value & 0x01) != 0;
|
|
|
|
|
|
|
|
let in_bit = match rtype {
|
|
|
|
RotateType::Bit9 => self.get_flag(Flags::Carry),
|
|
|
|
RotateType::Bit8 => out_bit,
|
|
|
|
};
|
|
|
|
|
|
|
|
value >>= 1;
|
|
|
|
value |= if in_bit { 0x80 } else { 0 };
|
2021-11-07 04:18:45 +00:00
|
|
|
(value, out_bit)
|
2021-11-06 22:08:03 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
fn add_to_regpair(&mut self, regpair: RegisterPair, value: i16) -> u16 {
|
|
|
|
let addr = match regpair {
|
|
|
|
RegisterPair::BC => &mut self.state.reg[0..2],
|
|
|
|
RegisterPair::DE => &mut self.state.reg[2..4],
|
|
|
|
RegisterPair::HL => &mut self.state.reg[4..6],
|
|
|
|
RegisterPair::AF => &mut self.state.reg[6..8],
|
|
|
|
_ => panic!("RegPair is not supported by inc/dec"),
|
|
|
|
};
|
|
|
|
|
|
|
|
let result = (read_beu16(addr) as i16).wrapping_add(value) as u16;
|
|
|
|
write_beu16(addr, result);
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-11-03 03:58:03 +00:00
|
|
|
fn push_word(&mut self, value: u16) -> Result<(), Error> {
|
2021-11-09 19:03:57 +00:00
|
|
|
self.state.sp = self.state.sp.wrapping_sub(1);
|
2021-11-03 03:58:03 +00:00
|
|
|
self.port.write_u8(self.state.sp as Address, (value >> 8) as u8)?;
|
2021-11-09 19:03:57 +00:00
|
|
|
self.state.sp = self.state.sp.wrapping_sub(1);
|
2021-11-03 03:58:03 +00:00
|
|
|
self.port.write_u8(self.state.sp as Address, (value & 0x00FF) as u8)?;
|
|
|
|
Ok(())
|
2021-11-02 05:06:40 +00:00
|
|
|
}
|
2021-11-03 03:58:03 +00:00
|
|
|
|
|
|
|
fn pop_word(&mut self) -> Result<u16, Error> {
|
|
|
|
let mut value = 0;
|
|
|
|
value = self.port.read_u8(self.state.sp as Address)? as u16;
|
2021-11-09 19:03:57 +00:00
|
|
|
self.state.sp = self.state.sp.wrapping_add(1);
|
2021-11-03 03:58:03 +00:00
|
|
|
value |= (self.port.read_u8(self.state.sp as Address)? as u16) << 8;
|
2021-11-09 19:03:57 +00:00
|
|
|
self.state.sp = self.state.sp.wrapping_add(1);
|
2021-11-03 03:58:03 +00:00
|
|
|
Ok(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_load_target_value(&mut self, target: LoadTarget) -> Result<u16, Error> {
|
|
|
|
let value = match target {
|
|
|
|
LoadTarget::DirectRegByte(reg) => self.get_register_value(reg) as u16,
|
2021-11-06 22:08:03 +00:00
|
|
|
LoadTarget::DirectRegHalfByte(reg) => self.get_index_register_half_value(reg) as u16,
|
2021-11-03 03:58:03 +00:00
|
|
|
LoadTarget::DirectRegWord(regpair) => self.get_register_pair_value(regpair),
|
|
|
|
LoadTarget::IndirectRegByte(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
self.port.read_u8(addr as Address)? as u16
|
|
|
|
},
|
2021-11-09 19:03:57 +00:00
|
|
|
LoadTarget::IndirectOffsetByte(index_reg, offset) => {
|
|
|
|
let addr = self.get_index_register_value(index_reg);
|
|
|
|
self.port.read_u8(((addr as i16).wrapping_add(offset as i16)) as Address)? as u16
|
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
LoadTarget::IndirectRegWord(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
self.port.read_leu16(addr as Address)?
|
|
|
|
},
|
|
|
|
//LoadTarget::DirectAltRegByte(reg),
|
|
|
|
//LoadTarget::DirectSpecialRegByte(reg),
|
|
|
|
LoadTarget::IndirectByte(addr) => {
|
|
|
|
self.port.read_u8(addr as Address)? as u16
|
|
|
|
},
|
|
|
|
LoadTarget::IndirectWord(addr) => {
|
|
|
|
self.port.read_leu16(addr as Address)?
|
|
|
|
},
|
|
|
|
LoadTarget::ImmediateByte(data) => data as u16,
|
|
|
|
LoadTarget::ImmediateWord(data) => data,
|
|
|
|
_ => panic!("Unsupported LoadTarget for set"),
|
|
|
|
};
|
|
|
|
Ok(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_load_target_value(&mut self, target: LoadTarget, value: u16) -> Result<(), Error> {
|
|
|
|
match target {
|
|
|
|
LoadTarget::DirectRegByte(reg) => self.set_register_value(reg, value as u8),
|
2021-11-06 22:08:03 +00:00
|
|
|
LoadTarget::DirectRegHalfByte(reg) => self.set_index_register_half_value(reg, value as u8),
|
2021-11-03 03:58:03 +00:00
|
|
|
LoadTarget::DirectRegWord(regpair) => self.set_register_pair_value(regpair, value),
|
|
|
|
LoadTarget::IndirectRegByte(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
self.port.write_u8(addr as Address, value as u8)?;
|
|
|
|
},
|
2021-11-09 19:03:57 +00:00
|
|
|
LoadTarget::IndirectOffsetByte(index_reg, offset) => {
|
|
|
|
let addr = self.get_index_register_value(index_reg);
|
|
|
|
self.port.write_u8(((addr as i16).wrapping_add(offset as i16)) as Address, value as u8)?;
|
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
LoadTarget::IndirectRegWord(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
self.port.write_leu16(addr as Address, value)?;
|
|
|
|
},
|
|
|
|
//LoadTarget::DirectAltRegByte(reg),
|
|
|
|
//LoadTarget::DirectSpecialRegByte(reg),
|
|
|
|
LoadTarget::IndirectByte(addr) => {
|
|
|
|
self.port.write_u8(addr as Address, value as u8)?;
|
|
|
|
},
|
|
|
|
LoadTarget::IndirectWord(addr) => {
|
|
|
|
self.port.write_leu16(addr as Address, value)?;
|
|
|
|
},
|
2021-11-09 19:03:57 +00:00
|
|
|
_ => panic!("Unsupported LoadTarget for set: {:?}", target),
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_target_value(&mut self, target: Target) -> Result<u8, Error> {
|
|
|
|
match target {
|
|
|
|
Target::DirectReg(reg) => Ok(self.get_register_value(reg)),
|
2021-11-06 22:08:03 +00:00
|
|
|
Target::DirectRegHalf(reg) => Ok(self.get_index_register_half_value(reg)),
|
2021-11-03 03:58:03 +00:00
|
|
|
Target::IndirectReg(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
Ok(self.port.read_u8(addr as Address)?)
|
|
|
|
},
|
2021-11-06 22:08:03 +00:00
|
|
|
Target::IndirectOffset(reg, offset) => {
|
|
|
|
let addr = (self.get_index_register_value(reg) as i16) + (offset as i16);
|
|
|
|
Ok(self.port.read_u8(addr as Address)?)
|
|
|
|
},
|
2021-11-03 03:58:03 +00:00
|
|
|
Target::Immediate(data) => Ok(data),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_target_value(&mut self, target: Target, value: u8) -> Result<(), Error> {
|
|
|
|
match target {
|
|
|
|
Target::DirectReg(reg) => self.set_register_value(reg, value),
|
2021-11-06 22:08:03 +00:00
|
|
|
Target::DirectRegHalf(reg) => self.set_index_register_half_value(reg, value),
|
2021-11-03 03:58:03 +00:00
|
|
|
Target::IndirectReg(regpair) => {
|
|
|
|
let addr = self.get_register_pair_value(regpair);
|
|
|
|
self.port.write_u8(addr as Address, value)?;
|
|
|
|
},
|
|
|
|
_ => panic!("Unsupported LoadTarget for set"),
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
fn get_opt_src_target_value(&mut self, opt_src: OptionalSource, target: Target) -> Result<u8, Error> {
|
|
|
|
match opt_src {
|
|
|
|
None => self.get_target_value(target),
|
|
|
|
Some((reg, offset)) => {
|
|
|
|
let addr = (self.get_index_register_value(reg) as i16) + (offset as i16);
|
|
|
|
self.port.read_u8(addr as Address)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-03 03:58:03 +00:00
|
|
|
fn get_register_value(&mut self, reg: Register) -> u8 {
|
2021-11-06 22:08:03 +00:00
|
|
|
self.state.reg[reg as usize]
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn set_register_value(&mut self, reg: Register, value: u8) {
|
2021-11-06 22:08:03 +00:00
|
|
|
self.state.reg[reg as usize] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_index_register_half_value(&mut self, reg: IndexRegisterHalf) -> u8 {
|
|
|
|
match reg {
|
|
|
|
IndexRegisterHalf::IXH => (self.state.ix >> 8) as u8,
|
|
|
|
IndexRegisterHalf::IXL => (self.state.ix & 0x00FF) as u8,
|
|
|
|
IndexRegisterHalf::IYH => (self.state.iy >> 8) as u8,
|
|
|
|
IndexRegisterHalf::IYL => (self.state.iy & 0x00FF) as u8,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_index_register_half_value(&mut self, reg: IndexRegisterHalf, value: u8) {
|
|
|
|
match reg {
|
|
|
|
IndexRegisterHalf::IXH => { self.state.ix |= (value as u16) << 8; },
|
|
|
|
IndexRegisterHalf::IXL => { self.state.ix |= value as u16; },
|
|
|
|
IndexRegisterHalf::IYH => { self.state.iy |= (value as u16) << 8; },
|
|
|
|
IndexRegisterHalf::IYL => { self.state.iy |= value as u16; },
|
|
|
|
}
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn get_register_pair_value(&mut self, regpair: RegisterPair) -> u16 {
|
|
|
|
match regpair {
|
|
|
|
RegisterPair::BC => read_beu16(&self.state.reg[0..2]),
|
|
|
|
RegisterPair::DE => read_beu16(&self.state.reg[2..4]),
|
|
|
|
RegisterPair::HL => read_beu16(&self.state.reg[4..6]),
|
|
|
|
RegisterPair::AF => read_beu16(&self.state.reg[6..8]),
|
|
|
|
RegisterPair::SP => self.state.sp,
|
2021-11-06 22:08:03 +00:00
|
|
|
RegisterPair::IX => self.state.ix,
|
|
|
|
RegisterPair::IY => self.state.iy,
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_register_pair_value(&mut self, regpair: RegisterPair, value: u16) {
|
|
|
|
match regpair {
|
|
|
|
RegisterPair::BC => { write_beu16(&mut self.state.reg[0..2], value); },
|
|
|
|
RegisterPair::DE => { write_beu16(&mut self.state.reg[2..4], value); },
|
|
|
|
RegisterPair::HL => { write_beu16(&mut self.state.reg[4..6], value); },
|
|
|
|
RegisterPair::AF => { write_beu16(&mut self.state.reg[6..8], value); },
|
|
|
|
RegisterPair::SP => { self.state.sp = value; },
|
2021-11-06 22:08:03 +00:00
|
|
|
RegisterPair::IX => { self.state.ix = value; },
|
|
|
|
RegisterPair::IY => { self.state.iy = value; },
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-06 22:08:03 +00:00
|
|
|
fn get_index_register_value(&mut self, reg: IndexRegister) -> u16 {
|
|
|
|
match reg {
|
|
|
|
IndexRegister::IX => self.state.ix,
|
|
|
|
IndexRegister::IY => self.state.iy,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-03 03:58:03 +00:00
|
|
|
fn get_current_condition(&mut self, cond: Condition) -> bool {
|
|
|
|
match cond {
|
|
|
|
Condition::NotZero => !self.get_flag(Flags::Zero),
|
|
|
|
Condition::Zero => self.get_flag(Flags::Zero),
|
|
|
|
Condition::NotCarry => !self.get_flag(Flags::Carry),
|
|
|
|
Condition::Carry => self.get_flag(Flags::Carry),
|
|
|
|
Condition::ParityOdd => !self.get_flag(Flags::Parity),
|
|
|
|
Condition::ParityEven => self.get_flag(Flags::Parity),
|
|
|
|
Condition::Positive => !self.get_flag(Flags::Sign),
|
|
|
|
Condition::Negative => self.get_flag(Flags::Sign),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
fn set_numeric_flags(&mut self, value: u16, size: Size) {
|
|
|
|
let sign = if get_msb(value, size) { Flags::Sign as u8 } else { 0 };
|
|
|
|
let zero = if value == 0 { Flags::Zero as u8 } else { 0 };
|
|
|
|
self.set_flags(FLAGS_NUMERIC, sign | zero);
|
2021-11-05 04:30:33 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
fn set_parity_flags(&mut self, value: u8) {
|
|
|
|
let mask = (Flags::Parity as u8) | (Flags::AddSubtract as u8);
|
|
|
|
let parity = if (value.count_ones() & 0x01) == 0 { Flags::Parity as u8 } else { 0 };
|
|
|
|
self.set_flags(mask, parity);
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
fn set_arithmetic_op_flags(&mut self, value: u16, size: Size, addsub: bool, carry: bool, overflow: bool, half_carry: bool) {
|
|
|
|
self.state.reg[Register::F as usize] = 0;
|
|
|
|
self.set_numeric_flags(value, size);
|
2021-11-05 04:30:33 +00:00
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
let addsub_flag = if addsub { Flags::AddSubtract as u8 } else { 0 };
|
|
|
|
let overflow_flag = if overflow { Flags::Parity as u8 } else { 0 };
|
|
|
|
let carry_flag = if carry { Flags::Carry as u8 } else { 0 };
|
|
|
|
let half_carry_flag = if half_carry { Flags::HalfCarry as u8 } else { 0 };
|
|
|
|
self.set_flags(FLAGS_ARITHMETIC, addsub_flag | overflow_flag | carry_flag | half_carry_flag);
|
2021-11-05 04:30:33 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
fn set_logic_op_flags(&mut self, value: u8, carry: bool, half_carry: bool) {
|
|
|
|
self.state.reg[Register::F as usize] = 0;
|
|
|
|
self.set_numeric_flags(value as u16, Size::Byte);
|
|
|
|
self.set_parity_flags(value);
|
|
|
|
|
|
|
|
let carry_flag = if carry { Flags::Carry as u8 } else { 0 };
|
|
|
|
let half_carry_flag = if half_carry { Flags::HalfCarry as u8 } else { 0 };
|
|
|
|
self.set_flags(FLAGS_CARRY_HALF_CARRY, carry_flag | half_carry_flag);
|
2021-11-03 03:58:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn get_flag(&self, flag: Flags) -> bool {
|
|
|
|
self.get_flags() & (flag as u8) != 0
|
|
|
|
}
|
|
|
|
|
2021-11-05 04:30:33 +00:00
|
|
|
#[inline(always)]
|
|
|
|
fn set_flag(&mut self, flag: Flags, value: bool) {
|
|
|
|
self.state.reg[Register::F as usize] = self.state.reg[Register::F as usize] & !(flag as u8);
|
|
|
|
if value {
|
|
|
|
self.state.reg[Register::F as usize] |= flag as u8;
|
|
|
|
}
|
|
|
|
}
|
2021-11-08 00:28:44 +00:00
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn get_flags(&self) -> u8 {
|
|
|
|
self.state.reg[Register::F as usize]
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn set_flags(&mut self, mask: u8, values: u8) {
|
|
|
|
self.state.reg[Register::F as usize] = (self.state.reg[Register::F as usize] & !mask) | values;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_bytes(operand1: u8, operand2: u8) -> (u8, bool, bool) {
|
|
|
|
let (result, carry) = operand1.overflowing_add(operand2);
|
|
|
|
let (_, overflow) = (operand1 as i8).overflowing_add(operand2 as i8);
|
|
|
|
(result, carry, overflow)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_words(operand1: u16, operand2: u16) -> (u16, bool, bool) {
|
|
|
|
let (result, carry) = operand1.overflowing_add(operand2);
|
|
|
|
let (_, overflow) = ((operand1 as i8) as i16).overflowing_add((operand2 as i8) as i16);
|
|
|
|
(result, carry, overflow)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sub_bytes(operand1: u8, operand2: u8) -> (u8, bool, bool) {
|
|
|
|
let (result, carry) = operand1.overflowing_sub(operand2);
|
|
|
|
let (_, overflow) = (operand1 as i8).overflowing_sub(operand2 as i8);
|
|
|
|
(result, carry, overflow)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sub_words(operand1: u16, operand2: u16) -> (u16, bool, bool) {
|
|
|
|
let (result, carry) = operand1.overflowing_sub(operand2);
|
|
|
|
let (_, overflow) = ((operand1 as i8) as i16).overflowing_sub((operand2 as i8) as i16);
|
|
|
|
(result, carry, overflow)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn get_msb_byte(value: u8) -> bool {
|
|
|
|
(value & 0x80) != 0
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn get_msb_word(value: u16) -> bool {
|
|
|
|
(value & 0x8000) != 0
|
2021-11-05 04:30:33 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:28:44 +00:00
|
|
|
#[inline(always)]
|
2021-11-05 04:30:33 +00:00
|
|
|
fn get_msb(value: u16, size: Size) -> bool {
|
|
|
|
match size {
|
|
|
|
Size::Byte => (value & 0x0080) != 0,
|
|
|
|
Size::Word => (value & 0x8000) != 0,
|
|
|
|
}
|
2021-11-02 05:06:40 +00:00
|
|
|
}
|
|
|
|
|