1
0
Fork 0
mirror of https://github.com/tailix/kernel.git synced 2024-12-11 11:35:39 -05:00

Revert "Remove page frame allocator"

This reverts commit db8a61a779.
This commit is contained in:
Alex Kotov 2020-11-25 23:55:53 +05:00
parent d9aeead3b6
commit 344226cb7a
Signed by: kotovalexarian
GPG key ID: 553C0EBBEB5D5F08
4 changed files with 129 additions and 0 deletions

View file

@ -14,6 +14,7 @@ OBJS += main.c.o
OBJS += init.c.o
OBJS += multiboot.c.o
OBJS += panic.c.o panic.asm.cpp.o
OBJS += pfa.c.o
OBJS += paging.c.o paging.asm.cpp.o
OBJS += page_dir.c.o

View file

@ -1,4 +1,5 @@
#include "panic.h"
#include "pfa.h"
#include "protected.h"
#include "paging.h"
@ -18,6 +19,8 @@ void init(const struct KernelMQ_Info *const kinfo_ptr)
assert(kernelmq_info_validate_and_copy(&kinfo, kinfo_ptr), "Invalid kernel information.");
pfa_initialize(&kinfo);
protected_initialize(&kinfo);
// Set up a new post-relocate bootstrap pagetable so that

111
kernelmq/pfa.c Normal file
View file

@ -0,0 +1,111 @@
#include "pfa.h"
#include "config.h"
#include "panic.h"
#include "logger.h"
#include "stdlib.h"
#define FRAMES_COUNT (PAGE_DIR_LENGTH * PAGE_TABLE_LENGTH)
static unsigned char frames[FRAMES_COUNT];
static void mark_used(unsigned long base, unsigned long limit);
void pfa_initialize(const struct KernelMQ_Info *const kinfo)
{
logger_info_from("pfa", "Initialize page frame allocator.");
kmemset(frames, 0, sizeof(frames));
mark_used(0, MEM_UPPER_BASE - 1);
mark_used(kinfo->kernel_phys_base, kinfo->kernel_phys_limit);
for (unsigned int i = 0; i < kinfo->modules_count; ++i) {
const struct KernelMQ_Info_Module *const module = &kinfo->modules[i];
mark_used(module->base, module->limit);
}
for (unsigned int i = 0; i < kinfo->areas_count; ++i) {
const struct KernelMQ_Info_Area *const area = &kinfo->areas[i];
if (!area->is_available) {
mark_used(area->base, area->limit);
}
}
}
unsigned long pfa_alloc_page()
{
for (unsigned int i = 0; i < FRAMES_COUNT; ++i) {
if (!frames[i]) {
frames[i] = 0xFF;
return i * PAGE_SIZE;
}
}
return 0;
}
unsigned long pfa_alloc_big_page()
{
unsigned int start = 0;
for (unsigned int i = 0; i < FRAMES_COUNT; ++i) {
if (frames[i]) {
start = i + 1;
continue;
}
if (start % (PAGE_BIG_SIZE / PAGE_SIZE)) {
continue;
}
if (i - start + 1 == PAGE_BIG_SIZE / PAGE_SIZE) {
for (unsigned int j = start; j <= i; ++j) {
frames[j] = 0xFF;
}
return start * PAGE_SIZE;
}
}
return 0;
}
void pfa_free_page(const unsigned long base)
{
assert(!(base % PAGE_SIZE), "Small page address to free is not aligned.");
const unsigned long i = base / PAGE_SIZE;
if (i >= FRAMES_COUNT) {
return;
}
frames[i] = 0;
}
void pfa_free_big_page(const unsigned long base)
{
assert(!(base % PAGE_BIG_SIZE), "Big page address to free is not aligned.");
const unsigned long start = base / PAGE_SIZE;
const unsigned long end = start + PAGE_BIG_SIZE / PAGE_SIZE;
for (unsigned int i = start; i <= end && i < FRAMES_COUNT; ++i) {
frames[i] = 0;
}
}
void mark_used(const unsigned long base, const unsigned long limit)
{
const unsigned int start = base / PAGE_SIZE;
const unsigned int end = limit / PAGE_SIZE;
for (unsigned int i = start; i <= end; ++i) {
frames[i] = 0xFF;
}
}

14
kernelmq/pfa.h Normal file
View file

@ -0,0 +1,14 @@
#ifndef KERNELMQ_INCLUDED_PFA
#define KERNELMQ_INCLUDED_PFA 1
#include "info.h"
void pfa_initialize(const struct KernelMQ_Info *kinfo);
unsigned long pfa_alloc_page();
unsigned long pfa_alloc_big_page();
void pfa_free_page(unsigned long base);
void pfa_free_big_page(unsigned long base);
#endif