304 lines
7.6 KiB
Rust
304 lines
7.6 KiB
Rust
use std::{mem::transmute, time::Duration};
|
|
|
|
use minifb::Window;
|
|
|
|
use crate::{processor::instructions::instructions::set, verbose_println, Memory};
|
|
|
|
use self::gpu::GPU;
|
|
|
|
pub mod gpu;
|
|
mod instructions;
|
|
mod opcodes;
|
|
|
|
#[derive(PartialEq)]
|
|
pub(crate) enum Flags {
|
|
Zero = 7,
|
|
NSubtract = 6,
|
|
HalfCarry = 5,
|
|
Carry = 4,
|
|
}
|
|
|
|
pub(crate) enum Direction {
|
|
Left,
|
|
Right,
|
|
}
|
|
|
|
pub struct CPU {
|
|
pub memory: Memory,
|
|
pub reg: Registers,
|
|
pub last_instruction: u8,
|
|
pub last_instruction_addr: u16,
|
|
pub window: Window,
|
|
pub gpu: GPU,
|
|
}
|
|
|
|
// Hz
|
|
const CLOCK_SPEED: f64 = 4.194304 * 1000000.;
|
|
const SPEEDUP: f64 = 1.;
|
|
const FF04_SPEED: f64 = 16384.;
|
|
|
|
impl CPU {
|
|
pub fn new(memory: Memory, window: Window) -> Self {
|
|
Self {
|
|
memory,
|
|
reg: Registers::default(),
|
|
last_instruction: 0x0,
|
|
last_instruction_addr: 0x0,
|
|
window,
|
|
gpu: GPU::default(),
|
|
}
|
|
}
|
|
|
|
pub fn exec_next(&mut self) {
|
|
self.last_instruction_addr = self.reg.pc;
|
|
let opcode = self.next_opcode();
|
|
self.last_instruction = opcode;
|
|
|
|
if self.memory.ime_scheduled > 0 {
|
|
self.memory.ime_scheduled = self.memory.ime_scheduled.saturating_sub(1);
|
|
if self.memory.ime_scheduled == 0 {
|
|
self.memory.ime = true;
|
|
}
|
|
}
|
|
|
|
verbose_println!(
|
|
"exec {:#4X} from pc: {:#X}",
|
|
opcode,
|
|
self.last_instruction_addr
|
|
);
|
|
let cycles = self.run_opcode(opcode);
|
|
self.increment_timers(cycles);
|
|
|
|
let interrupt_cycles = self.handle_interrupts();
|
|
self.increment_timers(interrupt_cycles);
|
|
}
|
|
|
|
fn increment_timers(&mut self, cycles: u8) {
|
|
let secs = (cycles * 4) as f64 / CLOCK_SPEED;
|
|
self.advance_gpu_clock(cycles);
|
|
|
|
self.memory.set(
|
|
0xFF04,
|
|
self.memory
|
|
.get(0xFF04)
|
|
.wrapping_add((FF04_SPEED * secs) as u8),
|
|
);
|
|
let (timer_enabled, timer_rate) = self.timer_scale();
|
|
if timer_enabled {
|
|
let (val, wrap) = self
|
|
.memory
|
|
.get(0xFF05)
|
|
.overflowing_add((secs * timer_rate as f64) as u8);
|
|
if wrap {
|
|
self.memory.set(0xFF05, self.memory.get(0xFF06));
|
|
self.memory.set(0xFF0F, set(self.memory.get(0xFF0F), 2));
|
|
} else {
|
|
self.memory.set(0xFF05, val);
|
|
}
|
|
}
|
|
spin_sleep::sleep(Duration::from_secs_f64(secs / SPEEDUP));
|
|
}
|
|
|
|
fn next_opcode(&mut self) -> u8 {
|
|
let opcode = self.memory.get(self.reg.pc);
|
|
self.reg.pc = self.reg.pc.wrapping_add(0x1);
|
|
return opcode;
|
|
}
|
|
|
|
fn handle_interrupts(&mut self) -> u8 {
|
|
if self.memory.ime {
|
|
let req_and_enabled = self.memory.get(0xFF0F) & self.memory.get(0xFFFF);
|
|
// all interrupts should last 5 cycles?
|
|
if get_bit(req_and_enabled, 0) {
|
|
// vblank
|
|
self.service_interrupt(0x40);
|
|
self.memory
|
|
.set(0xFF0F, clear_bit(self.memory.get(0xFF0F), 0));
|
|
5
|
|
} else if get_bit(req_and_enabled, 1) {
|
|
// lcd stat
|
|
self.service_interrupt(0x48);
|
|
self.memory
|
|
.set(0xFF0F, clear_bit(self.memory.get(0xFF0F), 1));
|
|
5
|
|
} else if get_bit(req_and_enabled, 2) {
|
|
// timer
|
|
self.service_interrupt(0x50);
|
|
self.memory
|
|
.set(0xFF0F, clear_bit(self.memory.get(0xFF0F), 2));
|
|
5
|
|
} else if get_bit(req_and_enabled, 3) {
|
|
// serial
|
|
self.service_interrupt(0x58);
|
|
self.memory
|
|
.set(0xFF0F, clear_bit(self.memory.get(0xFF0F), 3));
|
|
5
|
|
} else if get_bit(req_and_enabled, 4) {
|
|
// joypad
|
|
self.service_interrupt(0x60);
|
|
self.memory
|
|
.set(0xFF0F, clear_bit(self.memory.get(0xFF0F), 4));
|
|
5
|
|
} else {
|
|
0
|
|
}
|
|
} else {
|
|
0
|
|
}
|
|
}
|
|
|
|
fn service_interrupt(&mut self, addr: u16) {
|
|
self.push(self.reg.pc);
|
|
self.reg.pc = addr;
|
|
self.memory.ime = false;
|
|
}
|
|
|
|
fn timer_scale(&self) -> (bool, usize) {
|
|
let timer_control = self.memory.get(0xFF07);
|
|
let timer_enable = get_bit(timer_control, 2);
|
|
let timer_rate = match (get_bit(timer_control, 1), get_bit(timer_control, 0)) {
|
|
(true, true) => 256,
|
|
(true, false) => 64,
|
|
(false, true) => 16,
|
|
(false, false) => 1024,
|
|
};
|
|
(timer_enable, timer_rate)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy)]
|
|
pub enum Reg8 {
|
|
A,
|
|
B,
|
|
C,
|
|
D,
|
|
E,
|
|
H,
|
|
L,
|
|
}
|
|
|
|
#[derive(Clone, Copy)]
|
|
pub struct Registers {
|
|
pub af: u16,
|
|
pub bc: u16,
|
|
pub de: u16,
|
|
pub hl: u16,
|
|
pub sp: u16,
|
|
pub pc: u16,
|
|
}
|
|
|
|
impl Default for Registers {
|
|
fn default() -> Self {
|
|
// default post-bootrom values
|
|
Self {
|
|
af: 0x00B0,
|
|
// af: 0x01B0,
|
|
bc: 0x0013,
|
|
de: 0x00D8,
|
|
hl: 0x014D,
|
|
sp: 0xFFFE,
|
|
pc: 0x0000,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Registers {
|
|
fn get_8(&self, register: Reg8) -> u8 {
|
|
match register {
|
|
Reg8::A => self.af.get_high(),
|
|
Reg8::B => self.bc.get_high(),
|
|
Reg8::C => self.bc.get_low(),
|
|
Reg8::D => self.de.get_high(),
|
|
Reg8::E => self.de.get_low(),
|
|
Reg8::H => self.hl.get_high(),
|
|
Reg8::L => self.hl.get_low(),
|
|
}
|
|
}
|
|
fn set_8(&mut self, register: Reg8, val: u8) {
|
|
match register {
|
|
Reg8::A => self.af.set_high(val),
|
|
Reg8::B => self.bc.set_high(val),
|
|
Reg8::C => self.bc.set_low(val),
|
|
Reg8::D => self.de.set_high(val),
|
|
Reg8::E => self.de.set_low(val),
|
|
Reg8::H => self.hl.set_high(val),
|
|
Reg8::L => self.hl.set_low(val),
|
|
}
|
|
}
|
|
}
|
|
|
|
trait SplitRegister {
|
|
fn get_low(&self) -> u8;
|
|
fn get_high(&self) -> u8;
|
|
fn set_low(&mut self, val: u8);
|
|
fn set_high(&mut self, val: u8);
|
|
}
|
|
|
|
impl SplitRegister for u16 {
|
|
fn get_low(&self) -> u8 {
|
|
(*self & 0xFF) as u8
|
|
}
|
|
|
|
fn get_high(&self) -> u8 {
|
|
((*self >> 8) & 0xFF) as u8
|
|
}
|
|
|
|
fn set_low(&mut self, val: u8) {
|
|
*self = (*self & !0xff) | val as u16;
|
|
}
|
|
|
|
fn set_high(&mut self, val: u8) {
|
|
*self = (*self & !0xff00) | (val as u16) << 8;
|
|
}
|
|
}
|
|
|
|
fn as_signed(unsigned: u8) -> i8 {
|
|
unsafe {
|
|
return transmute(unsigned);
|
|
}
|
|
}
|
|
|
|
fn get_bit(byte: u8, flag: u8) -> bool {
|
|
let mask = 1 << flag;
|
|
let got = byte & mask;
|
|
return got > 0x0;
|
|
}
|
|
|
|
fn set_or_clear_bit(byte: u8, flag: u8, condition: bool) -> u8 {
|
|
if condition {
|
|
set_bit(byte, flag)
|
|
} else {
|
|
clear_bit(byte, flag)
|
|
}
|
|
}
|
|
|
|
fn set_bit(byte: u8, flag: u8) -> u8 {
|
|
byte | (1 << flag)
|
|
}
|
|
|
|
fn clear_bit(byte: u8, flag: u8) -> u8 {
|
|
byte & (!(1 << flag))
|
|
}
|
|
|
|
fn rotate(byte: u8, direction: &Direction) -> (u8, bool) {
|
|
match direction {
|
|
Direction::Left => {
|
|
let carry = get_bit(byte, 7);
|
|
let r = byte << 1;
|
|
return (r, carry);
|
|
}
|
|
Direction::Right => {
|
|
let carry = get_bit(byte, 0);
|
|
let r = byte >> 1;
|
|
return (r, carry);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn get_rotation_carry(direction: &Direction) -> u8 {
|
|
match direction {
|
|
Direction::Left => 0b1,
|
|
Direction::Right => 0b10000000,
|
|
}
|
|
}
|