//! Architecture support for x86_64 #![deny(unused_must_use)] #![feature(asm)] #![feature(concat_idents)] #![feature(const_fn)] #![feature(core_intrinsics)] #![feature(drop_types_in_const)] #![feature(lang_items)] #![feature(naked_functions)] #![feature(thread_local)] #![feature(unique)] #![no_std] extern crate hole_list_allocator as allocator; #[macro_use] extern crate bitflags; extern crate io; extern crate spin; extern crate syscall; pub extern crate x86; // Because the memory map is so important to not be aliased, it is defined here, in one place // The lower 256 PML4 entries are reserved for userspace // Each PML4 entry references up to 512 GB of memory // The top (511) PML4 is reserved for recursive mapping // The second from the top (510) PML4 is reserved for the kernel /// The size of a single PML4 pub const PML4_SIZE: usize = 0x0000_0080_0000_0000; /// Offset of recursive paging pub const RECURSIVE_PAGE_OFFSET: usize = (-(PML4_SIZE as isize)) as usize; /// Offset of kernel pub const KERNEL_OFFSET: usize = RECURSIVE_PAGE_OFFSET - PML4_SIZE; /// Offset to kernel heap pub const KERNEL_HEAP_OFFSET: usize = KERNEL_OFFSET + PML4_SIZE/2; /// Size of kernel heap pub const KERNEL_HEAP_SIZE: usize = 128 * 1024 * 1024; // 128 MB /// Offset to kernel percpu variables //TODO: Use 64-bit fs offset to enable this pub const KERNEL_PERCPU_OFFSET: usize = KERNEL_HEAP_OFFSET - PML4_SIZE; pub const KERNEL_PERCPU_OFFSET: usize = 0xC000_0000; /// Size of kernel percpu variables pub const KERNEL_PERCPU_SIZE: usize = 64 * 1024; // 64 KB /// Offset to user image pub const USER_OFFSET: usize = 0; /// Offset to user TCB pub const USER_TCB_OFFSET: usize = 0xB000_0000; /// Offset to user arguments pub const USER_ARG_OFFSET: usize = USER_OFFSET + PML4_SIZE/2; /// Offset to user heap pub const USER_HEAP_OFFSET: usize = USER_OFFSET + PML4_SIZE; /// Offset to user grants pub const USER_GRANT_OFFSET: usize = USER_HEAP_OFFSET + PML4_SIZE; /// Offset to user stack pub const USER_STACK_OFFSET: usize = USER_GRANT_OFFSET + PML4_SIZE; /// Size of user stack pub const USER_STACK_SIZE: usize = 1024 * 1024; // 1 MB /// Offset to user TLS pub const USER_TLS_OFFSET: usize = USER_STACK_OFFSET + PML4_SIZE; /// Offset to user temporary image (used when cloning) pub const USER_TMP_OFFSET: usize = USER_TLS_OFFSET + PML4_SIZE; /// Offset to user temporary heap (used when cloning) pub const USER_TMP_HEAP_OFFSET: usize = USER_TMP_OFFSET + PML4_SIZE; /// Offset to user temporary page for grants pub const USER_TMP_GRANT_OFFSET: usize = USER_TMP_HEAP_OFFSET + PML4_SIZE; /// Offset to user temporary stack (used when cloning) pub const USER_TMP_STACK_OFFSET: usize = USER_TMP_GRANT_OFFSET + PML4_SIZE; /// Offset to user temporary tls (used when cloning) pub const USER_TMP_TLS_OFFSET: usize = USER_TMP_STACK_OFFSET + PML4_SIZE; /// Offset for usage in other temporary pages pub const USER_TMP_MISC_OFFSET: usize = USER_TMP_TLS_OFFSET + PML4_SIZE; /// Print to console #[macro_export] macro_rules! print { ($($arg:tt)*) => ({ use core::fmt::Write; let _ = write!($crate::console::CONSOLE.lock(), $($arg)*); }); } /// Print with new line to console #[macro_export] macro_rules! println { ($fmt:expr) => (print!(concat!($fmt, "\n"))); ($fmt:expr, $($arg:tt)*) => (print!(concat!($fmt, "\n"), $($arg)*)); } /// Create an interrupt function that can safely run rust code #[macro_export] macro_rules! interrupt { ($name:ident, $func:block) => { #[naked] pub unsafe extern fn $name () { #[inline(never)] unsafe fn inner() { $func } // Push scratch registers asm!("push rax push rcx push rdx push rdi push rsi push r8 push r9 push r10 push r11 push fs mov rax, 0x18 mov fs, ax" : : : : "intel", "volatile"); // Call inner rust function inner(); // Pop scratch registers and return asm!("pop fs pop r11 pop r10 pop r9 pop r8 pop rsi pop rdi pop rdx pop rcx pop rax iretq" : : : : "intel", "volatile"); } }; } #[repr(packed)] pub struct InterruptStack { fs: usize, r11: usize, r10: usize, r9: usize, r8: usize, rsi: usize, rdi: usize, rdx: usize, rcx: usize, rax: usize, rip: usize, cs: usize, rflags: usize, } #[macro_export] macro_rules! interrupt_stack { ($name:ident, $stack: ident, $func:block) => { #[naked] pub unsafe extern fn $name () { #[inline(never)] unsafe fn inner($stack: &$crate::InterruptStack) { $func } // Push scratch registers asm!("push rax push rcx push rdx push rdi push rsi push r8 push r9 push r10 push r11 push fs mov rax, 0x18 mov fs, ax" : : : : "intel", "volatile"); // Get reference to stack variables let rsp: usize; asm!("" : "={rsp}"(rsp) : : : "intel", "volatile"); // Call inner rust function inner(&*(rsp as *const $crate::InterruptStack)); // Pop scratch registers and return asm!("pop fs pop r11 pop r10 pop r9 pop r8 pop rsi pop rdi pop rdx pop rcx pop rax iretq" : : : : "intel", "volatile"); } }; } #[repr(packed)] pub struct InterruptErrorStack { fs: usize, r11: usize, r10: usize, r9: usize, r8: usize, rsi: usize, rdi: usize, rdx: usize, rcx: usize, rax: usize, code: usize, rip: usize, cs: usize, rflags: usize, } #[macro_export] macro_rules! interrupt_error { ($name:ident, $stack:ident, $func:block) => { #[naked] pub unsafe extern fn $name () { #[inline(never)] unsafe fn inner($stack: &$crate::InterruptErrorStack) { $func } // Push scratch registers asm!("push rax push rcx push rdx push rdi push rsi push r8 push r9 push r10 push r11 push fs mov rax, 0x18 mov fs, ax" : : : : "intel", "volatile"); // Get reference to stack variables let rsp: usize; asm!("" : "={rsp}"(rsp) : : : "intel", "volatile"); // Call inner rust function inner(&*(rsp as *const $crate::InterruptErrorStack)); // Pop scratch registers, error code, and return asm!("pop fs pop r11 pop r10 pop r9 pop r8 pop rsi pop rdi pop rdx pop rcx pop rax add rsp, 8 iretq" : : : : "intel", "volatile"); } }; } /// ACPI table parsing pub mod acpi; /// Console handling pub mod console; /// Context switching pub mod context; /// Devices pub mod device; /// Memcpy, memmove, etc. pub mod externs; /// Global descriptor table pub mod gdt; /// Interrupt descriptor table pub mod idt; /// Interrupt instructions pub mod interrupt; /// Memory management pub mod memory; /// Paging pub mod paging; /// Panic pub mod panic; /// Initialization and start function pub mod start; /// Shutdown function pub mod stop; /// Time pub mod time;