2016-08-14 02:21:46 +02:00
|
|
|
/// This function is where the kernel sets up IRQ handlers
|
|
|
|
/// It is increcibly unsafe, and should be minimal in nature
|
|
|
|
/// It must create the IDT with the correct entries, those entries are
|
|
|
|
/// defined in other files inside of the `arch` module
|
2016-08-14 02:58:31 +02:00
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
use externs::memset;
|
|
|
|
use idt;
|
|
|
|
use memory::{self, Frame};
|
|
|
|
use paging::{self, entry, PhysicalAddress};
|
|
|
|
|
|
|
|
/// Test of zero values in BSS.
|
|
|
|
static BSS_TEST_ZERO: usize = 0;
|
|
|
|
/// Test of non-zero values in BSS.
|
|
|
|
static BSS_TEST_NONZERO: usize = 0xFFFFFFFFFFFFFFFF;
|
|
|
|
|
|
|
|
extern {
|
|
|
|
/// Kernel main function
|
|
|
|
fn kmain() -> !;
|
|
|
|
}
|
2016-08-14 17:31:35 +02:00
|
|
|
|
|
|
|
extern {
|
|
|
|
/// The starting byte of the text (code) data segment.
|
|
|
|
static mut __text_start: u8;
|
|
|
|
/// The ending byte of the text (code) data segment.
|
|
|
|
static mut __text_end: u8;
|
|
|
|
/// The starting byte of the _.rodata_ (read-only data) segment.
|
|
|
|
static mut __rodata_start: u8;
|
|
|
|
/// The ending byte of the _.rodata_ (read-only data) segment.
|
|
|
|
static mut __rodata_end: u8;
|
|
|
|
/// The starting byte of the _.data_ segment.
|
|
|
|
static mut __data_start: u8;
|
|
|
|
/// The ending byte of the _.data_ segment.
|
|
|
|
static mut __data_end: u8;
|
|
|
|
/// The starting byte of the _.bss_ (uninitialized data) segment.
|
|
|
|
static mut __bss_start: u8;
|
|
|
|
/// The ending byte of the _.bss_ (uninitialized data) segment.
|
|
|
|
static mut __bss_end: u8;
|
|
|
|
}
|
|
|
|
|
2016-08-14 02:21:46 +02:00
|
|
|
#[no_mangle]
|
2016-08-14 19:45:47 +02:00
|
|
|
pub unsafe extern fn kstart() -> ! {
|
2016-08-14 02:21:46 +02:00
|
|
|
asm!("xchg bx, bx" : : : : "intel", "volatile");
|
|
|
|
|
2016-08-14 17:31:35 +02:00
|
|
|
// Zero BSS, this initializes statics that are set to 0
|
|
|
|
{
|
|
|
|
let start_ptr = &mut __bss_start as *mut u8;
|
|
|
|
let end_ptr = & __bss_end as *const u8 as usize;
|
|
|
|
|
|
|
|
if start_ptr as usize <= end_ptr {
|
|
|
|
let size = end_ptr - start_ptr as usize;
|
|
|
|
memset(start_ptr, 0, size);
|
|
|
|
}
|
|
|
|
|
2016-08-14 19:45:47 +02:00
|
|
|
debug_assert_eq!(BSS_TEST_ZERO, 0);
|
|
|
|
debug_assert_eq!(BSS_TEST_NONZERO, 0xFFFFFFFFFFFFFFFF);
|
2016-08-14 17:31:35 +02:00
|
|
|
}
|
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
// Set up IDT
|
|
|
|
idt::init(blank);
|
2016-08-14 17:31:35 +02:00
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
// Initialize memory management
|
|
|
|
let mut allocator = memory::init(0, &__bss_end as *const u8 as usize);
|
2016-08-14 02:58:31 +02:00
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
// Initialize paging
|
|
|
|
let mut pager = paging::init();
|
2016-08-14 02:58:31 +02:00
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
// Remap a section with `flags`
|
|
|
|
let mut remap_section = |start_ref: &u8, end_ref: &u8, flags: entry::EntryFlags| {
|
|
|
|
let start = start_ref as *const _ as usize;
|
|
|
|
let end = end_ref as *const _ as usize;
|
2016-08-14 02:58:31 +02:00
|
|
|
|
2016-08-15 19:29:53 +02:00
|
|
|
for i in 0..(start - end + paging::PAGE_SIZE - 1)/paging::PAGE_SIZE {
|
|
|
|
let frame = Frame::containing_address(PhysicalAddress::new(start + i * paging::PAGE_SIZE));
|
|
|
|
pager.identity_map(frame, flags, &mut allocator);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Remap text read-only
|
|
|
|
{
|
|
|
|
asm!("xchg bx, bx" : : : : "intel", "volatile");
|
|
|
|
//TODO remap_section(& __text_start, & __text_end, entry::PRESENT);
|
|
|
|
}
|
2016-08-14 17:31:35 +02:00
|
|
|
|
2016-08-14 19:45:47 +02:00
|
|
|
kmain();
|
2016-08-14 02:58:31 +02:00
|
|
|
}
|
|
|
|
|
2016-08-14 20:08:42 +02:00
|
|
|
interrupt!(blank, {
|
|
|
|
println!("INTERRUPT");
|
|
|
|
});
|