moa/src/cpus/z80/execute.rs

655 lines
24 KiB
Rust
Raw Normal View History

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-05 04:30:33 +00:00
use super::decode::{Condition, Instruction, LoadTarget, Target, RegisterPair, 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-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) {
//self.dump_state(system);
}
}
impl Interruptable for Z80 { }
2021-11-05 04:30:33 +00:00
impl Debuggable for Z80 {
fn add_breakpoint(&mut self, addr: Address) {
//self.debugger.breakpoints.push(addr as u32);
}
fn remove_breakpoint(&mut self, addr: Address) {
//if let Some(index) = self.debugger.breakpoints.iter().position(|a| *a == addr as u32) {
// self.debugger.breakpoints.remove(index);
//}
}
fn print_current_step(&mut self, system: &System) -> Result<(), Error> {
//self.decoder.decode_at(&mut self.port, self.state.pc)?;
//self.decoder.dump_decoded(&mut self.port);
self.dump_state(system);
Ok(())
}
fn print_disassembly(&mut self, addr: Address, count: usize) {
//let mut decoder = M68kDecoder::new(self.cputype, 0);
//decoder.dump_disassembly(&mut self.port, self.state.pc, 0x1000);
}
fn execute_command(&mut self, system: &System, args: &[&str]) -> Result<bool, Error> {
Ok(false)
}
}
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)?;
Ok(())
}
pub fn decode_next(&mut self, system: &System) -> Result<(), Error> {
//self.check_breakpoints(system);
//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-05 04:30:33 +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 {
//Instruction::ADCa(target) => {
//},
//Instruction::ADChl(regpair) => {
//},
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);
let (result, carry) = acc.overflowing_add(src);
self.set_add_flags(result as u16, Size::Byte, carry);
self.set_register_value(Register::A, result);
},
Instruction::ADDhl(regpair) => {
let src = self.get_register_pair_value(regpair);
let hl = self.get_register_pair_value(RegisterPair::HL);
let (result, carry) = hl.overflowing_add(src);
self.set_add_flags(result as u16, Size::Word, carry);
self.set_register_pair_value(RegisterPair::HL, result);
},
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);
self.set_logic_op_flags(result as u16, Size::Byte, true);
},
Instruction::CP(target) => {
let src = self.get_target_value(target)?;
let acc = self.get_register_value(Register::A);
let (result, carry) = acc.overflowing_sub(src);
self.set_sub_flags(result as u16, Size::Byte, carry);
},
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-03 03:58:03 +00:00
//Instruction::NEG => {
//},
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-05 04:30:33 +00:00
self.set_logic_op_flags(result as u16, Size::Byte, false);
2021-11-03 03:58:03 +00:00
},
//Instruction::SBCa(target) => {
//},
//Instruction::SBChl(regpair) => {
//},
2021-11-05 04:30:33 +00:00
Instruction::SUB(target) => {
let src = self.get_target_value(target)?;
let acc = self.get_register_value(Register::A);
let (result, carry) = acc.overflowing_sub(src);
self.set_sub_flags(result as u16, Size::Byte, carry);
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);
self.set_logic_op_flags(result as u16, Size::Byte, false);
},
2021-11-03 03:58:03 +00:00
//Instruction::BIT(u8, target) => {
//},
//Instruction::RES(u8, target) => {
//},
//Instruction::RL(target) => {
//},
//Instruction::RLC(target) => {
//},
//Instruction::RR(target) => {
//},
//Instruction::RRC(target) => {
//},
//Instruction::SET(u8, target) => {
//},
//Instruction::SLA(target) => {
//},
//Instruction::SLL(target) => {
//},
//Instruction::SRA(target) => {
//},
//Instruction::SRL(target) => {
//},
Instruction::DEC8(target) => {
2021-11-05 04:30:33 +00:00
let value = self.get_target_value(target)?;
let (result, carry) = value.overflowing_sub(1);
self.set_sub_flags(result as u16, Size::Byte, carry);
self.set_target_value(target, result)?;
},
Instruction::DEC16(regpair) => {
let value = self.get_register_pair_value(regpair);
let (result, carry) = value.overflowing_sub(1);
self.set_sub_flags(result, Size::Word, carry);
self.set_register_pair_value(regpair, result);
},
Instruction::INC8(target) => {
let value = self.get_target_value(target)?;
let (result, carry) = value.overflowing_add(1);
self.set_add_flags(result as u16, Size::Byte, carry);
self.set_target_value(target, result)?;
},
Instruction::INC16(regpair) => {
let value = self.get_register_pair_value(regpair);
let (result, carry) = value.overflowing_add(1);
self.set_add_flags(result, Size::Word, carry);
self.set_register_pair_value(regpair, result);
2021-11-03 03:58:03 +00:00
},
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::EXhlsp => {
let (sp_addr, hl) = (self.get_register_pair_value(RegisterPair::SP), self.get_register_pair_value(RegisterPair::HL));
let sp = self.port.read_leu16(sp_addr as Address)?;
self.set_register_pair_value(RegisterPair::HL, sp);
self.port.write_leu16(sp_addr as Address, hl)?;
},
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-03 03:58:03 +00:00
Instruction::LD(dest, src) => {
let src_value = self.get_load_target_value(src)?;
self.set_load_target_value(dest, src_value)?;
},
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::INx(u8) => {
//},
//Instruction::INic(reg) => {
//},
Instruction::OUTx(port) => {
// TODO this needs to be fixed
println!("OUT: {:x}", self.state.reg[Register::A as usize]);
},
//Instruction::OUTic(reg) => {
//},
Instruction::CALL(addr) => {
self.push_word(self.decoder.end)?;
self.state.pc = addr;
},
Instruction::CALLcc(cond, addr) => {
if self.get_current_condition(cond) {
self.push_word(self.decoder.end)?;
self.state.pc = addr;
}
},
2021-11-05 04:30:33 +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);
if result != 0 {
self.state.pc = ((self.state.pc as i16) + (offset as i16)) as u16;
}
},
2021-11-03 03:58:03 +00:00
Instruction::JP(addr) => {
self.state.pc = addr;
},
2021-11-05 04:30:33 +00:00
Instruction::JPIndirectHL => {
let hl = self.get_register_pair_value(RegisterPair::HL);
let addr = self.port.read_leu16(hl as Address)?;
self.state.pc = addr;
},
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;
}
},
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-03 03:58:03 +00:00
2021-11-05 04:30:33 +00:00
Instruction::DI => {
self.state.interrupts_enabled = false;
},
Instruction::EI => {
self.state.interrupts_enabled = true;
},
2021-11-03 03:58:03 +00:00
//Instruction::IM(u8) => {
//},
Instruction::NOP => { },
2021-11-05 04:30:33 +00:00
Instruction::HALT => {
self.state.status = Status::Halted;
},
Instruction::RST(addr) => {
self.push_word(self.decoder.end)?;
self.state.pc = addr as u16;
},
2021-11-03 03:58:03 +00:00
2021-11-05 04:30:33 +00:00
Instruction::CCF => {
self.set_flag(Flags::Carry, false);
},
2021-11-03 03:58:03 +00:00
//Instruction::DAA => {
//},
//Instruction::RLA => {
//},
//Instruction::RLCA => {
//},
//Instruction::RRA => {
//},
//Instruction::RRCA => {
//},
//Instruction::RRD => {
//},
//Instruction::RLD => {
//},
2021-11-05 04:30:33 +00:00
Instruction::SCF => {
self.set_flag(Flags::Carry, true);
},
2021-11-03 03:58:03 +00:00
//Instruction::CPD => {
//},
//Instruction::CPDR => {
//},
//Instruction::CPI => {
//},
//Instruction::CPIR => {
//},
//Instruction::IND => {
//},
//Instruction::INDR => {
//},
//Instruction::INI => {
//},
//Instruction::INIR => {
//},
//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)?;
self.register_pair_add_value(RegisterPair::DE, 1);
self.register_pair_add_value(RegisterPair::HL, 1);
let count = self.register_pair_add_value(RegisterPair::BC, -1);
if count != 0 {
self.state.pc -= 2;
}
},
//Instruction::OTDR => {
//},
//Instruction::OTIR => {
//},
//Instruction::OUTD => {
//},
//Instruction::OUTI => {
//},
_ => {
panic!("unimplemented");
}
}
Ok(())
}
fn push_word(&mut self, value: u16) -> Result<(), Error> {
self.state.sp -= 1;
self.port.write_u8(self.state.sp as Address, (value >> 8) as u8)?;
self.state.sp -= 1;
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;
self.state.sp += 1;
value |= (self.port.read_u8(self.state.sp as Address)? as u16) << 8;
self.state.sp += 1;
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,
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
},
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),
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)?;
},
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)?;
},
_ => panic!("Unsupported LoadTarget for set"),
}
Ok(())
}
fn get_target_value(&mut self, target: Target) -> Result<u8, Error> {
match target {
Target::DirectReg(reg) => Ok(self.get_register_value(reg)),
Target::IndirectReg(regpair) => {
let addr = self.get_register_pair_value(regpair);
Ok(self.port.read_u8(addr as Address)?)
},
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),
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(())
}
fn get_register_value(&mut self, reg: Register) -> u8 {
let i = (reg as u8) as usize;
self.state.reg[i]
}
fn set_register_value(&mut self, reg: Register, value: u8) {
let i = (reg as u8) as usize;
self.state.reg[i] = value;
}
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,
}
}
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; },
}
}
fn register_pair_add_value(&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],
RegisterPair::SP => panic!("SP is not supported by inc/dec"),
};
let result = (read_beu16(addr) as i16).wrapping_add(value) as u16;
write_beu16(addr, result);
result
}
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-05 04:30:33 +00:00
fn set_add_flags(&mut self, value: u16, size: Size, carry: bool) {
2021-11-03 03:58:03 +00:00
let mut flags = 0;
2021-11-05 04:30:33 +00:00
if get_msb(value, size) {
flags |= Flags::Sign as u8;
}
2021-11-03 03:58:03 +00:00
if value == 0 {
flags |= Flags::Zero as u8;
}
2021-11-05 04:30:33 +00:00
if (value & 0x10) != 0 {
flags |= Flags::HalfCarry as u8;
}
// TODO need overflow
if carry {
flags |= Flags::Carry as u8;
}
self.state.reg[Register::F as usize] = flags;
}
fn set_sub_flags(&mut self, value: u16, size: Size, carry: bool) {
let mut flags = Flags::AddSubtract as u8;
if get_msb(value, size) {
2021-11-03 03:58:03 +00:00
flags |= Flags::Sign as u8;
}
2021-11-05 04:30:33 +00:00
if value == 0 {
flags |= Flags::Zero as u8;
}
// TODO need overflow and half carry
2021-11-03 03:58:03 +00:00
if carry {
flags |= Flags::Carry as u8;
}
self.state.reg[Register::F as usize] = flags;
}
2021-11-05 04:30:33 +00:00
fn set_logic_op_flags(&mut self, value: u16, size: Size, half_carry: bool) {
let mut flags = 0;
if get_msb(value, size) {
flags |= Flags::Sign as u8;
}
if value == 0 {
flags |= Flags::Zero as u8;
}
if half_carry {
flags |= Flags::HalfCarry as u8;
}
if (value.count_ones() & 0x01) == 0 {
flags |= Flags::Parity as u8;
}
self.state.reg[Register::F as usize] = flags;
}
2021-11-03 03:58:03 +00:00
#[inline(always)]
fn get_flags(&self) -> u8 {
self.state.reg[Register::F as usize]
}
#[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;
}
}
}
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
}