aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/mm/pmm.cpp
blob: d25e673e6dc88be9afe8ec647713463d62f312bc (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#include <mm/pmm.h>
#include <kernel.h>
#include <stdio.h>

#define PAGE_SIZE 0x1000ul
// makes sure double frees aren't happening (probably)
#define PMM_MAGIC_NUM 0xE1321ul

struct freelist {
    freelist* next = nullptr;
    uint64_t magic_num;
};

freelist* head = nullptr;

void pmm_init(limine_memmap_response* memmap) {

    for (int i = 0; i < memmap->entry_count; i++) {
        // only maps usable entries
        if (memmap->entries[i]->type == LIMINE_MEMMAP_USABLE) {
            printf("%d\n", i);
            // iterates through all addresses
            for (uint64_t addr = memmap->entries[i]->base; addr < (memmap->entries[i]->length + memmap->entries[i]->base); addr += PAGE_SIZE) {
                freelist* temp = (freelist*)(addr + _hhdm_offset);
                temp->magic_num = PMM_MAGIC_NUM;
                if (head) {
                    temp->next = head;
                    head = temp;
                } else {
                    temp->next = nullptr;
                    head = temp;
                }
            }
        }
    }

    printf("Initialized PMM\n");
}

uint64_t page_alloc() {
    freelist* alloced = head;
    alloced->magic_num = 0;
    head = head->next;
    return (uint64_t)alloced;
}

void page_free(uint64_t page) {
    freelist* freed = (freelist*)page;

    // double free check
    if (freed->magic_num == PMM_MAGIC_NUM) {
        return;
    }

    freed->next = head;
    freed->magic_num = PMM_MAGIC_NUM;
    head = freed;
}