mirror of
https://github.com/justinian/jsix.git
synced 2025-12-10 08:24:32 -08:00
The kernel log levels are now numerically reversed so that more-verbose levels can be added to the end. Replaced 'debug' with 'verbose', and added new 'spam' level.
138 lines
3.7 KiB
C++
138 lines
3.7 KiB
C++
#include <stdint.h>
|
|
|
|
#include "apic.h"
|
|
#include "clock.h"
|
|
#include "device_manager.h"
|
|
#include "interrupts.h"
|
|
#include "logger.h"
|
|
#include "memory.h"
|
|
#include "objects/vm_area.h"
|
|
#include "scheduler.h"
|
|
#include "smp.h"
|
|
#include "vm_space.h"
|
|
|
|
extern "C" {
|
|
void long_ap_startup(cpu_data *cpu);
|
|
void ap_startup();
|
|
void ap_idle();
|
|
void init_ap_trampoline(void*, cpu_data *, void (*)());
|
|
}
|
|
|
|
extern size_t ap_startup_code_size;
|
|
extern obj::process &g_kernel_process;
|
|
extern obj::vm_area_guarded &g_kernel_stacks;
|
|
|
|
extern cpu_data **g_cpu_data;
|
|
|
|
namespace smp {
|
|
|
|
volatile size_t ap_startup_count;
|
|
volatile bool scheduler_ready = false;
|
|
|
|
|
|
unsigned
|
|
start(cpu_data &bsp, void *kpml4)
|
|
{
|
|
using mem::frame_size;
|
|
using mem::kernel_stack_pages;
|
|
using obj::vm_flags;
|
|
|
|
ap_startup_count = 1; // Count the BSP
|
|
|
|
clock &clk = clock::get();
|
|
|
|
const auto &ids = device_manager::get().get_apic_ids();
|
|
|
|
g_cpu_data = new cpu_data* [ids.count()];
|
|
g_cpu_data[bsp.index] = &bsp;
|
|
|
|
log::info(logs::boot, "Starting %d other CPUs", ids.count() - 1);
|
|
|
|
// Since we're using address space outside kernel space, make sure
|
|
// the kernel's vm_space is used
|
|
bsp.process = &g_kernel_process;
|
|
|
|
uint16_t index = bsp.index;
|
|
|
|
// Copy the startup code somwhere the real mode trampoline can run
|
|
uintptr_t addr = 0x8000; // TODO: find a valid address, rewrite addresses
|
|
isr vector = static_cast<isr>(addr >> 12);
|
|
obj::vm_area *vma = new obj::vm_area_fixed(addr, 0x1000, vm_flags::write);
|
|
vm_space::kernel_space().add(addr, vma);
|
|
memcpy(
|
|
reinterpret_cast<void*>(addr),
|
|
reinterpret_cast<void*>(&ap_startup),
|
|
ap_startup_code_size);
|
|
|
|
size_t free_stack_count = 0;
|
|
|
|
lapic &apic = *bsp.apic;
|
|
lapic::ipi mode = lapic::ipi::init | lapic::ipi::level | lapic::ipi::assert;
|
|
apic.send_ipi_broadcast(mode, false, static_cast<isr>(0));
|
|
|
|
for (uint8_t id : ids) {
|
|
if (id == bsp.id) continue;
|
|
|
|
cpu_data *cpu = cpu_create(id, ++index);
|
|
|
|
static constexpr size_t stack_bytes = kernel_stack_pages * frame_size;
|
|
const uintptr_t stack_start = g_kernel_stacks.get_section();
|
|
uintptr_t stack_end = stack_start + stack_bytes;
|
|
|
|
stack_end -= 2 * sizeof(void*); // Null frame
|
|
*reinterpret_cast<uint64_t*>(stack_end) = 0; // pre-fault the page
|
|
cpu->rsp0 = stack_end;
|
|
|
|
// Set up the trampoline with this CPU's data
|
|
init_ap_trampoline(kpml4, cpu, ap_idle);
|
|
|
|
// Kick it off!
|
|
size_t current_count = ap_startup_count;
|
|
log::verbose(logs::boot, "Starting AP %d: stack %llx", cpu->index, stack_end);
|
|
|
|
lapic::ipi startup = lapic::ipi::startup | lapic::ipi::assert;
|
|
|
|
apic.send_ipi(startup, vector, id);
|
|
for (unsigned i = 0; i < 20; ++i) {
|
|
if (ap_startup_count > current_count) break;
|
|
clk.spinwait(20);
|
|
}
|
|
|
|
// If the CPU already incremented ap_startup_count, it's done
|
|
if (ap_startup_count > current_count)
|
|
continue;
|
|
|
|
// Send the second SIPI (intel recommends this)
|
|
apic.send_ipi(startup, vector, id);
|
|
for (unsigned i = 0; i < 100; ++i) {
|
|
if (ap_startup_count > current_count) break;
|
|
clk.spinwait(100);
|
|
}
|
|
|
|
log::warn(logs::boot, "No response from AP %d within timeout", id);
|
|
}
|
|
|
|
log::info(logs::boot, "%d CPUs running", ap_startup_count);
|
|
vm_space::kernel_space().remove(vma);
|
|
return ap_startup_count;
|
|
}
|
|
|
|
void
|
|
ready()
|
|
{
|
|
scheduler_ready = true;
|
|
}
|
|
|
|
} // namespace smp
|
|
|
|
void
|
|
long_ap_startup(cpu_data *cpu)
|
|
{
|
|
__atomic_add_fetch(&smp::ap_startup_count, 1, __ATOMIC_SEQ_CST);
|
|
|
|
cpu_init(cpu, false);
|
|
|
|
while (!smp::scheduler_ready) asm ("pause");
|
|
scheduler::get().start();
|
|
}
|