/* * SPDX-License-Identifier: BlueOak-1.0.0 * Copyright (c) Berkus Decker */ //! Vesper single-address-space nanokernel. //! //! This crate implements the kernel binary proper. #![no_std] #![no_main] #![feature(ptr_internals)] #![feature(format_args_nl)] #![feature(custom_test_frameworks)] #![test_runner(machine::tests::test_runner)] #![reexport_test_harness_main = "test_main"] #![deny(missing_docs)] #![deny(warnings)] #[cfg(not(test))] use core::panic::PanicInfo; #[allow(unused_imports)] use machine::devices::SerialOps; use { cfg_if::cfg_if, machine::{ arch, entry, memory, platform::rpi3::{ display::{Color, DrawError}, mailbox::{channel, Mailbox, MailboxOps}, vc::VC, }, println, CONSOLE, }, }; entry!(kmain); #[cfg(not(test))] #[panic_handler] fn panicked(info: &PanicInfo) -> ! { machine::panic::handler(info) } fn print_mmu_state_and_features() { memory::mmu::print_features(); } fn init_mmu() { unsafe { memory::mmu::init().unwrap(); } println!("[!] MMU initialised"); print_mmu_state_and_features(); } fn init_exception_traps() { extern "C" { static __exception_vectors_start: u64; } unsafe { let exception_vectors_start: u64 = &__exception_vectors_start as *const _ as u64; arch::traps::set_vbar_el1_checked(exception_vectors_start) .expect("Vector table properly aligned!"); } println!("[!] Exception traps set up"); } #[cfg(not(feature = "noserial"))] fn init_uart_serial() { use machine::platform::rpi3::{gpio::GPIO, mini_uart::MiniUart, pl011_uart::PL011Uart}; let gpio = GPIO::default(); let uart = MiniUart::default(); let uart = uart.prepare(&gpio); CONSOLE.lock(|c| { // Move uart into the global CONSOLE. c.replace_with(uart.into()); }); println!("[0] MiniUART is live!"); // Then immediately switch to PL011 (just as an example) let uart = PL011Uart::default(); let mbox = Mailbox::default(); // uart.init() will reconfigure the GPIO, which causes a race against // the MiniUart that is still putting out characters on the physical // line that are already buffered in its TX FIFO. // // To ensure the CPU doesn't rewire the GPIO before the MiniUart has put // its last character, explicitly flush it before rewiring. // // If you switch to an output that happens to not use the same pair of // physical wires (e.g. the Framebuffer), you don't need to do this, // because flush() is anyways called implicitly by replace_with(). This // is just a special case. CONSOLE.lock(|c| c.flush()); match uart.prepare(mbox, &gpio) { Ok(uart) => { CONSOLE.lock(|c| { // Move uart into the global CONSOLE. c.replace_with(uart.into()); }); println!("[0] UART0 is live!"); } Err(_) => println!("[0] Error switching to PL011 UART, continue with MiniUART"), } } /// Kernel entry point. /// `arch` crate is responsible for calling it. #[inline] pub fn kmain() -> ! { #[cfg(feature = "jtag")] machine::arch::jtag::wait_debugger(); init_mmu(); init_exception_traps(); #[cfg(not(feature = "noserial"))] init_uart_serial(); #[cfg(test)] test_main(); command_prompt(); reboot() } //------------------------------------------------------------ // Start a command prompt //------------------------------------------------------------ fn command_prompt() { 'cmd_loop: loop { let mut buf = [0u8; 64]; match CONSOLE.lock(|c| c.command_prompt(&mut buf)) { b"mmu" => init_mmu(), b"feats" => print_mmu_state_and_features(), #[cfg(not(feature = "noserial"))] b"uart" => init_uart_serial(), b"disp" => check_display_init(), b"trap" => check_data_abort_trap(), b"map" => machine::arch::memory::print_layout(), b"led on" => set_led(true), b"led off" => set_led(false), b"help" => print_help(), b"end" => break 'cmd_loop, x => println!("[!] Unknown command {:?}, try 'help'", x), } } } fn print_help() { println!("Supported console commands:"); println!(" mmu - initialize MMU"); println!(" feats - print MMU state and supported features"); #[cfg(not(feature = "noserial"))] println!(" uart - try to reinitialize UART serial"); println!(" disp - try to init VC framebuffer and draw some text"); println!(" trap - trigger and recover from a data abort exception"); println!(" map - show kernel memory layout"); println!(" led [on|off] - change RPi LED status"); println!(" end - leave console and reset board"); } fn set_led(enable: bool) { let mut mbox = Mailbox::default(); let index = mbox.request(); let index = mbox.set_led_on(index, enable); let mbox = mbox.end(index); mbox.call(channel::PropertyTagsArmToVc) .map_err(|e| { println!("Mailbox call returned error {}", e); println!("Mailbox contents: {:?}", mbox); }) .ok(); } fn reboot() -> ! { cfg_if! { if #[cfg(feature = "qemu")] { println!("Bye, shutting down QEMU"); machine::qemu::semihosting::exit_success() } else { use machine::platform::rpi3::power::Power; println!("Bye, going to reset now"); Power::default().reset() } } } fn check_display_init() { display_graphics() .map_err(|e| { println!("Error in display: {}", e); }) .ok(); } fn display_graphics() -> Result<(), DrawError> { if let Ok(mut display) = VC::init_fb(800, 600, 32) { println!("Display created"); display.clear(Color::black()); println!("Display cleared"); display.rect(10, 10, 250, 250, Color::rgb(32, 96, 64)); display.draw_text(50, 50, "Hello there!", Color::rgb(128, 192, 255))?; let mut buf = [0u8; 64]; let s = machine::write_to::show(&mut buf, format_args!("Display width {}", display.width)); if s.is_err() { display.draw_text(50, 150, "Error displaying", Color::red())? } else { display.draw_text(50, 150, s.unwrap(), Color::white())? } display.draw_text(150, 50, "RED", Color::red())?; display.draw_text(160, 60, "GREEN", Color::green())?; display.draw_text(170, 70, "BLUE", Color::blue())?; } Ok(()) } fn check_data_abort_trap() { // Cause an exception by accessing a virtual address for which no // address translations have been set up. // // This line of code accesses the address 3 GiB, but page tables are // only set up for the range [0..1) GiB. let big_addr: u64 = 3 * 1024 * 1024 * 1024; unsafe { core::ptr::read_volatile(big_addr as *mut u64) }; println!("[i] Whoa! We recovered from an exception."); } #[cfg(test)] mod main_tests { use {super::*, core::panic::PanicInfo}; #[panic_handler] fn panicked(info: &PanicInfo) -> ! { machine::panic::handler_for_tests(info) } #[test_case] fn test_data_abort_trap() { check_data_abort_trap() } }