189 lines
5.6 KiB
C++
189 lines
5.6 KiB
C++
#include "loader/init_loader.h"
|
|
|
|
#include <glacier/memory/ref_ptr.h>
|
|
#include <glacier/string/string.h>
|
|
|
|
#include "boot/acpi.h"
|
|
#include "boot/boot_info.h"
|
|
#include "debug/debug.h"
|
|
#include "include/zcall.h"
|
|
#include "memory/paging_util.h"
|
|
#include "object/process.h"
|
|
#include "object/thread.h"
|
|
#include "scheduler/process_manager.h"
|
|
#include "scheduler/scheduler.h"
|
|
|
|
#define K_INIT_DEBUG 0
|
|
|
|
namespace {
|
|
|
|
typedef struct {
|
|
char ident[16];
|
|
uint16_t type;
|
|
uint16_t machine;
|
|
uint32_t version;
|
|
uint64_t entry;
|
|
uint64_t phoff;
|
|
uint64_t shoff;
|
|
uint32_t flags;
|
|
uint16_t ehsize;
|
|
uint16_t phentsize;
|
|
uint16_t phnum;
|
|
uint16_t shentsize;
|
|
uint16_t shnum;
|
|
uint16_t shstrndx;
|
|
} Elf64Header;
|
|
|
|
typedef struct {
|
|
uint32_t name;
|
|
uint32_t type;
|
|
uint64_t flags;
|
|
uint64_t addr;
|
|
uint64_t offset;
|
|
uint64_t size;
|
|
uint32_t link;
|
|
uint32_t info;
|
|
uint64_t addralign;
|
|
uint64_t entsize;
|
|
} Elf64SectionHeader;
|
|
|
|
typedef struct {
|
|
uint32_t type;
|
|
uint32_t flags;
|
|
uint64_t offset;
|
|
uint64_t vaddr;
|
|
uint64_t paddr;
|
|
uint64_t filesz;
|
|
uint64_t memsz;
|
|
uint64_t align;
|
|
} Elf64ProgramHeader;
|
|
|
|
uint64_t LoadElfProgram(Process& dest_proc, uint64_t base, uint64_t offset) {
|
|
Elf64Header* header = reinterpret_cast<Elf64Header*>(base);
|
|
#if K_INIT_DEBUG
|
|
dbgln("phoff: {} phnum: {}", header->phoff, header->phnum);
|
|
#endif
|
|
Elf64ProgramHeader* programs =
|
|
reinterpret_cast<Elf64ProgramHeader*>(base + header->phoff);
|
|
for (uint64_t i = 0; i < header->phnum; i++) {
|
|
Elf64ProgramHeader& program = programs[i];
|
|
#if K_INIT_DEBUG
|
|
dbgln(
|
|
"prog: type: {}, flags: {}, offset: {}\n vaddr: {x}, paddr: {x}\n "
|
|
"filesz: {x}, memsz: {x}, align: {x}",
|
|
program.type, program.flags, program.offset, program.vaddr,
|
|
program.paddr, program.filesz, program.memsz, program.align);
|
|
#endif
|
|
auto mem_obj = glcr::MakeRefCounted<VariableMemoryObject>(program.memsz);
|
|
mem_obj->CopyBytesToObject(base + program.offset, program.filesz);
|
|
PANIC_ON_ERR(
|
|
dest_proc.vmas()->MapInMemoryObject(
|
|
program.vaddr, glcr::StaticCastRefPtr<MemoryObject>(mem_obj)),
|
|
"Couldn't map in init program.");
|
|
}
|
|
return header->entry;
|
|
}
|
|
|
|
void DumpModules() {
|
|
#if K_INIT_DEBUG
|
|
const limine_module_response& resp = boot::GetModules();
|
|
dbgln("[boot] Dumping bootloader modules.");
|
|
for (uint64_t i = 0; i < resp.module_count; i++) {
|
|
const limine_file& file = *resp.modules[i];
|
|
dbgln(" {},{x},{x}", glcr::String(file.path), (uint64_t)file.address,
|
|
file.size);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
const limine_file& GetInitProgram(const glcr::String& path) {
|
|
const limine_module_response& resp = boot::GetModules();
|
|
for (uint64_t i = 0; i < resp.module_count; i++) {
|
|
const limine_file& file = *resp.modules[i];
|
|
if (path == file.path) {
|
|
return file;
|
|
}
|
|
}
|
|
panic("Program not found: {}", path.view());
|
|
UNREACHABLE
|
|
}
|
|
|
|
void WriteInitProgram(glcr::RefPtr<Port> port, glcr::String name, uint64_t id) {
|
|
const limine_file& prog = GetInitProgram(name);
|
|
glcr::RefPtr<VariableMemoryObject> prog_vmmo =
|
|
glcr::MakeRefCounted<VariableMemoryObject>(prog.size);
|
|
// TODO: These seem to be page aligned we should just construct an object
|
|
// around them.
|
|
prog_vmmo->CopyBytesToObject(reinterpret_cast<uint64_t>(prog.address),
|
|
prog.size);
|
|
port->WriteKernel(id, MakeRefCounted<Capability>(prog_vmmo));
|
|
}
|
|
|
|
glcr::ErrorCode WritePciVmmo(glcr::RefPtr<Port> port, uint64_t id) {
|
|
ASSIGN_OR_RETURN(PcieConfiguration config, GetPciExtendedConfiguration());
|
|
auto vmmo =
|
|
glcr::MakeRefCounted<ViewMemoryObject>(config.base, config.offset);
|
|
|
|
#if K_INIT_DEBUG
|
|
dbgln("PCI Configuration found at: {x}:{x}", config.base, config.offset);
|
|
#endif
|
|
|
|
port->WriteKernel(id, MakeRefCounted<Capability>(vmmo));
|
|
|
|
return glcr::OK;
|
|
}
|
|
|
|
void WriteFramebufferVmmo(glcr::RefPtr<Port> port) {
|
|
const limine_framebuffer& buf = boot::GetFramebuffer();
|
|
ZFramebufferInfo ubuf{
|
|
.address_phys = reinterpret_cast<uint64_t>(buf.address) -
|
|
boot::GetHigherHalfDirectMap(),
|
|
.width = buf.width,
|
|
.height = buf.height,
|
|
.pitch = buf.pitch,
|
|
.bpp = buf.bpp,
|
|
.memory_model = buf.memory_model,
|
|
.red_mask_size = buf.red_mask_size,
|
|
.red_mask_shift = buf.red_mask_shift,
|
|
.green_mask_size = buf.green_mask_size,
|
|
.green_mask_shift = buf.green_mask_shift,
|
|
.blue_mask_size = buf.blue_mask_size,
|
|
.blue_mask_shift = buf.blue_mask_shift,
|
|
};
|
|
glcr::RefPtr<VariableMemoryObject> ubuf_vmmo =
|
|
glcr::MakeRefCounted<VariableMemoryObject>(sizeof(ubuf));
|
|
ubuf_vmmo->CopyBytesToObject(reinterpret_cast<uint64_t>(&ubuf), sizeof(ubuf));
|
|
port->WriteKernel(Z_BOOT_FRAMEBUFFER_INFO_VMMO,
|
|
MakeRefCounted<Capability>(ubuf_vmmo));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void LoadInitProgram() {
|
|
DumpModules();
|
|
|
|
// Create process.
|
|
glcr::RefPtr<Process> proc = Process::Create();
|
|
gProcMan->InsertProcess(proc);
|
|
|
|
// Write init data.
|
|
auto port = glcr::MakeRefCounted<Port>();
|
|
uint64_t port_cap = proc->AddNewCapability(port);
|
|
port->WriteKernel(Z_INIT_SELF_PROC, MakeRefCounted<Capability>(proc));
|
|
port->WriteKernel(Z_INIT_SELF_VMAS, MakeRefCounted<Capability>(proc->vmas()));
|
|
WriteInitProgram(port, "/sys/denali", Z_BOOT_DENALI_VMMO);
|
|
WriteInitProgram(port, "/sys/victoriafalls", Z_BOOT_VICTORIA_FALLS_VMMO);
|
|
|
|
WriteFramebufferVmmo(port);
|
|
|
|
if (WritePciVmmo(port, Z_BOOT_PCI_VMMO) != glcr::OK) {
|
|
panic("Failed to provide PCI info to init.");
|
|
}
|
|
|
|
// Start process.
|
|
limine_file init_prog = GetInitProgram("/sys/yellowstone");
|
|
uint64_t entry = LoadElfProgram(
|
|
*proc, reinterpret_cast<uint64_t>(init_prog.address), init_prog.size);
|
|
proc->CreateThread()->Start(entry, port_cap, 0);
|
|
}
|