vesper/src/main.rs

165 lines
4.6 KiB
Rust

#![no_std]
#![no_main]
#![feature(asm)]
#![feature(global_asm)]
#![feature(const_fn)]
#![feature(format_args_nl)]
#![feature(lang_items)]
#![feature(ptr_internals)] // until we mark with PhantomData instead?
#![feature(core_intrinsics)]
#![feature(try_from)]
#![doc(html_root_url = "https://docs.metta.systems/")]
#![allow(dead_code)]
#![allow(unused_assignments)]
#![allow(unused_must_use)]
//any(target_arch = "aarch64", target_arch = "x86_64")
#[cfg(not(target_arch = "aarch64"))]
use architecture_not_supported_sorry;
extern crate bitflags;
#[macro_use]
extern crate register;
extern crate cortex_a;
extern crate rlibc;
#[macro_use]
pub mod arch;
pub use arch::*;
mod devices;
pub mod macros;
pub use macros::*;
pub mod platform;
mod sync;
mod write_to;
// use core::fmt::Write;
use platform::{
display::{Color, Size2d},
gpio::GPIO,
// uart::MiniUart,
vc::VC,
};
// User-facing kernel parts - syscalls and capability invocations.
// pub mod vesper; -- no mod exported, because available through syscall interface
// Actual interfaces to call these syscalls are in vesper-user (similar to libsel4)
// pub mod vesper; -- exported from vesper-user
/// The global console. Output of the print! and println! macros.
static CONSOLE: sync::NullLock<devices::Console> = sync::NullLock::new(devices::Console::new());
// Kernel entry point
// arch crate is responsible for calling this
fn kmain() -> ! {
let gpio = GPIO::new_default();
let uart = platform::MiniUart::new_default();
uart.init(&gpio);
CONSOLE.lock(|c| {
// Moves uart into the global CONSOLE. It is not accessible
// anymore for the remaining parts of kernel_entry().
c.replace_with(uart.into());
});
let uart = platform::PL011Uart::new_default();
let mut mbox = platform::mailbox::Mailbox::new();
match uart.init(&mut mbox, &gpio) {
Ok(_) => {
CONSOLE.lock(|c| {
// Moves uart into the global CONSOLE. It is not accessible
// anymore for the remaining parts of kernel_entry().
c.replace_with(uart.into());
});
}
Err(_) => endless_sleep(),
}
println!("\n[0] UART is live!");
println!("Hello, world!");
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);
}
unsafe {
mmu::init();
}
if let Some(mut display) = VC::init_fb(Size2d { x: 800, y: 600 } /*, &mut uart*/) {
display.clear(Color::black());
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];
// Crashes if uncommenting next line: vvv
// let s = write_to::show(&mut buf, format_args!("Display width {}", display.width));
// So, some rust runtime things are breaking it, why?
// if s.is_err() {
// display.draw_text(50, 150, "Error displaying", Color::red())
// } else {
// display.draw_text(50, 150, s.unwrap(), Color::white());
// }
//==============================================
// Since formatted output doesn't work, lets do some other preparatory steps:
// 1. Initialize MMU
// 2. Set up exception handlers
// Obviously, things should keep working after that...
//==============================================
display.draw_text(150, 50, "RED", Color::red());
display.draw_text(160, 60, "GREEN", Color::green());
display.draw_text(170, 70, "BLUE", Color::blue());
}
// unsafe {
// mmu::init();
// }
// writeln!(uart, "Bye, going to sleep now");
// qemu_aarch64_exit()
endless_sleep()
}
entry!(kmain);
// From https://stackoverflow.com/a/49930361/895245
// @todo specify exit value depending on tests result?
fn qemu_aarch64_exit() -> ! {
unsafe {
asm!("
/* 0x20026 == ADP_Stopped_ApplicationExit */
mov x1, #0x26
movk x1, #2, lsl #16
str x1, [sp,#0]
/* Exit status code. Host QEMU process exits with that status. */
mov x0, #0
str x0, [sp,#8]
/* x1 contains the address of parameter block.
* Any memory address could be used. */
mov x1, sp
/* SYS_EXIT */
mov w0, #0x18
/* Do the semihosting call on A64. */
hlt 0xf000"
:::: "volatile");
}
unreachable!();
}