#![no_std] #![cfg_attr(test, no_main)] #![feature(custom_test_frameworks)] #![feature(abi_x86_interrupt)] #![feature(alloc_error_handler)] #![feature(const_mut_refs)] #![test_runner(crate::test_runner)] #![reexport_test_harness_main = "test_main"] extern crate alloc; use core::panic::PanicInfo; pub mod allocator; pub mod gdt; pub mod interrupts; pub mod memory; pub mod serial; pub mod vga_buffer; pub trait Testable { fn run(&self) -> (); } impl Testable for T where T: Fn(), { fn run(&self) { serial_print!("{}...\t", core::any::type_name::()); // 打印函数名 self(); // 执行这个函数。由于 T 具有 Fn() trait 所以它能够作为一个函数被直接调用 serial_println!("[ok]"); // 打印 "[ok]" } } pub fn test_runner(tests: &[&dyn Testable]) { serial_println!("Running {} tests", tests.len()); for test in tests { test.run(); } exit_qemu(QemuExitCode::Success); } pub fn test_panic_handler(info: &PanicInfo) -> ! { serial_println!("[failed]\n"); serial_println!("Error: {}\n", info); exit_qemu(QemuExitCode::Failed); hlt_loop(); } #[cfg(test)] use bootloader::{entry_point, BootInfo}; #[cfg(test)] entry_point!(test_kernel_main); #[cfg(test)] fn test_kernel_main(_boot_info: &'static BootInfo) -> ! { init(); test_main(); hlt_loop(); } #[cfg(test)] #[panic_handler] fn panic(info: &PanicInfo) -> ! { test_panic_handler(info) } #[test_case] fn test_breakpoint_exception() { x86_64::instructions::interrupts::int3(); // 如果程序继续运行则说明测试成功 // 用 cargo test 运行全部测试 // 或 cargo test --lib 仅运行库测试 } #[derive(Debug, Clone, Copy, PartialEq, Eq)] #[repr(u32)] pub enum QemuExitCode { Success = 0x10, Failed = 0x11, } pub fn exit_qemu(exit_code: QemuExitCode) { use x86_64::instructions::port::Port; unsafe { let mut port = Port::new(0xf4); // iobase port port.write(exit_code as u32); } } pub fn hlt_loop() -> ! { loop { x86_64::instructions::hlt(); } } #[alloc_error_handler] fn alloc_error_handler(layout: alloc::alloc::Layout) -> ! { panic!("allocation error: {:?}", layout); } pub fn init() { gdt::init(); interrupts::init_idt(); unsafe { interrupts::PICS.lock().initialize() }; x86_64::instructions::interrupts::enable(); }