我研究了裸机工作Rust
。我从 VGA 开始:我发现我可以像这样向它写入数据:
fn main(){
buffer: unsafe { &mut *(0xb8000 as *mut Buffer) };
}
struct Buffer {
chars: [[Volatile<ScreenChar>; BUFFER_WIDTH]; BUFFER_HEIGHT]
}
pub struct VgaWriter {
column_position: usize,
buffer: &'static mut Buffer
}
// ...
也就是,其实直接表明:我0xb8000
在地址处有一个VGA缓冲区,我想在那里写信息。它可以工作(好吧,或者至少它不会崩溃)(我正在测试QEMU
,通过 crate 将其构建为引导映像bootloader
)。
现在我走得更远了:我想向串口写一些东西。我发现在 QEMU 中它位于0x3f8
. 现在我想以类似的方式从它和相邻地址中读取一些东西(据我所知,这些是以下端口引脚的输入):
let a = unsafe { &mut *(0x3f8 as *mut [u8; 25]) };
for i in a{
print!("{}", i);
}
这样的代码会导致 QEMU 以每秒数次的速度循环重启映像,甚至不会出现恐慌。我以为我配置了错误,但通过端口uart_16550
工作正常:
use uart_16550::SerialPort;
fn main(){
unsafe{ SerialPort::new(0x3f8).write_str("Hello world"); };
}
往里看,我意识到在深处的某个地方,从地址读取是这样的:
fn main(){
for i in 0..25{
let value: u8;
unsafe {
asm!("in al, dx", out("al") value, in("dx") 0x3f8+i, options(nomem, nostack, preserves_flags));
}
println!("{}", value);
}
}
是的,通过这种方式我可以在屏幕上获得一些信息(在相同的 VGA 模式下)(我还不知道如何使用它,但问题是不同的):
我的问题是:为什么我可以在地址直接读写VGA端口0xb8000
,但我不能在地址读取,0x3f8
一般在其他地址。
而且,更重要的是,为什么我仍然可以这样做,但只能通过asm
命令?
我怎样才能绕过这个限制?还是我在做一些根本错误的事情/