Files
nel_os/src/vmm/io.rs
Masato Imai 95bd14b149 wip: rootfs
2025-06-24 11:46:37 +00:00

142 lines
3.9 KiB
Rust

use crate::info;
use crate::vmm::{qual::QualIo, vcpu::VCpu};
#[derive(Default)]
pub struct Serial {
pub ier: u8,
pub mcr: u8,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum InitPhase {
Uninitialized,
Phase1,
Phase2,
Phase3,
Initialized,
}
pub struct PIC {
pub primary_mask: u8,
pub secondary_mask: u8,
pub primary_phase: InitPhase,
pub secondary_phase: InitPhase,
pub primary_base: u8,
pub secondary_base: u8,
}
impl PIC {
pub fn new() -> Self {
Self {
primary_mask: 0xFF,
secondary_mask: 0xFF,
primary_phase: InitPhase::Uninitialized,
secondary_phase: InitPhase::Uninitialized,
primary_base: 0,
secondary_base: 0,
}
}
}
pub fn handle_io(vcpu: &mut VCpu, qual: QualIo) {
match qual.direction() {
0 => {
handle_io_out(vcpu, qual);
}
1 => {
handle_io_in(vcpu, qual);
}
_ => {}
}
}
pub fn handle_io_in(vcpu: &mut VCpu, qual: QualIo) {
let regs = &mut vcpu.guest_registers;
match qual.port() {
0x0CF8..=0x0CFF => regs.rax = 0,
0xC000..=0xCFFF => {} //ignore
0x20..=0x21 => handle_pic_in(vcpu, qual),
0xA0..=0xA1 => handle_pic_in(vcpu, qual),
0x0070..=0x0071 => regs.rax = 0,
_ => regs.rax = 0,
}
}
pub fn handle_io_out(vcpu: &mut VCpu, qual: QualIo) {
match qual.port() {
0x0CF8..=0x0CFF => {} //ignore
0xC000..=0xCFFF => {} //ignore
0x20..=0x21 => handle_pic_out(vcpu, qual),
0xA0..=0xA1 => handle_pic_out(vcpu, qual),
0x0070..=0x0071 => {} //ignore
_ => {}
}
}
pub fn handle_pic_in(vcpu: &mut VCpu, qual: QualIo) {
let regs = &mut vcpu.guest_registers;
match qual.port() {
0x21 => match vcpu.pic.primary_phase {
InitPhase::Uninitialized | InitPhase::Initialized => {
regs.rax = vcpu.pic.primary_mask as u64;
}
_ => {}
},
0xA1 => match vcpu.pic.secondary_phase {
InitPhase::Uninitialized | InitPhase::Initialized => {
regs.rax = vcpu.pic.secondary_mask as u64;
}
_ => {}
},
_ => {}
}
}
pub fn handle_pic_out(vcpu: &mut VCpu, qual: QualIo) {
let regs = &mut vcpu.guest_registers;
let pic = &mut vcpu.pic;
let dx = regs.rax as u8;
match qual.port() {
0x20 => match dx {
0x11 => pic.primary_phase = InitPhase::Phase1,
0x60..=0x67 => {}
_ => panic!("Primary PIC command: {:#x}", dx),
},
0x21 => match pic.primary_phase {
InitPhase::Uninitialized | InitPhase::Initialized => pic.primary_mask = dx,
InitPhase::Phase1 => {
pic.primary_base = dx;
pic.primary_phase = InitPhase::Phase2;
}
InitPhase::Phase2 => {
pic.primary_phase = InitPhase::Phase3;
}
InitPhase::Phase3 => {
info!("Primary PIC Initialized");
pic.primary_phase = InitPhase::Initialized
}
},
0xA0 => match dx {
0x11 => pic.secondary_phase = InitPhase::Phase1,
0x60..=0x67 => {}
_ => panic!("Secondary PIC command: {:#x}", dx),
},
0xA1 => match pic.secondary_phase {
InitPhase::Uninitialized | InitPhase::Initialized => pic.secondary_mask = dx,
InitPhase::Phase1 => {
pic.secondary_base = dx;
pic.secondary_phase = InitPhase::Phase2;
}
InitPhase::Phase2 => {
pic.secondary_phase = InitPhase::Phase3;
}
InitPhase::Phase3 => {
info!("Secondary PIC Initialized");
pic.secondary_phase = InitPhase::Initialized
}
},
_ => {}
}
}