use std::{
cell::{Cell, RefCell},
rc::Rc,
sync::Arc,
};
use crate::{
cpu::mos6502::{Mos6502, Mos6502Variant},
cpu::Cpu,
keyboard::{
commodore::{C64KeyboardAdapter, C64SymbolAdapter, C64VirtualAdapter},
KeyAdapter, KeyMappingStrategy, SymbolAdapter,
},
memory::{
mos652x::Cia, BankedMemory, BlockMemory, BranchMemory, Mos6510Port, NullMemory, NullPort, Port,
},
platform::{PlatformProvider, WindowConfig},
systems::System,
};
mod keyboard;
mod roms;
mod vic_ii;
use instant::Duration;
pub use roms::C64SystemRoms;
use self::{
keyboard::KEYBOARD_MAPPING,
vic_ii::{VicIIChip, VicIIChipIO},
};
use super::BuildableSystem;
struct C64Cia1PortA {
keyboard_row: Rc<Cell<u8>>,
}
impl C64Cia1PortA {
pub fn new() -> Self {
Self {
keyboard_row: Rc::new(Cell::new(0)),
}
}
pub fn get_keyboard_row(&self) -> Rc<Cell<u8>> {
self.keyboard_row.clone()
}
}
impl Port for C64Cia1PortA {
fn read(&mut self) -> u8 {
self.keyboard_row.get()
}
fn write(&mut self, value: u8) {
self.keyboard_row.set(value);
}
fn poll(&mut self, _cycles_since_poll: u64, _total_cycle_count: u64) -> bool {
false
}
fn reset(&mut self) {}
}
struct C64Cia1PortB {
keyboard_row: Rc<Cell<u8>>,
mapping_strategy: KeyMappingStrategy,
platform: Arc<dyn PlatformProvider>,
}
impl C64Cia1PortB {
pub fn new(
keyboard_row: Rc<Cell<u8>>,
mapping_strategy: KeyMappingStrategy,
platform: Arc<dyn PlatformProvider>,
) -> Self {
Self {
keyboard_row,
mapping_strategy,
platform,
}
}
}
impl Port for C64Cia1PortB {
fn read(&mut self) -> u8 {
let row_mask = self.keyboard_row.get();
let mut value = 0b1111_1111;
let state = match &self.mapping_strategy {
KeyMappingStrategy::Physical => C64KeyboardAdapter::map(&self.platform.get_key_state()),
KeyMappingStrategy::Symbolic => {
C64SymbolAdapter::map(&SymbolAdapter::map(&self.platform.get_key_state()))
}
};
let state = state | C64VirtualAdapter::map(&self.platform.get_virtual_key_state());
for (y, row) in KEYBOARD_MAPPING.iter().enumerate() {
for (x, key) in row.iter().enumerate() {
if ((!row_mask & (1 << y)) != 0) && state.is_pressed(*key) {
value &= !(1 << x);
}
}
}
value
}
fn write(&mut self, _value: u8) {
panic!("Tried to write to keyboard row");
}
fn poll(&mut self, _cycles_since_poll: u64, _total_cycle_count: u64) -> bool {
false
}
fn reset(&mut self) {}
}
pub struct C64BankSwitching {
hiram: bool,
loram: bool,
charen: bool,
selectors: [Rc<Cell<usize>>; 6],
}
impl C64BankSwitching {
pub fn new(mut selectors: [Rc<Cell<usize>>; 6]) -> Self {
selectors.iter_mut().for_each(|s| s.set(0));
Self {
hiram: true,
loram: true,
charen: true,
selectors,
}
}
}
impl Port for C64BankSwitching {
fn read(&mut self) -> u8 {
(self.loram as u8) | (self.hiram as u8) << 1 | (self.charen as u8) << 2
}
#[allow(clippy::bool_to_int_with_if)]
fn write(&mut self, value: u8) {
self.loram = (value & 0b001) != 0;
self.hiram = (value & 0b010) != 0;
self.charen = (value & 0b100) != 0;
self.selectors[0].set(0);
self.selectors[1].set(0);
self.selectors[2].set(if self.hiram && self.loram { 0 } else { 1 });
self.selectors[3].set(0);
self.selectors[4].set(if !self.hiram && !self.loram {
1
} else if !self.charen {
2
} else {
0
});
self.selectors[5].set(if !self.hiram { 1 } else { 0 });
}
fn poll(&mut self, _cycles_since_poll: u64, _total_cycle_count: u64) -> bool {
false
}
fn reset(&mut self) {
self.hiram = true;
self.loram = true;
self.charen = true;
}
}
pub struct C64SystemConfig {
pub mapping: KeyMappingStrategy,
}
impl BuildableSystem<C64SystemRoms, C64SystemConfig> for C64System {
fn build(
roms: C64SystemRoms,
config: C64SystemConfig,
platform: Arc<dyn PlatformProvider>,
) -> Box<dyn System> {
platform.request_window(WindowConfig::new(
vic_ii::FULL_WIDTH,
vic_ii::FULL_HEIGHT,
2.0,
));
let region1 = BlockMemory::ram(0x1000);
let selector2 = Rc::new(Cell::new(0));
let region2 = BankedMemory::new(selector2.clone())
.bank(BlockMemory::ram(0x7000))
.bank(NullMemory::new());
let selector3 = Rc::new(Cell::new(0));
let region3 = BankedMemory::new(selector3.clone())
.bank(BlockMemory::ram(0x2000))
.bank(NullMemory::new()); let selector4 = Rc::new(Cell::new(0));
let region4 = BankedMemory::new(selector4.clone())
.bank(BlockMemory::from_file(0x2000, roms.basic))
.bank(BlockMemory::ram(0x2000))
.bank(NullMemory::new()) .bank(NullMemory::new());
let selector5 = Rc::new(Cell::new(0));
let region5 = BankedMemory::new(selector5.clone())
.bank(BlockMemory::ram(0x1000))
.bank(NullMemory::new());
let selector6 = Rc::new(Cell::new(0));
let character_rom = BlockMemory::from_file(0x1000, roms.character.clone());
let vic_ii = Rc::new(RefCell::new(VicIIChip::new(Box::new(character_rom))));
let vic_io = VicIIChipIO::new(vic_ii.clone()); let port_a = C64Cia1PortA::new();
let keyboard_col = port_a.get_keyboard_row();
let cia_1 = Cia::new(
Box::new(port_a),
Box::new(C64Cia1PortB::new(
keyboard_col,
config.mapping,
platform.clone(),
)),
);
let cia_2 = Cia::new(Box::new(NullPort::new()), Box::new(NullPort::new()));
let region6 = BankedMemory::new(selector6.clone())
.bank(
BranchMemory::new()
.map(0x000, vic_io)
.map(0x400, NullMemory::new()) .map(0x800, BlockMemory::ram(0x0400))
.map(0xC00, cia_1)
.map(0xD00, cia_2)
.map(0xE00, NullMemory::new()) .map(0xF00, NullMemory::new()), )
.bank(BlockMemory::ram(0x1000))
.bank(BlockMemory::from_file(0x1000, roms.character));
let selector7 = Rc::new(Cell::new(0));
let region7 = BankedMemory::new(selector7.clone())
.bank(BlockMemory::from_file(0x2000, roms.kernal))
.bank(BlockMemory::ram(0x2000))
.bank(NullMemory::new()); let bank_switching = C64BankSwitching::new([
selector2, selector3, selector4, selector5, selector6, selector7,
]);
let memory = BranchMemory::new()
.map(0x0000, Mos6510Port::new(Box::new(bank_switching)))
.map(0x0002, region1)
.map(0x1000, region2)
.map(0x8000, region3)
.map(0xA000, region4)
.map(0xC000, region5)
.map(0xD000, region6)
.map(0xE000, region7);
let cpu = Mos6502::new(memory, Mos6502Variant::NMOS);
Box::new(C64System { cpu, vic: vic_ii })
}
}
pub struct C64System {
cpu: Mos6502,
vic: Rc<RefCell<VicIIChip>>,
}
impl System for C64System {
fn get_cpu_mut(&mut self) -> Box<&mut dyn Cpu> {
Box::new(&mut self.cpu)
}
fn tick(&mut self) -> Duration {
Duration::from_secs_f64(1.0 / 1_000_000.0) * self.cpu.tick() as u32
}
fn reset(&mut self) {
self.cpu.reset();
}
fn render(&mut self, framebuffer: &mut [u8], config: WindowConfig) {
self
.vic
.borrow_mut()
.draw_screen(&mut self.cpu.memory, framebuffer, config)
}
}