fox32/src/bus.rs
Ry 342c3e6061 fox32+fox32rom: Remove the concept of "fast" and "shared" memory
Instead of having two banks of RAM, just have one bank. This requires
the use of some cursed code, but it's fine :P

Co-authored-by: Lua <lua@foxgirl.dev>
2022-03-11 17:13:18 -08:00

205 lines
9.1 KiB
Rust

// bus.rs
use crate::{DiskController, Memory, Mouse, Overlay};
use std::io::{stdout, Write};
use std::sync::{Arc, Mutex};
pub struct Bus {
pub disk_controller: DiskController,
pub memory: Memory,
pub mouse: Arc<Mutex<Mouse>>,
pub overlays: Arc<Mutex<Vec<Overlay>>>,
}
impl Bus {
pub fn read_io(&mut self, port: u32) -> u32 {
match port {
0x80000000..=0x8000031F => { // overlay port
let overlay_lock = self.overlays.lock().unwrap();
let overlay_number = (port & 0x000000FF) as usize;
let setting = (port & 0x0000FF00) >> 8;
match setting {
0x00 => {
// we're reading the position of this overlay
let x = overlay_lock[overlay_number].x as u32;
let y = overlay_lock[overlay_number].y as u32;
(y << 16) | x
}
0x01 => {
// we're reading the size of this overlay
let width = overlay_lock[overlay_number].width as u32;
let height = overlay_lock[overlay_number].height as u32;
(height << 16) | width
}
0x02 => {
// we're reading the framebuffer pointer of this overlay
overlay_lock[overlay_number].framebuffer_pointer
}
0x03 => {
// we're reading the enable status of this overlay
overlay_lock[overlay_number].enabled as u32
}
_ => panic!("invalid overlay control port"),
}
}
0x80000400..=0x80000401 => { // mouse port
let setting = (port & 0x000000FF) as u8;
match setting {
0x00 => {
// we're reading the button states
let mut byte: u8 = 0x00;
let mut mouse_lock = self.mouse.lock().expect("failed to lock the mouse mutex");
if mouse_lock.click {
byte |= 0b01;
mouse_lock.click = false;
}
if mouse_lock.held {
byte |= 0b10;
} else {
byte &= !0b10;
}
byte as u32
}
0x01 => {
// we're reading the position
let mouse_lock = self.mouse.lock().expect("failed to lock the mouse mutex");
let x = mouse_lock.x as u32;
let y = mouse_lock.y as u32;
(y << 16) | x
}
_ => panic!("invalid mouse control port"),
}
}
0x80001000..=0x80002200 => { // disk controller port
let address_or_id = (port & 0x00000FFF) as usize;
let operation = (port & 0x0000F000) >> 8;
match operation {
0x10 => {
// we're reading the current insert state of the specified disk id
if address_or_id > 3 {
panic!("invalid disk ID");
}
match &self.disk_controller.disk[address_or_id] {
Some(disk) => disk.size as u32, // return size if this disk is inserted
None => 0, // return 0 if this disk is not inserted
}
}
0x20 => {
// we're reading from the sector buffer
if address_or_id > 512 {
panic!("attempted to read past the end of the disk controller sector buffer");
}
self.disk_controller.sector_buffer[address_or_id] as u32
}
_ => panic!("invalid disk controller port"),
}
}
_ => 0,
}
}
pub fn write_io(&mut self, port: u32, word: u32) {
match port {
0x00000000 => { // terminal output port
print!("{}", word as u8 as char);
stdout().flush().expect("could not flush stdout");
}
0x80000000..=0x8000031F => { // overlay port
let mut overlay_lock = self.overlays.lock().unwrap();
let overlay_number = (port & 0x000000FF) as usize;
let setting = (port & 0x0000FF00) >> 8;
match setting {
0x00 => {
// we're setting the position of this overlay
let x = (word & 0x0000FFFF) as usize;
let y = ((word & 0xFFFF0000) >> 16) as usize;
overlay_lock[overlay_number].x = x;
overlay_lock[overlay_number].y = y;
}
0x01 => {
// we're setting the size of this overlay
let width = (word & 0x0000FFFF) as usize;
let height = ((word & 0xFFFF0000) >> 16) as usize;
overlay_lock[overlay_number].width = width;
overlay_lock[overlay_number].height = height;
}
0x02 => {
// we're setting the framebuffer pointer of this overlay
overlay_lock[overlay_number].framebuffer_pointer = word;
}
0x03 => {
// we're setting the enable status of this overlay
overlay_lock[overlay_number].enabled = word != 0;
}
_ => panic!("invalid overlay control port"),
}
}
0x80000400..=0x80000401 => { // mouse port
let setting = (port & 0x000000FF) as u8;
match setting {
0x00 => {
// we're setting the button states
let mut mouse_lock = self.mouse.lock().expect("failed to lock the mouse mutex");
mouse_lock.click = word & (1 << 0) != 0;
mouse_lock.held = word & (1 << 1) != 0;
}
0x01 => {
// we're setting the position
let mut mouse_lock = self.mouse.lock().expect("failed to lock the mouse mutex");
let x = (word & 0x0000FFFF) as u16;
let y = ((word & 0xFFFF0000) >> 16) as u16;
mouse_lock.x = x;
mouse_lock.y = y;
}
_ => panic!("invalid mouse control port"),
}
}
0x80001000..=0x80004200 => { // disk controller port
let address_or_id = (port & 0x00000FFF) as usize;
let operation = (port & 0x0000F000) >> 8;
match operation {
0x10 => {
// we're requesting a disk to be inserted to the specified disk id
if address_or_id > 3 {
panic!("invalid disk ID");
}
let file = self.disk_controller.select_file();
match file {
Some(file) => self.disk_controller.insert(file, address_or_id as u8),
None => {},
};
}
0x20 => {
// we're writing to the sector buffer
if address_or_id > 512 {
panic!("attempted to read past the end of the disk controller sector buffer");
}
self.disk_controller.sector_buffer[address_or_id] = word as u8;
}
0x30 => {
// we're reading the specified sector of the specified disk id into the sector buffer
if address_or_id > 3 {
panic!("invalid disk ID");
}
self.disk_controller.set_current_sector(address_or_id as u8, word);
self.disk_controller.read_into_buffer(address_or_id as u8);
}
0x40 => {
// we're writing the specified sector to the specified disk id from the sector buffer
if address_or_id > 3 {
panic!("invalid disk ID");
}
self.disk_controller.set_current_sector(address_or_id as u8, word);
self.disk_controller.write_from_buffer(address_or_id as u8);
}
_ => panic!("invalid disk controller port"),
}
}
_ => return,
}
}
}