This commit is contained in:
parent
392572ea80
commit
fe1c702a8f
14 changed files with 218 additions and 55 deletions
30
Cargo.lock
generated
30
Cargo.lock
generated
|
@ -35,6 +35,12 @@ version = "0.1.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "6b2d54853319fd101b8dd81de382bcbf3e03410a64d8928bbee85a3e7dcde483"
|
checksum = "6b2d54853319fd101b8dd81de382bcbf3e03410a64d8928bbee85a3e7dcde483"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "allocator-api2"
|
||||||
|
version = "0.2.21"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.4.0"
|
version = "1.4.0"
|
||||||
|
@ -173,6 +179,12 @@ dependencies = [
|
||||||
"byteorder",
|
"byteorder",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "equivalent"
|
||||||
|
version = "1.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "float-cmp"
|
name = "float-cmp"
|
||||||
version = "0.9.0"
|
version = "0.9.0"
|
||||||
|
@ -182,6 +194,12 @@ dependencies = [
|
||||||
"num-traits",
|
"num-traits",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "foldhash"
|
||||||
|
version = "0.1.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f81ec6369c545a7d40e4589b5597581fa1c441fe1cce96dd1de43159910a36a2"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "funty"
|
name = "funty"
|
||||||
version = "2.0.0"
|
version = "2.0.0"
|
||||||
|
@ -194,6 +212,17 @@ version = "0.3.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b"
|
checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hashbrown"
|
||||||
|
version = "0.15.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289"
|
||||||
|
dependencies = [
|
||||||
|
"allocator-api2",
|
||||||
|
"equivalent",
|
||||||
|
"foldhash",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "itertools"
|
name = "itertools"
|
||||||
version = "0.13.0"
|
version = "0.13.0"
|
||||||
|
@ -213,6 +242,7 @@ dependencies = [
|
||||||
"bitvec",
|
"bitvec",
|
||||||
"buddy_system_allocator",
|
"buddy_system_allocator",
|
||||||
"embedded-graphics",
|
"embedded-graphics",
|
||||||
|
"hashbrown",
|
||||||
"kernel-common",
|
"kernel-common",
|
||||||
"log",
|
"log",
|
||||||
"spin",
|
"spin",
|
||||||
|
|
|
@ -11,6 +11,7 @@ bitfield = "0.17.0"
|
||||||
bitvec = {version = "1.0.1", default-features = false, features = ["alloc", "atomic"]}
|
bitvec = {version = "1.0.1", default-features = false, features = ["alloc", "atomic"]}
|
||||||
buddy_system_allocator = "0.11.0"
|
buddy_system_allocator = "0.11.0"
|
||||||
embedded-graphics = "0.8.1"
|
embedded-graphics = "0.8.1"
|
||||||
|
hashbrown = "0.15.2"
|
||||||
kernel-common = {path = "../lib/kernel-common"}
|
kernel-common = {path = "../lib/kernel-common"}
|
||||||
log = "0.4.22"
|
log = "0.4.22"
|
||||||
spin = "0.9.8"
|
spin = "0.9.8"
|
||||||
|
|
|
@ -11,6 +11,7 @@ _start:
|
||||||
call rdrand
|
call rdrand
|
||||||
mov %rax, __stack_chk_guard
|
mov %rax, __stack_chk_guard
|
||||||
call early_main
|
call early_main
|
||||||
|
|
||||||
.section .bss
|
.section .bss
|
||||||
.align 8
|
.align 8
|
||||||
.global __stack_chk_guard
|
.global __stack_chk_guard
|
||||||
|
|
|
@ -2,3 +2,4 @@ pub mod gdt;
|
||||||
pub mod idt;
|
pub mod idt;
|
||||||
pub mod isr;
|
pub mod isr;
|
||||||
pub mod paging;
|
pub mod paging;
|
||||||
|
pub mod smp;
|
||||||
|
|
|
@ -25,7 +25,7 @@ extern "C" {
|
||||||
}
|
}
|
||||||
|
|
||||||
static PAGING_ACTIVE: AtomicBool = AtomicBool::new(false);
|
static PAGING_ACTIVE: AtomicBool = AtomicBool::new(false);
|
||||||
static CURRENT_PML4: Mutex<Option<&mut PageTable>> = Mutex::new(None);
|
pub static CURRENT_PML4: Mutex<Option<&mut PageTable>> = Mutex::new(None);
|
||||||
static HEAP_PHYS_START: AtomicU64 = AtomicU64::new(0);
|
static HEAP_PHYS_START: AtomicU64 = AtomicU64::new(0);
|
||||||
static PHYSICAL_FRAMES: Mutex<Option<BitVec<u64>>> = Mutex::new(None);
|
static PHYSICAL_FRAMES: Mutex<Option<BitVec<u64>>> = Mutex::new(None);
|
||||||
static HEAP_PHYS_MAPPING: Mutex<Vec<u64>> = Mutex::new(Vec::new());
|
static HEAP_PHYS_MAPPING: Mutex<Vec<u64>> = Mutex::new(Vec::new());
|
||||||
|
@ -42,11 +42,12 @@ fn _get_free_frame() -> u64 {
|
||||||
panic!("No free memory left");
|
panic!("No free memory left");
|
||||||
}
|
}
|
||||||
fn invlpg(addr: u64) {
|
fn invlpg(addr: u64) {
|
||||||
|
// TODO: Broadcast to all cores
|
||||||
unsafe {
|
unsafe {
|
||||||
asm!("invlpg [{}]", in(reg) addr);
|
asm!("invlpg [{}]", in(reg) addr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn virt_to_phys(virt: u64) -> u64 {
|
pub fn virt_to_phys(virt: u64) -> u64 {
|
||||||
if !PAGING_ACTIVE.load(Ordering::SeqCst) {
|
if !PAGING_ACTIVE.load(Ordering::SeqCst) {
|
||||||
return virt - KERNEL_HEAP_START + HEAP_PHYS_START.load(Ordering::SeqCst);
|
return virt - KERNEL_HEAP_START + HEAP_PHYS_START.load(Ordering::SeqCst);
|
||||||
}
|
}
|
||||||
|
@ -113,7 +114,7 @@ pub fn find_free_virt_range(mut start: u64, end: u64, size: u64) -> u64 {
|
||||||
}
|
}
|
||||||
fn map(pml4: &mut PageTable, virt: u64, phys: u64, user: bool, write: bool, exec: bool, cache_disable: bool) {
|
fn map(pml4: &mut PageTable, virt: u64, phys: u64, user: bool, write: bool, exec: bool, cache_disable: bool) {
|
||||||
assert!(virt >= 0x1000, "First page shouldn't be mapped");
|
assert!(virt >= 0x1000, "First page shouldn't be mapped");
|
||||||
assert!(!write || !exec);
|
assert!(!write || !exec || virt == 0x1000);
|
||||||
{
|
{
|
||||||
let mut frames_vec = PHYSICAL_FRAMES.lock();
|
let mut frames_vec = PHYSICAL_FRAMES.lock();
|
||||||
let frame = phys as usize / 0x1000;
|
let frame = phys as usize / 0x1000;
|
||||||
|
@ -129,12 +130,16 @@ fn map(pml4: &mut PageTable, virt: u64, phys: u64, user: bool, write: bool, exec
|
||||||
let pdpt = get_table_entry(pml4, pml4_i);
|
let pdpt = get_table_entry(pml4, pml4_i);
|
||||||
let directory = get_table_entry(pdpt, pdpt_i);
|
let directory = get_table_entry(pdpt, pdpt_i);
|
||||||
let table = get_table_entry(directory, directory_i);
|
let table = get_table_entry(directory, directory_i);
|
||||||
|
let should_invalidate = table.entries_phys[table_i].present() == 1;
|
||||||
table.entries_phys[table_i].set_address(phys / 0x1000);
|
table.entries_phys[table_i].set_address(phys / 0x1000);
|
||||||
table.entries_phys[table_i].set_user(user as u64);
|
table.entries_phys[table_i].set_user(user as u64);
|
||||||
table.entries_phys[table_i].set_write(write as u64);
|
table.entries_phys[table_i].set_write(write as u64);
|
||||||
table.entries_phys[table_i].set_execute_disable(!exec as u64);
|
table.entries_phys[table_i].set_execute_disable(!exec as u64);
|
||||||
table.entries_phys[table_i].set_cache_disable(cache_disable as u64);
|
table.entries_phys[table_i].set_cache_disable(cache_disable as u64);
|
||||||
table.entries_phys[table_i].set_present(1);
|
table.entries_phys[table_i].set_present(1);
|
||||||
|
if should_invalidate {
|
||||||
|
invlpg(virt);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
pub unsafe fn unmap(address: u64) {
|
pub unsafe fn unmap(address: u64) {
|
||||||
let mut current_pml4 = CURRENT_PML4.lock();
|
let mut current_pml4 = CURRENT_PML4.lock();
|
||||||
|
|
23
kernel/src/cpu/smp.rs
Normal file
23
kernel/src/cpu/smp.rs
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
use kernel_common::instructions::sti;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
cpu::{gdt::setup_gdt, idt::setup_idt},
|
||||||
|
sys::{
|
||||||
|
lapic::{setup_lapic, setup_lapic_timer},
|
||||||
|
scheduler::yield_task,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
extern "C" fn ap_main() -> ! {
|
||||||
|
setup_gdt();
|
||||||
|
setup_idt();
|
||||||
|
setup_lapic(0);
|
||||||
|
// TODO: Also calibrate other cores
|
||||||
|
setup_lapic_timer(false);
|
||||||
|
unsafe {
|
||||||
|
sti();
|
||||||
|
}
|
||||||
|
yield_task();
|
||||||
|
panic!("Yielding to idle task failed");
|
||||||
|
}
|
|
@ -1,5 +1,46 @@
|
||||||
.text
|
.section .rodata
|
||||||
|
.code16
|
||||||
|
.align 0x1000
|
||||||
|
|
||||||
.global ap_trampoline
|
|
||||||
ap_trampoline:
|
ap_trampoline:
|
||||||
hlt
|
mov %cr4, %eax
|
||||||
|
bts $5, %eax // Physical Address Extension
|
||||||
|
mov %eax, %cr4
|
||||||
|
mov trampoline_pml4 - ap_trampoline + 0x1000, %eax
|
||||||
|
mov %eax, %cr3
|
||||||
|
mov $0xc0000080, %ecx
|
||||||
|
rdmsr
|
||||||
|
bts $8, %eax // Long Mode Enable
|
||||||
|
bts $11, %eax // No Execute Enable
|
||||||
|
wrmsr
|
||||||
|
mov %cr0, %eax
|
||||||
|
bts $0, %eax // Protected Mode Enable
|
||||||
|
bts $16, %eax // Write Protect
|
||||||
|
bts $31, %eax // Paging Enable
|
||||||
|
mov %eax, %cr0
|
||||||
|
lgdt gdt_descriptor - ap_trampoline + 0x1000
|
||||||
|
ljmp $8, $ap_trampoline_2 - ap_trampoline + 0x1000
|
||||||
|
|
||||||
|
.code64
|
||||||
|
|
||||||
|
ap_trampoline_2:
|
||||||
|
mov trampoline_stack - ap_trampoline + 0x1000, %rsp
|
||||||
|
mov $ap_main, %rax
|
||||||
|
call *%rax
|
||||||
|
|
||||||
|
.align 4
|
||||||
|
trampoline_pml4:
|
||||||
|
.int 0
|
||||||
|
|
||||||
|
.align 8
|
||||||
|
trampoline_stack:
|
||||||
|
.quad 0
|
||||||
|
|
||||||
|
gdt:
|
||||||
|
.quad 0
|
||||||
|
.quad 0x209a0000000000
|
||||||
|
.quad 0x920000000000
|
||||||
|
|
||||||
|
gdt_descriptor:
|
||||||
|
.short . - gdt - 1
|
||||||
|
.quad gdt - ap_trampoline + 0x1000
|
||||||
|
|
|
@ -85,11 +85,11 @@ extern "C" fn early_main(temp_loader_struct: *const LoaderStruct) -> ! {
|
||||||
}
|
}
|
||||||
parse_madt(&early_acpi_tables);
|
parse_madt(&early_acpi_tables);
|
||||||
setup_hpet(&early_acpi_tables);
|
setup_hpet(&early_acpi_tables);
|
||||||
setup_lapic_timer();
|
setup_lapic_timer(true);
|
||||||
setup_multitasking();
|
setup_multitasking();
|
||||||
}
|
}
|
||||||
fn main() {
|
fn main() {
|
||||||
info!("Starting main kernel task");
|
info!("Starting main kernel task...");
|
||||||
let mut status = unsafe { AcpiInitializeSubsystem() };
|
let mut status = unsafe { AcpiInitializeSubsystem() };
|
||||||
assert_eq!(status, AE_OK);
|
assert_eq!(status, AE_OK);
|
||||||
status = unsafe { AcpiInitializeTables(null_mut(), 0, 0) };
|
status = unsafe { AcpiInitializeTables(null_mut(), 0, 0) };
|
||||||
|
|
|
@ -25,8 +25,9 @@ static SCI_CONTEXT: AtomicPtr<c_void> = AtomicPtr::new(null_mut());
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
hpet::get_current_time,
|
hpet::get_current_time,
|
||||||
|
lapic::get_current_lapic_id,
|
||||||
sync::{create_semaphore, lock_semaphore, unlock_semaphore, Semaphore, Spinlock},
|
sync::{create_semaphore, lock_semaphore, unlock_semaphore, Semaphore, Spinlock},
|
||||||
task::{CURRENT_TASK, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
task::{CURRENT_TASKS, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub const AE_OK: ACPI_STATUS = 0;
|
pub const AE_OK: ACPI_STATUS = 0;
|
||||||
|
@ -95,7 +96,7 @@ extern "C" fn AcpiOsGetThreadId() -> UINT64 {
|
||||||
let task_id;
|
let task_id;
|
||||||
CURRENT_TASK_LOCK.lock();
|
CURRENT_TASK_LOCK.lock();
|
||||||
{
|
{
|
||||||
task_id = CURRENT_TASK.lock().as_ref().unwrap().id;
|
task_id = CURRENT_TASKS.lock().get(&get_current_lapic_id()).unwrap().id;
|
||||||
}
|
}
|
||||||
CURRENT_TASK_LOCK.unlock();
|
CURRENT_TASK_LOCK.unlock();
|
||||||
task_id as UINT64
|
task_id as UINT64
|
||||||
|
|
|
@ -13,9 +13,10 @@ use crate::cpu::paging::map_physical;
|
||||||
use super::{
|
use super::{
|
||||||
early_acpi::EarlyACPIHandler,
|
early_acpi::EarlyACPIHandler,
|
||||||
ioapic::register_irq_handler,
|
ioapic::register_irq_handler,
|
||||||
|
lapic::get_current_lapic_id,
|
||||||
scheduler::{schedule_task, yield_task, SCHEDULER_LOCK},
|
scheduler::{schedule_task, yield_task, SCHEDULER_LOCK},
|
||||||
sync::Spinlock,
|
sync::Spinlock,
|
||||||
task::{Task, TaskState, CURRENT_TASK, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
task::{Task, TaskState, CURRENT_TASKS, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
||||||
};
|
};
|
||||||
|
|
||||||
const REGISTER_CAPABILITIES: usize = 0;
|
const REGISTER_CAPABILITIES: usize = 0;
|
||||||
|
@ -98,8 +99,8 @@ pub fn sleep(us: usize) {
|
||||||
if MULTITASKING_ENABLED.load(Ordering::SeqCst) {
|
if MULTITASKING_ENABLED.load(Ordering::SeqCst) {
|
||||||
CURRENT_TASK_LOCK.lock();
|
CURRENT_TASK_LOCK.lock();
|
||||||
{
|
{
|
||||||
let mut _current_task = CURRENT_TASK.lock();
|
let mut _current_task = CURRENT_TASKS.lock();
|
||||||
let current_task = _current_task.as_mut().unwrap();
|
let current_task = _current_task.get_mut(&get_current_lapic_id()).unwrap();
|
||||||
current_task.sleep_until_us = get_current_time() + us;
|
current_task.sleep_until_us = get_current_time() + us;
|
||||||
current_task.task_state = TaskState::Sleeping;
|
current_task.task_state = TaskState::Sleeping;
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,12 +3,20 @@ use core::{
|
||||||
sync::atomic::{AtomicBool, AtomicPtr, AtomicUsize, Ordering},
|
sync::atomic::{AtomicBool, AtomicPtr, AtomicUsize, Ordering},
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::cpu::paging::{map_physical, map_range, unmap};
|
use alloc::{vec, vec::Vec};
|
||||||
|
use kernel_common::{instructions::pause, paging::PageTable};
|
||||||
|
|
||||||
use super::hpet::sleep;
|
use crate::cpu::paging::{map_physical, map_range, unmap, virt_to_phys, CURRENT_PML4};
|
||||||
|
|
||||||
|
use super::{
|
||||||
|
hpet::sleep,
|
||||||
|
task::{ALL_APS_STARTED, STACK_SIZE, STARTING_AP_ID},
|
||||||
|
};
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
fn ap_trampoline();
|
fn ap_trampoline();
|
||||||
|
static trampoline_pml4: u8;
|
||||||
|
static trampoline_stack: u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct LAPIC {
|
struct LAPIC {
|
||||||
|
@ -37,7 +45,7 @@ static ADDRESS: AtomicPtr<u32> = AtomicPtr::new(null_mut());
|
||||||
pub static BSP_LAPIC_ID: AtomicUsize = AtomicUsize::new(0);
|
pub static BSP_LAPIC_ID: AtomicUsize = AtomicUsize::new(0);
|
||||||
static LAPICS: [LAPIC; 256] = [EMPTY_LAPIC; 256];
|
static LAPICS: [LAPIC; 256] = [EMPTY_LAPIC; 256];
|
||||||
static TICKS_PER_MS: AtomicUsize = AtomicUsize::new(0);
|
static TICKS_PER_MS: AtomicUsize = AtomicUsize::new(0);
|
||||||
static NEXT_LAPIC_ID: AtomicUsize = AtomicUsize::new(0);
|
pub static NEXT_LAPIC_ID: AtomicUsize = AtomicUsize::new(0);
|
||||||
|
|
||||||
pub fn send_eoi() {
|
pub fn send_eoi() {
|
||||||
let address = ADDRESS.load(Ordering::SeqCst);
|
let address = ADDRESS.load(Ordering::SeqCst);
|
||||||
|
@ -45,11 +53,18 @@ pub fn send_eoi() {
|
||||||
address.add(REGISTER_EOI).write_volatile(0);
|
address.add(REGISTER_EOI).write_volatile(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
pub fn get_current_lapic_id() -> usize {
|
||||||
|
let address = ADDRESS.load(Ordering::SeqCst);
|
||||||
|
unsafe { address.add(REGISTER_ID).read_volatile() as usize >> 24 }
|
||||||
|
}
|
||||||
pub fn setup_lapic(phys: u64) {
|
pub fn setup_lapic(phys: u64) {
|
||||||
|
if phys != 0 {
|
||||||
let address = unsafe { map_physical(phys, 0x400, false) as *mut u32 };
|
let address = unsafe { map_physical(phys, 0x400, false) as *mut u32 };
|
||||||
ADDRESS.store(address, Ordering::SeqCst);
|
ADDRESS.store(address, Ordering::SeqCst);
|
||||||
|
BSP_LAPIC_ID.store(get_current_lapic_id(), Ordering::SeqCst);
|
||||||
|
}
|
||||||
|
let address = ADDRESS.load(Ordering::SeqCst);
|
||||||
unsafe {
|
unsafe {
|
||||||
BSP_LAPIC_ID.store(address.add(REGISTER_ID).read_volatile() as usize, Ordering::SeqCst);
|
|
||||||
address.add(REGISTER_SPURIOUS_INT).write_volatile(0x1ff);
|
address.add(REGISTER_SPURIOUS_INT).write_volatile(0x1ff);
|
||||||
}
|
}
|
||||||
send_eoi();
|
send_eoi();
|
||||||
|
@ -62,9 +77,6 @@ fn calibrate_timer() {
|
||||||
sleep(10000);
|
sleep(10000);
|
||||||
let ticks_in_10ms = 0xffffffff - unsafe { address.add(REGISTER_TIMER_CURRENT_COUNT).read_volatile() };
|
let ticks_in_10ms = 0xffffffff - unsafe { address.add(REGISTER_TIMER_CURRENT_COUNT).read_volatile() };
|
||||||
TICKS_PER_MS.store(ticks_in_10ms as usize / 10, Ordering::SeqCst);
|
TICKS_PER_MS.store(ticks_in_10ms as usize / 10, Ordering::SeqCst);
|
||||||
unsafe {
|
|
||||||
address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(0);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
fn send_ipi(lapic_id: usize, data: u32) {
|
fn send_ipi(lapic_id: usize, data: u32) {
|
||||||
let address = ADDRESS.load(Ordering::SeqCst);
|
let address = ADDRESS.load(Ordering::SeqCst);
|
||||||
|
@ -79,13 +91,16 @@ pub fn schedule_timer_interrupt() {
|
||||||
address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(10 * TICKS_PER_MS.load(Ordering::SeqCst) as u32);
|
address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(10 * TICKS_PER_MS.load(Ordering::SeqCst) as u32);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn setup_lapic_timer() {
|
pub fn setup_lapic_timer(calibrate: bool) {
|
||||||
let address = ADDRESS.load(Ordering::SeqCst);
|
let address = ADDRESS.load(Ordering::SeqCst);
|
||||||
unsafe {
|
unsafe {
|
||||||
address.add(REGISTER_TIMER_DIVIDE).write_volatile(3);
|
address.add(REGISTER_TIMER_DIVIDE).write_volatile(3);
|
||||||
}
|
}
|
||||||
|
if calibrate {
|
||||||
calibrate_timer();
|
calibrate_timer();
|
||||||
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
|
address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(0);
|
||||||
address.add(REGISTER_TIMER_LVT).write_volatile(254);
|
address.add(REGISTER_TIMER_LVT).write_volatile(254);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -95,22 +110,42 @@ pub fn add_lapic(lapic_id: usize) {
|
||||||
LAPICS[next_id].present.store(true, Ordering::SeqCst);
|
LAPICS[next_id].present.store(true, Ordering::SeqCst);
|
||||||
}
|
}
|
||||||
pub fn start_aps() {
|
pub fn start_aps() {
|
||||||
|
let stack: Vec<u8> = vec![0; STACK_SIZE];
|
||||||
|
let pml4_phys_addr;
|
||||||
|
{
|
||||||
|
let pml4 = CURRENT_PML4.lock();
|
||||||
|
let pml4 = pml4.as_ref().unwrap();
|
||||||
|
let pml4_phys_addr_u64 = virt_to_phys(*pml4 as *const PageTable as u64);
|
||||||
|
pml4_phys_addr = u32::try_from(pml4_phys_addr_u64).unwrap();
|
||||||
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
map_range(0x1000, 0x1000, 0x1000, false, true, false, false);
|
map_range(0x1000, 0x1000, 0x1000, false, true, false, false);
|
||||||
let dest_ptr = 0x1000 as *mut u8;
|
let dest_ptr = 0x1000 as *mut u8;
|
||||||
let src_ptr = ap_trampoline as *const u8;
|
let src_ptr = ap_trampoline as *const u8;
|
||||||
copy(src_ptr, dest_ptr, 0x1000);
|
copy(src_ptr, dest_ptr, 0x1000);
|
||||||
|
let pml4_offset = (&raw const trampoline_pml4).offset_from(src_ptr);
|
||||||
|
let pml4_addr = (0x1000 + pml4_offset) as *mut u32;
|
||||||
|
*pml4_addr = pml4_phys_addr;
|
||||||
|
let stack_offset = (&raw const trampoline_stack).offset_from(src_ptr);
|
||||||
|
let stack_addr = (0x1000 + stack_offset) as *mut u64;
|
||||||
|
*stack_addr = stack.as_ptr() as u64 + STACK_SIZE as u64;
|
||||||
|
map_range(0x1000, 0x1000, 0x1000, false, true, true, false);
|
||||||
}
|
}
|
||||||
for i in 0..NEXT_LAPIC_ID.load(Ordering::SeqCst) {
|
for i in 0..NEXT_LAPIC_ID.load(Ordering::SeqCst) {
|
||||||
let lapic_id = LAPICS[i].lapic_id.load(Ordering::SeqCst);
|
let lapic_id = LAPICS[i].lapic_id.load(Ordering::SeqCst);
|
||||||
if lapic_id == BSP_LAPIC_ID.load(Ordering::SeqCst) {
|
if lapic_id == BSP_LAPIC_ID.load(Ordering::SeqCst) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
STARTING_AP_ID.store(lapic_id as i64, Ordering::SeqCst);
|
||||||
send_ipi(lapic_id, IPI_INIT);
|
send_ipi(lapic_id, IPI_INIT);
|
||||||
sleep(10000);
|
sleep(10000);
|
||||||
send_ipi(lapic_id, IPI_STARTUP | 1);
|
send_ipi(lapic_id, IPI_STARTUP | 1);
|
||||||
|
while STARTING_AP_ID.load(Ordering::SeqCst) != -1 {
|
||||||
|
pause();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
unmap(0x1000);
|
unmap(0x1000);
|
||||||
}
|
}
|
||||||
|
ALL_APS_STARTED.store(true, Ordering::SeqCst);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,18 +1,18 @@
|
||||||
use core::{arch::asm, sync::atomic::Ordering};
|
use core::{arch::asm, sync::atomic::Ordering};
|
||||||
|
|
||||||
use alloc::collections::vec_deque::VecDeque;
|
use alloc::{collections::vec_deque::VecDeque, vec::Vec};
|
||||||
use spin::Mutex;
|
use spin::Mutex;
|
||||||
|
|
||||||
use crate::cpu::isr::ISRState;
|
use crate::cpu::isr::ISRState;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
lapic::schedule_timer_interrupt,
|
lapic::{get_current_lapic_id, schedule_timer_interrupt},
|
||||||
sync::{Spinlock, IN_ISR_HANDLER, LOCKS_HELD},
|
sync::{Spinlock, IN_ISR_HANDLER, LOCKS_HELD},
|
||||||
task::{switch_task, Task, TaskState, CURRENT_TASK, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
task::{switch_task, Task, TaskState, CURRENT_TASKS, CURRENT_TASK_LOCK, MULTITASKING_ENABLED},
|
||||||
};
|
};
|
||||||
|
|
||||||
static SCHEDULER_LIST: Mutex<VecDeque<Task>> = Mutex::new(VecDeque::new());
|
static SCHEDULER_LIST: Mutex<VecDeque<Task>> = Mutex::new(VecDeque::new());
|
||||||
pub static IDLE_TASK: Mutex<Option<Task>> = Mutex::new(None);
|
pub static IDLE_TASKS: Mutex<Vec<Task>> = Mutex::new(Vec::new());
|
||||||
pub static SCHEDULER_LOCK: Spinlock = Spinlock::new();
|
pub static SCHEDULER_LOCK: Spinlock = Spinlock::new();
|
||||||
|
|
||||||
pub fn scheduler(state: &mut ISRState) {
|
pub fn scheduler(state: &mut ISRState) {
|
||||||
|
@ -37,14 +37,20 @@ pub fn scheduler(state: &mut ISRState) {
|
||||||
}
|
}
|
||||||
let mut switch_idle = false;
|
let mut switch_idle = false;
|
||||||
{
|
{
|
||||||
let _current_task = CURRENT_TASK.lock();
|
let _current_task = CURRENT_TASKS.lock();
|
||||||
let current_task = _current_task.as_ref().unwrap();
|
let current_task = _current_task.get(&get_current_lapic_id());
|
||||||
|
if current_task.is_none() {
|
||||||
|
switch_idle = true;
|
||||||
|
}
|
||||||
|
if !switch_idle {
|
||||||
|
let current_task = current_task.unwrap();
|
||||||
if current_task.task_state != TaskState::Ready && current_task.task_state != TaskState::Idle {
|
if current_task.task_state != TaskState::Ready && current_task.task_state != TaskState::Idle {
|
||||||
switch_idle = true;
|
switch_idle = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if switch_idle {
|
if switch_idle {
|
||||||
switch_task(state, IDLE_TASK.lock().take().unwrap());
|
switch_task(state, IDLE_TASKS.lock().pop().unwrap());
|
||||||
}
|
}
|
||||||
CURRENT_TASK_LOCK.unlock();
|
CURRENT_TASK_LOCK.unlock();
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,8 +7,9 @@ use spin::Mutex;
|
||||||
use crate::sys::task::MULTITASKING_ENABLED;
|
use crate::sys::task::MULTITASKING_ENABLED;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
|
lapic::get_current_lapic_id,
|
||||||
scheduler::{schedule_task, yield_task, SCHEDULER_LOCK},
|
scheduler::{schedule_task, yield_task, SCHEDULER_LOCK},
|
||||||
task::{Task, TaskState, CURRENT_TASK, CURRENT_TASK_LOCK},
|
task::{Task, TaskState, CURRENT_TASKS, CURRENT_TASK_LOCK},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub static IN_ISR_HANDLER: AtomicBool = AtomicBool::new(false);
|
pub static IN_ISR_HANDLER: AtomicBool = AtomicBool::new(false);
|
||||||
|
@ -72,8 +73,8 @@ pub fn lock_semaphore(semaphore: Arc<Semaphore>, count: usize) {
|
||||||
}
|
}
|
||||||
CURRENT_TASK_LOCK.lock();
|
CURRENT_TASK_LOCK.lock();
|
||||||
{
|
{
|
||||||
let mut current_task = CURRENT_TASK.lock();
|
let mut current_task = CURRENT_TASKS.lock();
|
||||||
let current_task = current_task.as_mut().unwrap();
|
let current_task = current_task.get_mut(&get_current_lapic_id()).unwrap();
|
||||||
current_task.task_state = TaskState::SemaphoreBlocked;
|
current_task.task_state = TaskState::SemaphoreBlocked;
|
||||||
current_task.block_on_semaphore = Some(semaphore.clone());
|
current_task.block_on_semaphore = Some(semaphore.clone());
|
||||||
current_task.semaphore_requested_count = count;
|
current_task.semaphore_requested_count = count;
|
||||||
|
|
|
@ -1,21 +1,25 @@
|
||||||
use core::{
|
use core::{
|
||||||
arch::asm,
|
arch::asm,
|
||||||
sync::atomic::{AtomicBool, AtomicU64, AtomicUsize, Ordering},
|
sync::atomic::{AtomicBool, AtomicI64, AtomicU64, AtomicUsize, Ordering},
|
||||||
};
|
};
|
||||||
|
|
||||||
use alloc::{sync::Arc, vec, vec::Vec};
|
use alloc::{sync::Arc, vec, vec::Vec};
|
||||||
use kernel_common::instructions::hlt;
|
use hashbrown::HashMap;
|
||||||
use spin::Mutex;
|
use kernel_common::instructions::{hlt, pause};
|
||||||
|
use spin::{Lazy, Mutex};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
cpu::isr::ISRState,
|
cpu::isr::ISRState,
|
||||||
main,
|
main,
|
||||||
sys::scheduler::{schedule_task, IDLE_TASK, SCHEDULER_LOCK},
|
sys::{
|
||||||
|
lapic::NEXT_LAPIC_ID,
|
||||||
|
scheduler::{schedule_task, IDLE_TASKS, SCHEDULER_LOCK},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
hpet::sleep_internal,
|
hpet::sleep_internal,
|
||||||
lapic::schedule_timer_interrupt,
|
lapic::{get_current_lapic_id, schedule_timer_interrupt},
|
||||||
scheduler::yield_task,
|
scheduler::yield_task,
|
||||||
sync::{lock_semaphore_internal, Semaphore, Spinlock},
|
sync::{lock_semaphore_internal, Semaphore, Spinlock},
|
||||||
};
|
};
|
||||||
|
@ -40,13 +44,14 @@ pub struct Task {
|
||||||
pub semaphore_requested_count: usize,
|
pub semaphore_requested_count: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
const STACK_SIZE: usize = 64 * 1024;
|
pub const STACK_SIZE: usize = 64 * 1024;
|
||||||
static NEXT_TASK_ID: AtomicUsize = AtomicUsize::new(2);
|
static NEXT_TASK_ID: AtomicUsize = AtomicUsize::new(2);
|
||||||
|
pub static CURRENT_TASKS: Lazy<Mutex<HashMap<usize, Task>>> = Lazy::new(|| Mutex::new(HashMap::new()));
|
||||||
pub static CURRENT_TASK: Mutex<Option<Task>> = Mutex::new(None);
|
|
||||||
pub static CURRENT_TASK_LOCK: Spinlock = Spinlock::new();
|
pub static CURRENT_TASK_LOCK: Spinlock = Spinlock::new();
|
||||||
static RFLAGS: AtomicU64 = AtomicU64::new(0);
|
static RFLAGS: AtomicU64 = AtomicU64::new(0);
|
||||||
pub static MULTITASKING_ENABLED: AtomicBool = AtomicBool::new(false);
|
pub static MULTITASKING_ENABLED: AtomicBool = AtomicBool::new(false);
|
||||||
|
pub static STARTING_AP_ID: AtomicI64 = AtomicI64::new(-1);
|
||||||
|
pub static ALL_APS_STARTED: AtomicBool = AtomicBool::new(false);
|
||||||
|
|
||||||
pub fn allocate_stack() -> u64 {
|
pub fn allocate_stack() -> u64 {
|
||||||
let stack: Vec<u8> = vec![0; STACK_SIZE];
|
let stack: Vec<u8> = vec![0; STACK_SIZE];
|
||||||
|
@ -54,8 +59,8 @@ pub fn allocate_stack() -> u64 {
|
||||||
}
|
}
|
||||||
pub fn switch_task(current_state: &mut ISRState, new_task: Task) {
|
pub fn switch_task(current_state: &mut ISRState, new_task: Task) {
|
||||||
assert!(CURRENT_TASK_LOCK.is_locked());
|
assert!(CURRENT_TASK_LOCK.is_locked());
|
||||||
let mut _current_task = CURRENT_TASK.lock();
|
let mut _current_task = CURRENT_TASKS.lock();
|
||||||
if let Some(mut current_task) = _current_task.take() {
|
if let Some(mut current_task) = _current_task.remove(&get_current_lapic_id()) {
|
||||||
current_task.state = *current_state;
|
current_task.state = *current_state;
|
||||||
match current_task.task_state {
|
match current_task.task_state {
|
||||||
TaskState::Ready => {
|
TaskState::Ready => {
|
||||||
|
@ -63,14 +68,15 @@ pub fn switch_task(current_state: &mut ISRState, new_task: Task) {
|
||||||
schedule_task(current_task);
|
schedule_task(current_task);
|
||||||
SCHEDULER_LOCK.unlock();
|
SCHEDULER_LOCK.unlock();
|
||||||
}
|
}
|
||||||
TaskState::Idle => *IDLE_TASK.lock() = Some(current_task),
|
TaskState::Idle => IDLE_TASKS.lock().push(current_task),
|
||||||
TaskState::Terminated => {}
|
TaskState::Terminated => {}
|
||||||
TaskState::Sleeping => sleep_internal(current_task),
|
TaskState::Sleeping => sleep_internal(current_task),
|
||||||
TaskState::SemaphoreBlocked => lock_semaphore_internal(current_task),
|
TaskState::SemaphoreBlocked => lock_semaphore_internal(current_task),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*current_state = new_task.state;
|
*current_state = new_task.state;
|
||||||
*_current_task = Some(new_task);
|
let result = _current_task.insert(get_current_lapic_id(), new_task);
|
||||||
|
assert!(result.is_none());
|
||||||
schedule_timer_interrupt();
|
schedule_timer_interrupt();
|
||||||
}
|
}
|
||||||
pub fn create_task(func: fn()) -> Task {
|
pub fn create_task(func: fn()) -> Task {
|
||||||
|
@ -110,24 +116,37 @@ pub fn create_task(func: fn()) -> Task {
|
||||||
};
|
};
|
||||||
task
|
task
|
||||||
}
|
}
|
||||||
|
fn create_idle_task() {
|
||||||
|
let mut idle_task = create_task(idle_main);
|
||||||
|
idle_task.task_state = TaskState::Idle;
|
||||||
|
{
|
||||||
|
IDLE_TASKS.lock().push(idle_task);
|
||||||
|
}
|
||||||
|
}
|
||||||
extern "C" fn task_entry() -> ! {
|
extern "C" fn task_entry() -> ! {
|
||||||
CURRENT_TASK_LOCK.lock();
|
CURRENT_TASK_LOCK.lock();
|
||||||
let func;
|
let func;
|
||||||
{
|
{
|
||||||
let task = CURRENT_TASK.lock();
|
let task = CURRENT_TASKS.lock();
|
||||||
func = task.as_ref().unwrap().initial_func;
|
func = task.get(&get_current_lapic_id()).unwrap().initial_func;
|
||||||
}
|
}
|
||||||
CURRENT_TASK_LOCK.unlock();
|
CURRENT_TASK_LOCK.unlock();
|
||||||
func();
|
func();
|
||||||
CURRENT_TASK_LOCK.lock();
|
CURRENT_TASK_LOCK.lock();
|
||||||
{
|
{
|
||||||
CURRENT_TASK.lock().as_mut().unwrap().task_state = TaskState::Terminated;
|
CURRENT_TASKS.lock().get_mut(&get_current_lapic_id()).unwrap().task_state = TaskState::Terminated;
|
||||||
}
|
}
|
||||||
CURRENT_TASK_LOCK.unlock();
|
CURRENT_TASK_LOCK.unlock();
|
||||||
yield_task();
|
yield_task();
|
||||||
panic!("Failed to terminate task");
|
panic!("Failed to terminate task");
|
||||||
}
|
}
|
||||||
fn idle() {
|
fn idle_main() {
|
||||||
|
while !ALL_APS_STARTED.load(Ordering::SeqCst) {
|
||||||
|
if STARTING_AP_ID.load(Ordering::SeqCst) == get_current_lapic_id() as i64 {
|
||||||
|
let _ = STARTING_AP_ID.compare_exchange(get_current_lapic_id() as i64, -1, Ordering::SeqCst, Ordering::SeqCst);
|
||||||
|
}
|
||||||
|
pause();
|
||||||
|
}
|
||||||
loop {
|
loop {
|
||||||
hlt();
|
hlt();
|
||||||
}
|
}
|
||||||
|
@ -138,10 +157,8 @@ pub fn setup_multitasking() -> ! {
|
||||||
asm!("pushf; pop {0:r}", out(reg) rflags);
|
asm!("pushf; pop {0:r}", out(reg) rflags);
|
||||||
}
|
}
|
||||||
RFLAGS.store(rflags, core::sync::atomic::Ordering::SeqCst);
|
RFLAGS.store(rflags, core::sync::atomic::Ordering::SeqCst);
|
||||||
let mut idle_task = create_task(idle);
|
for _ in 0..NEXT_LAPIC_ID.load(Ordering::SeqCst) {
|
||||||
idle_task.task_state = TaskState::Idle;
|
create_idle_task();
|
||||||
{
|
|
||||||
*IDLE_TASK.lock() = Some(idle_task);
|
|
||||||
}
|
}
|
||||||
let task = create_task(main);
|
let task = create_task(main);
|
||||||
schedule_task(task);
|
schedule_task(task);
|
||||||
|
|
Loading…
Add table
Reference in a new issue