From fe1c702a8f2b596ba414031fd6001654450395bc Mon Sep 17 00:00:00 2001 From: Mathieu Strypsteen Date: Thu, 12 Dec 2024 16:24:49 +0100 Subject: [PATCH] Start all cores --- Cargo.lock | 30 +++++++++++++++++++ kernel/Cargo.toml | 1 + kernel/src/cpu/boot.s | 1 + kernel/src/cpu/mod.rs | 1 + kernel/src/cpu/paging.rs | 11 +++++-- kernel/src/cpu/smp.rs | 23 +++++++++++++++ kernel/src/cpu/trampoline.s | 47 +++++++++++++++++++++++++++-- kernel/src/main.rs | 4 +-- kernel/src/sys/acpica_osl.rs | 5 ++-- kernel/src/sys/hpet.rs | 7 +++-- kernel/src/sys/lapic.rs | 57 +++++++++++++++++++++++++++++------- kernel/src/sys/scheduler.rs | 22 +++++++++----- kernel/src/sys/sync.rs | 7 +++-- kernel/src/sys/task.rs | 57 +++++++++++++++++++++++------------- 14 files changed, 218 insertions(+), 55 deletions(-) create mode 100644 kernel/src/cpu/smp.rs diff --git a/Cargo.lock b/Cargo.lock index a13823d..89ea2f7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -35,6 +35,12 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6b2d54853319fd101b8dd81de382bcbf3e03410a64d8928bbee85a3e7dcde483" +[[package]] +name = "allocator-api2" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" + [[package]] name = "autocfg" version = "1.4.0" @@ -173,6 +179,12 @@ dependencies = [ "byteorder", ] +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + [[package]] name = "float-cmp" version = "0.9.0" @@ -182,6 +194,12 @@ dependencies = [ "num-traits", ] +[[package]] +name = "foldhash" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f81ec6369c545a7d40e4589b5597581fa1c441fe1cce96dd1de43159910a36a2" + [[package]] name = "funty" version = "2.0.0" @@ -194,6 +212,17 @@ version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" 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]] name = "itertools" version = "0.13.0" @@ -213,6 +242,7 @@ dependencies = [ "bitvec", "buddy_system_allocator", "embedded-graphics", + "hashbrown", "kernel-common", "log", "spin", diff --git a/kernel/Cargo.toml b/kernel/Cargo.toml index 2ab6488..a4afae0 100644 --- a/kernel/Cargo.toml +++ b/kernel/Cargo.toml @@ -11,6 +11,7 @@ bitfield = "0.17.0" bitvec = {version = "1.0.1", default-features = false, features = ["alloc", "atomic"]} buddy_system_allocator = "0.11.0" embedded-graphics = "0.8.1" +hashbrown = "0.15.2" kernel-common = {path = "../lib/kernel-common"} log = "0.4.22" spin = "0.9.8" diff --git a/kernel/src/cpu/boot.s b/kernel/src/cpu/boot.s index 2b211f3..45f0cb3 100644 --- a/kernel/src/cpu/boot.s +++ b/kernel/src/cpu/boot.s @@ -11,6 +11,7 @@ _start: call rdrand mov %rax, __stack_chk_guard call early_main + .section .bss .align 8 .global __stack_chk_guard diff --git a/kernel/src/cpu/mod.rs b/kernel/src/cpu/mod.rs index fdb7f9e..3ed47e8 100644 --- a/kernel/src/cpu/mod.rs +++ b/kernel/src/cpu/mod.rs @@ -2,3 +2,4 @@ pub mod gdt; pub mod idt; pub mod isr; pub mod paging; +pub mod smp; diff --git a/kernel/src/cpu/paging.rs b/kernel/src/cpu/paging.rs index 750c182..83dc583 100644 --- a/kernel/src/cpu/paging.rs +++ b/kernel/src/cpu/paging.rs @@ -25,7 +25,7 @@ extern "C" { } static PAGING_ACTIVE: AtomicBool = AtomicBool::new(false); -static CURRENT_PML4: Mutex> = Mutex::new(None); +pub static CURRENT_PML4: Mutex> = Mutex::new(None); static HEAP_PHYS_START: AtomicU64 = AtomicU64::new(0); static PHYSICAL_FRAMES: Mutex>> = Mutex::new(None); static HEAP_PHYS_MAPPING: Mutex> = Mutex::new(Vec::new()); @@ -42,11 +42,12 @@ fn _get_free_frame() -> u64 { panic!("No free memory left"); } fn invlpg(addr: u64) { + // TODO: Broadcast to all cores unsafe { 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) { 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) { 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 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 directory = get_table_entry(pdpt, pdpt_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_user(user 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_cache_disable(cache_disable as u64); table.entries_phys[table_i].set_present(1); + if should_invalidate { + invlpg(virt); + } } pub unsafe fn unmap(address: u64) { let mut current_pml4 = CURRENT_PML4.lock(); diff --git a/kernel/src/cpu/smp.rs b/kernel/src/cpu/smp.rs new file mode 100644 index 0000000..e5ecf41 --- /dev/null +++ b/kernel/src/cpu/smp.rs @@ -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"); +} diff --git a/kernel/src/cpu/trampoline.s b/kernel/src/cpu/trampoline.s index 8bb33f7..65f8550 100644 --- a/kernel/src/cpu/trampoline.s +++ b/kernel/src/cpu/trampoline.s @@ -1,5 +1,46 @@ -.text +.section .rodata +.code16 +.align 0x1000 -.global 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 diff --git a/kernel/src/main.rs b/kernel/src/main.rs index 2a11d4b..311c174 100644 --- a/kernel/src/main.rs +++ b/kernel/src/main.rs @@ -85,11 +85,11 @@ extern "C" fn early_main(temp_loader_struct: *const LoaderStruct) -> ! { } parse_madt(&early_acpi_tables); setup_hpet(&early_acpi_tables); - setup_lapic_timer(); + setup_lapic_timer(true); setup_multitasking(); } fn main() { - info!("Starting main kernel task"); + info!("Starting main kernel task..."); let mut status = unsafe { AcpiInitializeSubsystem() }; assert_eq!(status, AE_OK); status = unsafe { AcpiInitializeTables(null_mut(), 0, 0) }; diff --git a/kernel/src/sys/acpica_osl.rs b/kernel/src/sys/acpica_osl.rs index 70dfb03..1512cdc 100644 --- a/kernel/src/sys/acpica_osl.rs +++ b/kernel/src/sys/acpica_osl.rs @@ -25,8 +25,9 @@ static SCI_CONTEXT: AtomicPtr = AtomicPtr::new(null_mut()); use super::{ hpet::get_current_time, + lapic::get_current_lapic_id, 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; @@ -95,7 +96,7 @@ extern "C" fn AcpiOsGetThreadId() -> UINT64 { let task_id; 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(); task_id as UINT64 diff --git a/kernel/src/sys/hpet.rs b/kernel/src/sys/hpet.rs index 474273f..9d1f5ef 100644 --- a/kernel/src/sys/hpet.rs +++ b/kernel/src/sys/hpet.rs @@ -13,9 +13,10 @@ use crate::cpu::paging::map_physical; use super::{ early_acpi::EarlyACPIHandler, ioapic::register_irq_handler, + lapic::get_current_lapic_id, scheduler::{schedule_task, yield_task, SCHEDULER_LOCK}, 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; @@ -98,8 +99,8 @@ pub fn sleep(us: usize) { if MULTITASKING_ENABLED.load(Ordering::SeqCst) { CURRENT_TASK_LOCK.lock(); { - let mut _current_task = CURRENT_TASK.lock(); - let current_task = _current_task.as_mut().unwrap(); + let mut _current_task = CURRENT_TASKS.lock(); + let current_task = _current_task.get_mut(&get_current_lapic_id()).unwrap(); current_task.sleep_until_us = get_current_time() + us; current_task.task_state = TaskState::Sleeping; } diff --git a/kernel/src/sys/lapic.rs b/kernel/src/sys/lapic.rs index d8276b3..ef8a746 100644 --- a/kernel/src/sys/lapic.rs +++ b/kernel/src/sys/lapic.rs @@ -3,12 +3,20 @@ use core::{ 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" { fn ap_trampoline(); + static trampoline_pml4: u8; + static trampoline_stack: u8; } struct LAPIC { @@ -37,7 +45,7 @@ static ADDRESS: AtomicPtr = AtomicPtr::new(null_mut()); pub static BSP_LAPIC_ID: AtomicUsize = AtomicUsize::new(0); static LAPICS: [LAPIC; 256] = [EMPTY_LAPIC; 256]; 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() { let address = ADDRESS.load(Ordering::SeqCst); @@ -45,11 +53,18 @@ pub fn send_eoi() { 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) { - let address = unsafe { map_physical(phys, 0x400, false) as *mut u32 }; - ADDRESS.store(address, Ordering::SeqCst); + if phys != 0 { + let address = unsafe { map_physical(phys, 0x400, false) as *mut u32 }; + ADDRESS.store(address, Ordering::SeqCst); + BSP_LAPIC_ID.store(get_current_lapic_id(), Ordering::SeqCst); + } + let address = ADDRESS.load(Ordering::SeqCst); unsafe { - BSP_LAPIC_ID.store(address.add(REGISTER_ID).read_volatile() as usize, Ordering::SeqCst); address.add(REGISTER_SPURIOUS_INT).write_volatile(0x1ff); } send_eoi(); @@ -62,9 +77,6 @@ fn calibrate_timer() { sleep(10000); 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); - unsafe { - address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(0); - } } fn send_ipi(lapic_id: usize, data: u32) { 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); } } -pub fn setup_lapic_timer() { +pub fn setup_lapic_timer(calibrate: bool) { let address = ADDRESS.load(Ordering::SeqCst); unsafe { address.add(REGISTER_TIMER_DIVIDE).write_volatile(3); } - calibrate_timer(); + if calibrate { + calibrate_timer(); + } unsafe { + address.add(REGISTER_TIMER_INITIAL_COUNT).write_volatile(0); 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); } pub fn start_aps() { + let stack: Vec = 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 { map_range(0x1000, 0x1000, 0x1000, false, true, false, false); let dest_ptr = 0x1000 as *mut u8; let src_ptr = ap_trampoline as *const u8; 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) { let lapic_id = LAPICS[i].lapic_id.load(Ordering::SeqCst); if lapic_id == BSP_LAPIC_ID.load(Ordering::SeqCst) { continue; } + STARTING_AP_ID.store(lapic_id as i64, Ordering::SeqCst); send_ipi(lapic_id, IPI_INIT); sleep(10000); send_ipi(lapic_id, IPI_STARTUP | 1); + while STARTING_AP_ID.load(Ordering::SeqCst) != -1 { + pause(); + } } unsafe { unmap(0x1000); } + ALL_APS_STARTED.store(true, Ordering::SeqCst); } diff --git a/kernel/src/sys/scheduler.rs b/kernel/src/sys/scheduler.rs index e119049..fccddf4 100644 --- a/kernel/src/sys/scheduler.rs +++ b/kernel/src/sys/scheduler.rs @@ -1,18 +1,18 @@ 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 crate::cpu::isr::ISRState; use super::{ - lapic::schedule_timer_interrupt, + lapic::{get_current_lapic_id, schedule_timer_interrupt}, 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> = Mutex::new(VecDeque::new()); -pub static IDLE_TASK: Mutex> = Mutex::new(None); +pub static IDLE_TASKS: Mutex> = Mutex::new(Vec::new()); pub static SCHEDULER_LOCK: Spinlock = Spinlock::new(); pub fn scheduler(state: &mut ISRState) { @@ -37,14 +37,20 @@ pub fn scheduler(state: &mut ISRState) { } let mut switch_idle = false; { - let _current_task = CURRENT_TASK.lock(); - let current_task = _current_task.as_ref().unwrap(); - if current_task.task_state != TaskState::Ready && current_task.task_state != TaskState::Idle { + let _current_task = CURRENT_TASKS.lock(); + 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 { + switch_idle = true; + } + } } if switch_idle { - switch_task(state, IDLE_TASK.lock().take().unwrap()); + switch_task(state, IDLE_TASKS.lock().pop().unwrap()); } CURRENT_TASK_LOCK.unlock(); } diff --git a/kernel/src/sys/sync.rs b/kernel/src/sys/sync.rs index 2f42608..3bf1fbb 100644 --- a/kernel/src/sys/sync.rs +++ b/kernel/src/sys/sync.rs @@ -7,8 +7,9 @@ use spin::Mutex; use crate::sys::task::MULTITASKING_ENABLED; use super::{ + lapic::get_current_lapic_id, 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); @@ -72,8 +73,8 @@ pub fn lock_semaphore(semaphore: Arc, count: usize) { } CURRENT_TASK_LOCK.lock(); { - let mut current_task = CURRENT_TASK.lock(); - let current_task = current_task.as_mut().unwrap(); + let mut current_task = CURRENT_TASKS.lock(); + let current_task = current_task.get_mut(&get_current_lapic_id()).unwrap(); current_task.task_state = TaskState::SemaphoreBlocked; current_task.block_on_semaphore = Some(semaphore.clone()); current_task.semaphore_requested_count = count; diff --git a/kernel/src/sys/task.rs b/kernel/src/sys/task.rs index d3b5a99..0eca00a 100644 --- a/kernel/src/sys/task.rs +++ b/kernel/src/sys/task.rs @@ -1,21 +1,25 @@ use core::{ arch::asm, - sync::atomic::{AtomicBool, AtomicU64, AtomicUsize, Ordering}, + sync::atomic::{AtomicBool, AtomicI64, AtomicU64, AtomicUsize, Ordering}, }; use alloc::{sync::Arc, vec, vec::Vec}; -use kernel_common::instructions::hlt; -use spin::Mutex; +use hashbrown::HashMap; +use kernel_common::instructions::{hlt, pause}; +use spin::{Lazy, Mutex}; use crate::{ cpu::isr::ISRState, main, - sys::scheduler::{schedule_task, IDLE_TASK, SCHEDULER_LOCK}, + sys::{ + lapic::NEXT_LAPIC_ID, + scheduler::{schedule_task, IDLE_TASKS, SCHEDULER_LOCK}, + }, }; use super::{ hpet::sleep_internal, - lapic::schedule_timer_interrupt, + lapic::{get_current_lapic_id, schedule_timer_interrupt}, scheduler::yield_task, sync::{lock_semaphore_internal, Semaphore, Spinlock}, }; @@ -40,13 +44,14 @@ pub struct Task { 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); - -pub static CURRENT_TASK: Mutex> = Mutex::new(None); +pub static CURRENT_TASKS: Lazy>> = Lazy::new(|| Mutex::new(HashMap::new())); pub static CURRENT_TASK_LOCK: Spinlock = Spinlock::new(); static RFLAGS: AtomicU64 = AtomicU64::new(0); 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 { let stack: Vec = vec![0; STACK_SIZE]; @@ -54,8 +59,8 @@ pub fn allocate_stack() -> u64 { } pub fn switch_task(current_state: &mut ISRState, new_task: Task) { assert!(CURRENT_TASK_LOCK.is_locked()); - let mut _current_task = CURRENT_TASK.lock(); - if let Some(mut current_task) = _current_task.take() { + let mut _current_task = CURRENT_TASKS.lock(); + if let Some(mut current_task) = _current_task.remove(&get_current_lapic_id()) { current_task.state = *current_state; match current_task.task_state { TaskState::Ready => { @@ -63,14 +68,15 @@ pub fn switch_task(current_state: &mut ISRState, new_task: Task) { schedule_task(current_task); SCHEDULER_LOCK.unlock(); } - TaskState::Idle => *IDLE_TASK.lock() = Some(current_task), + TaskState::Idle => IDLE_TASKS.lock().push(current_task), TaskState::Terminated => {} TaskState::Sleeping => sleep_internal(current_task), TaskState::SemaphoreBlocked => lock_semaphore_internal(current_task), } } *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(); } pub fn create_task(func: fn()) -> Task { @@ -110,24 +116,37 @@ pub fn create_task(func: fn()) -> 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() -> ! { CURRENT_TASK_LOCK.lock(); let func; { - let task = CURRENT_TASK.lock(); - func = task.as_ref().unwrap().initial_func; + let task = CURRENT_TASKS.lock(); + func = task.get(&get_current_lapic_id()).unwrap().initial_func; } CURRENT_TASK_LOCK.unlock(); func(); 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(); yield_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 { hlt(); } @@ -138,10 +157,8 @@ pub fn setup_multitasking() -> ! { asm!("pushf; pop {0:r}", out(reg) rflags); } RFLAGS.store(rflags, core::sync::atomic::Ordering::SeqCst); - let mut idle_task = create_task(idle); - idle_task.task_state = TaskState::Idle; - { - *IDLE_TASK.lock() = Some(idle_task); + for _ in 0..NEXT_LAPIC_ID.load(Ordering::SeqCst) { + create_idle_task(); } let task = create_task(main); schedule_task(task);