Correctly implement MBC1 emulation
This commit is contained in:
parent
19f31cfbbf
commit
25d33d6e82
@ -1,16 +1,17 @@
|
|||||||
use super::mbc::MBC;
|
use super::mbc::MBC;
|
||||||
|
|
||||||
|
enum BankMode {
|
||||||
|
RomBankMode,
|
||||||
|
RamBankMode,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct MBC1 {
|
pub struct MBC1 {
|
||||||
rom: Box<[u8]>,
|
rom: Box<[u8]>,
|
||||||
ram: Box<[u8]>,
|
ram: Box<[u8]>,
|
||||||
bank_no: u8,
|
rom_bank_no: u8,
|
||||||
ram_rtc_enabled: bool,
|
bank_mode: BankMode,
|
||||||
ram_bank_no: u8,
|
|
||||||
|
|
||||||
rom_banks: u16,
|
|
||||||
|
|
||||||
bank_mode: u8,
|
|
||||||
bank_no_high: u8,
|
bank_no_high: u8,
|
||||||
|
ram_enable: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MBC1 {
|
impl MBC1 {
|
||||||
@ -18,15 +19,28 @@ impl MBC1 {
|
|||||||
MBC1 {
|
MBC1 {
|
||||||
rom: rom,
|
rom: rom,
|
||||||
ram: ram,
|
ram: ram,
|
||||||
bank_no: 0,
|
rom_bank_no: 0,
|
||||||
ram_rtc_enabled: false,
|
bank_mode: BankMode::RomBankMode,
|
||||||
ram_bank_no: 0,
|
|
||||||
rom_banks: 0,
|
|
||||||
bank_mode: 0,
|
|
||||||
bank_no_high: 0,
|
bank_no_high: 0,
|
||||||
|
|
||||||
|
ram_enable: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
fn active_rom_bank(&self) -> u8 {
|
||||||
|
match self.bank_mode {
|
||||||
|
BankMode::RomBankMode => self.rom_bank_no | self.bank_no_high << 5,
|
||||||
|
BankMode::RamBankMode => self.rom_bank_no
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn active_ram_bank(&self) -> u8 {
|
||||||
|
match self.bank_mode {
|
||||||
|
BankMode::RomBankMode => 0,
|
||||||
|
BankMode::RamBankMode => self.bank_no_high
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl MBC for MBC1 {
|
impl MBC for MBC1 {
|
||||||
fn read_byte(&self, addr: u16) -> u8 {
|
fn read_byte(&self, addr: u16) -> u8 {
|
||||||
@ -34,21 +48,14 @@ impl MBC for MBC1 {
|
|||||||
0x0000 ... 0x3FFF => self.rom[addr as usize],
|
0x0000 ... 0x3FFF => self.rom[addr as usize],
|
||||||
0x4000 ... 0x7FFF => {
|
0x4000 ... 0x7FFF => {
|
||||||
let addr = addr - 0x4000;
|
let addr = addr - 0x4000;
|
||||||
// println!("BankNo: {:02X}", self.bank_no);
|
let abs_addr: usize = addr as usize + self.active_rom_bank() as usize * 0x4000;
|
||||||
let abs_addr: usize = addr as usize + self.bank_no as usize * 0x4000;
|
|
||||||
let val: u8 = self.rom[abs_addr];
|
let val: u8 = self.rom[abs_addr];
|
||||||
val
|
val
|
||||||
},
|
},
|
||||||
0xA000 ... 0xBFFF => {
|
0xA000 ... 0xBFFF => {
|
||||||
// TODO: Safty checks? Or let rust handle this?
|
|
||||||
let addr = addr - 0xA000;
|
let addr = addr - 0xA000;
|
||||||
if self.ram_bank_no < 4 {
|
println!("Access [{:02X}] {:04X}", self.active_ram_bank(), addr);
|
||||||
println!("Access [{:02X}] {:04X}", self.ram_bank_no, addr);
|
self.ram[self.active_ram_bank() as usize * 0x2000 + addr as usize]
|
||||||
self.ram[self.ram_bank_no as usize * 0x2000 + addr as usize]
|
|
||||||
} else {
|
|
||||||
println!("Ignoring RTC read");
|
|
||||||
0
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
panic!("Cartride: Unable to read from {:04X}", addr);
|
panic!("Cartride: Unable to read from {:04X}", addr);
|
||||||
@ -60,33 +67,31 @@ impl MBC for MBC1 {
|
|||||||
match addr {
|
match addr {
|
||||||
0x0000 ... 0x1FFF => {
|
0x0000 ... 0x1FFF => {
|
||||||
match val {
|
match val {
|
||||||
0x0A => self.ram_rtc_enabled = true,
|
0x0A => self.ram_enable = true,
|
||||||
0x00 => self.ram_rtc_enabled = false,
|
0x00 => self.ram_enable = false,
|
||||||
_ => println!("Unknown MBC value {:02X} for {:04X}", val, addr)
|
_ => println!("Unknown MBC1 value {:02X} for {:04X}", val, addr)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
0x2000 ... 0x3FFF => self.bank_no = val & 0x7F,
|
0x2000 ... 0x3FFF => {
|
||||||
0x4000 ... 0x5FFF => {
|
println!("Selecting bank {:02X}", self.rom_bank_no);
|
||||||
// RAM bank select
|
if val != 0 {
|
||||||
match val {
|
self.rom_bank_no = val & 0x1F;
|
||||||
0x00 ... 0x03 => self.ram_bank_no = 0, //val,
|
|
||||||
0x08 ... 0x0C => self.ram_bank_no = val, // RTC clock values, TODO
|
|
||||||
_ => panic!("Unknown MBC1 RAM BANK NO: {:02X}", val)
|
|
||||||
}
|
|
||||||
|
|
||||||
},
|
|
||||||
0x6000 ... 0x7FFF => {
|
|
||||||
// Latch clock data, ignore.
|
|
||||||
},
|
|
||||||
0xA000 ... 0xBFFF => {
|
|
||||||
// TODO: Safty checks? Or let rust handle this?
|
|
||||||
let addr = addr - 0xA000;
|
|
||||||
if self.ram_bank_no < 4 {
|
|
||||||
self.ram[self.ram_bank_no as usize * 0x2000 + addr as usize] = val;
|
|
||||||
} else {
|
} else {
|
||||||
println!("Ignoring RTC write");
|
self.rom_bank_no = 1; // 0x00 -> 0x01, 0x20 -> 0x21 etc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
0x4000 ... 0x5FFF => {
|
||||||
|
// Upper ROM bank / RAM bank select
|
||||||
|
self.bank_no_high = val & 3;
|
||||||
|
},
|
||||||
|
0x6000 ... 0x7FFF => {
|
||||||
|
// Select upper ROM bytes or RAM bytes
|
||||||
|
match val {
|
||||||
|
0 => self.bank_mode = BankMode::RomBankMode,
|
||||||
|
1 => self.bank_mode = BankMode::RamBankMode,
|
||||||
|
_ => panic!("Invalid bank mode {:02X}", val),
|
||||||
|
}
|
||||||
|
},
|
||||||
_ => panic!("MBC1: Writing {:02X} to {:04X} not supported", val, addr),
|
_ => panic!("MBC1: Writing {:02X} to {:04X} not supported", val, addr),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user