kernel/kernelmq/main.c

109 lines
2.9 KiB
C
Raw Normal View History

#include "multiboot.h"
2017-11-04 15:08:59 +00:00
#include "paging.h"
2017-11-04 02:32:23 +00:00
2017-11-09 16:00:36 +00:00
#include "info.h"
#include "stdlib.h"
2020-11-26 18:50:25 +00:00
#include "panic.h"
#include "protected.h"
#include "module.h"
#include "process.h"
#include "tasks.h"
#include "elf.h"
#include "logger.h"
2020-11-28 00:19:36 +00:00
#include <kernaux/multiboot2.h>
// Defined in linker script
extern char _kernel_offset;
extern char _kernel_size;
extern char _kernel_phys_base;
extern char _kernel_virt_base;
extern char _kernel_stack_top;
2017-11-04 10:38:28 +00:00
static struct KernelMQ_Info kinfo;
2020-11-26 18:50:25 +00:00
static KernelMQ_Process_List process_list;
void main(unsigned long multiboot_magic, unsigned long multiboot_info_base)
2017-11-04 02:32:23 +00:00
{
2020-11-28 00:19:36 +00:00
if (multiboot_magic != KERNAUX_MULTIBOOT2_MAGIC) {
2020-11-26 18:52:35 +00:00
panic("Multiboot 2 magic number is invalid.");
}
2017-11-04 11:45:48 +00:00
2020-11-28 01:38:32 +00:00
KernAux_Multiboot2_print(
(struct KernAux_Multiboot2*)multiboot_info_base,
print
);
2020-11-28 03:24:03 +00:00
if (!KernAux_Multiboot2_is_valid(
(struct KernAux_Multiboot2*)multiboot_info_base
)) {
panic("Multiboot 2 info is invalid.");
}
2017-11-04 10:38:28 +00:00
kmemset(&kinfo, 0, sizeof(struct KernelMQ_Info));
if (!multiboot_parse(&kinfo, multiboot_info_base)) {
2020-11-26 18:52:35 +00:00
panic("Can not parse Multiboot 2 info.");
}
kinfo.kernel_offset = (unsigned long)&_kernel_offset;
kinfo.kernel_size = (unsigned long)&_kernel_size;
kinfo.kernel_phys_base = (unsigned long)&_kernel_phys_base;
kinfo.kernel_virt_base = (unsigned long)&_kernel_virt_base;
2017-11-04 10:38:28 +00:00
kinfo.kernel_phys_limit = kinfo.kernel_phys_base + kinfo.kernel_size - 1;
kinfo.kernel_virt_limit = kinfo.kernel_virt_base + kinfo.kernel_size - 1;
2017-11-05 07:13:54 +00:00
kinfo.kernel_and_modules_total_size = kinfo.kernel_size + kinfo.modules_total_size;
kinfo.kernel_stack_top = (unsigned long)&_kernel_stack_top;
2017-11-04 02:32:23 +00:00
2017-11-04 15:08:59 +00:00
paging_clear();
paging_identity();
/* kinfo.freepde_start = */ paging_mapkernel(&kinfo);
2017-11-04 15:08:59 +00:00
paging_load();
paging_enable();
2020-11-26 18:50:25 +00:00
assert(kernelmq_info_validate(&kinfo), "Invalid kernel information.");
protected_initialize(&kinfo);
const enum KernelMQ_Process_Error process_list_init_result =
KernelMQ_Process_List_init(&process_list, &kinfo);
if (process_list_init_result != KERNELMQ_PROCESS_ERROR_OK) {
logger_fail_from(
"init",
"Process list initialization failed with %u.",
process_list_init_result
);
panic("Can not initialize process list.");
}
logger_debug_from("init", "Process list initialized.");
KernelMQ_Process_List_print(&process_list);
if (kinfo.modules_count > 0) {
const struct KernelMQ_ELF_Header *const elf_header =
(void*)kinfo.modules[0].base;
if (KernelMQ_ELF_Header_is_valid(elf_header)) {
const unsigned long real_entrypoint =
kinfo.modules[0].base + elf_header->entrypoint;
tasks_switch_to_user(real_entrypoint);
}
else {
logger_warn_from("init", "Invalid ELF header");
}
}
2017-11-04 02:32:23 +00:00
}