add more instructions

This commit is contained in:
Lauchmelder 2022-08-24 01:57:42 +02:00
parent be3ebe9041
commit a2695b5ca1
3 changed files with 178 additions and 16 deletions

View file

@ -25,6 +25,21 @@ impl CPU
print!("{: <30}", format!("#${:02X}", bus.borrow().read_cpu(self.absolute_addr)));
}
pub fn imp(&mut self)
{
print!("{: <30}", "");
}
pub fn rel(&mut self)
{
let bus = self.bus.upgrade().unwrap();
self.relative_addr = bus.borrow().read_cpu(self.pc) as i8;
self.pc += 1;
print!("{: <30}", format!("${:02X}", self.relative_addr));
}
pub fn zpg(&mut self)
{
let bus = self.bus.upgrade().unwrap();

View file

@ -22,7 +22,7 @@ macro_rules! instr
{
($instr: ident, $addr: ident, $cyc: literal, $len: literal) =>
{
Instruction
Option::Some(Instruction
{
action: CPU::$instr,
addressing: CPU::$addr,
@ -30,7 +30,7 @@ macro_rules! instr
length: $len,
name: s3::new(stringify!($instr)).unwrap()
}
})
}
}
@ -39,7 +39,7 @@ pub struct CPU
pub cycle: u8,
total_cycles: u64,
pub absolute_addr: u16,
pub relative_addr: u8,
pub relative_addr: i8,
pub acc: u8,
pub x: u8,
@ -60,9 +60,36 @@ impl CPU
const UNDEF_INSTR: Option<Instruction> = None;
let mut instr_set = [UNDEF_INSTR; 256];
instr_set[0x4C] = Option::Some(instr!(jmp, abs, 3, 3));
instr_set[0x86] = Option::Some(instr!(stx, zpg, 3, 2));
instr_set[0xA2] = Option::Some(instr!(ldx, imm, 2, 2));
instr_set[0x10] = instr!(bpl, rel, 2, 2);
instr_set[0x18] = instr!(clc, imp, 2, 1);
instr_set[0x20] = instr!(jsr, abs, 6, 3);
instr_set[0x24] = instr!(bit, zpg, 3, 2);
instr_set[0x30] = instr!(bmi, rel, 2, 2);
instr_set[0x38] = instr!(sec, imp, 2, 1);
instr_set[0x4C] = instr!(jmp, abs, 3, 3);
instr_set[0x50] = instr!(bvc, rel, 2, 2);
instr_set[0x70] = instr!(bvs, rel, 2, 2);
instr_set[0x85] = instr!(sta, zpg, 3, 2);
instr_set[0x86] = instr!(stx, zpg, 3, 2);
instr_set[0x90] = instr!(bcc, rel, 2, 2);
instr_set[0xA2] = instr!(ldx, imm, 2, 2);
instr_set[0xA9] = instr!(lda, imm, 2, 2);
instr_set[0xB0] = instr!(bcs, rel, 2, 2);
instr_set[0xD0] = instr!(bne, rel, 2, 2);
instr_set[0xEA] = instr!(nop, imp, 2, 1);
instr_set[0xF0] = instr!(beq, rel, 2, 2);
CPU {
cycle: 0,

View file

@ -35,6 +35,94 @@ macro_rules! set_flag_to
}
}
macro_rules! test_flag
{
($target: expr, $flag: expr) =>
{
($target & (1u8 << ($flag as u8))) != 0
}
}
macro_rules! push
{
($bus: expr, $sp: expr, $val: expr) =>
{
$bus.write_cpu(0x0100 + $sp as u16, ($val & 0xFF) as u8);
$sp -= 1;
}
}
macro_rules! branch
{
($self: ident) =>
{
let branch_target = $self.pc.wrapping_add($self.relative_addr as u16);
$self.cycle += 1;
if (branch_target & 0xFF00) != ($self.pc & 0xFF00) // Branched to different page
{
$self.cycle += 1;
}
$self.pc = branch_target;
}
}
macro_rules! branch_on_fn
{
($name: tt, $flag: expr, $result: literal) =>
{
pub fn $name(&mut self)
{
if test_flag!(self.p, $flag) == $result
{
branch!(self);
}
}
}
}
macro_rules! set_flag_fn
{
($name: tt, $flag: expr, $result: literal) =>
{
pub fn $name(&mut self)
{
match $result
{
false => clear_flag!(self.p, $flag),
true => set_flag!(self.p, $flag)
}
}
}
}
macro_rules! load_fn
{
($name: tt, $register: ident) =>
{
pub fn $name(&mut self)
{
self.$register = self.fetch();
set_flag_to!(self.p, Bit::Negative, (self.$register & (1u8 << 7)) > 0);
set_flag_to!(self.p, Bit::Zero, self.$register == 0);
}
};
}
macro_rules! store_fn
{
($name: tt, $register: ident) =>
{
pub fn $name(&mut self)
{
let bus = self.bus.upgrade().unwrap();
bus.borrow_mut().write_cpu(self.absolute_addr, self.$register);
}
};
}
impl CPU
{
fn fetch(&mut self) -> u8
@ -43,23 +131,55 @@ impl CPU
return bus.borrow().read_cpu(self.absolute_addr);
}
branch_on_fn!(bcc, Bit::Carry, false);
branch_on_fn!(bcs, Bit::Carry, true);
branch_on_fn!(bne, Bit::Zero, false);
branch_on_fn!(beq, Bit::Zero, true);
branch_on_fn!(bpl, Bit::Negative, false);
branch_on_fn!(bmi, Bit::Negative, true);
branch_on_fn!(bvc, Bit::Overflow, false);
branch_on_fn!(bvs, Bit::Overflow, true);
set_flag_fn!(clc, Bit::Carry, false);
set_flag_fn!(sec, Bit::Carry, true);
load_fn!(lda, acc);
load_fn!(ldx, x);
load_fn!(ldy, y);
store_fn!(sta, acc);
store_fn!(stx, x);
store_fn!(sty, y);
pub fn bit(&mut self)
{
let bus = self.bus.upgrade().unwrap();
let value = bus.borrow().read_cpu(self.absolute_addr);
set_flag_to!(self.p, Bit::Negative, (value >> 7) & 0x1);
set_flag_to!(self.p, Bit::Overflow, (value >> 6) & 0x1);
set_flag_to!(self.p, Bit::Zero, (self.acc & value) == 0);
}
pub fn jmp(&mut self)
{
self.pc = self.absolute_addr;
}
pub fn ldx(&mut self)
{
self.x = self.fetch();
set_flag_to!(self.p, Bit::Negative, (self.x & (1u8 << 7)) > 0);
set_flag_to!(self.p, Bit::Zero, self.x == 0);
}
pub fn stx(&mut self)
pub fn jsr(&mut self)
{
let bus = self.bus.upgrade().unwrap();
bus.borrow_mut().write_cpu(self.absolute_addr, self.x);
push!(bus.borrow_mut(), self.sp, self.pc >> 8);
push!(bus.borrow_mut(), self.sp, self.pc);
self.pc = self.absolute_addr;
}
pub fn nop(&mut self)
{
}
}