
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>
205 lines
9.1 KiB
Rust
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,
|
|
}
|
|
}
|
|
} |