summaryrefslogtreecommitdiff
path: root/src/kernel/malloc.c
blob: 44434ed6c28d2dab621e73bbc67dd0ead26eac8c (plain)
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
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
#include <kernel/arch/generic.h>
#include <kernel/malloc.h>
#include <kernel/panic.h>
#include <kernel/util.h>
#include <shared/mem.h>
#include <stdbool.h>
#include <stdint.h>

#define MALLOC_MAGIC 0x616c6c6f63686472 /* "allochdr" */
#define DESCLEN 8

typedef struct Allocation Allocation;
struct Allocation {
	uint64_t magic;
	uint64_t len;
	Allocation *next, *prev;
	void *stacktrace[4];
	char desc[DESCLEN];
};

static Allocation *malloc_last = NULL;

static size_t
page_amt(size_t bytes)
{
	return (bytes + PAGE_SIZE - 1) / PAGE_SIZE;
}

void
kmalloc_debugprint(void)
{
	size_t count = 0, bytes = 0, pages = 0;
	kprintf("[kern] current allocations:\n");
	for (Allocation *iter = malloc_last; iter; iter = iter->prev) {
		kprintf(
			"%08p %6dB %.8s ",
			((void*)iter) + sizeof(Allocation),
			iter->len - sizeof(Allocation),
			iter->desc
		);
		for (size_t i = 0; i < 4; i++) {
			kprintf(" k/%08x", iter->stacktrace[i]);
		}
		kprintf("\n");

		count++;
		bytes += iter->len;
		pages += page_amt(iter->len);
	}
	kprintf(
		"%d in total, %d bytes, %d pages = %dB used\n",
		count, bytes, pages, pages*PAGE_SIZE
	);
}

static void
kmalloc_sanity(const void *addr)
{
	assert(addr);
	const Allocation *hdr = addr - sizeof(Allocation);
	assert(hdr->magic == MALLOC_MAGIC);
	if (hdr->next) assert(hdr->next->prev == hdr);
	if (hdr->prev) assert(hdr->prev->next == hdr);
}

void *
kmalloc(size_t len, const char *desc)
{
	Allocation *hdr;
	void *addr;

	if (KMALLOC_MAX < len) {
		panic_invalid_state();
	}
	len += sizeof(Allocation);
	assert(len <= PAGE_SIZE);
	hdr = page_alloc(page_amt(len));
	hdr->magic = MALLOC_MAGIC;
	hdr->len = len;

	memset(hdr->desc, ' ', DESCLEN);
	if (desc) {
		for (int i = 0; i < DESCLEN; i++) {
			if (desc[i] == '\0') break;
			hdr->desc[i] = desc[i];
		}
	}

	hdr->next = NULL;
	hdr->prev = malloc_last;
	if (hdr->prev) {
		assert(!hdr->prev->next);
		hdr->prev->next = hdr;
	}

	for (size_t i = 0; i < 4; i++)
		hdr->stacktrace[i] = debug_caller(i);

	malloc_last = hdr;

	addr = (void*)hdr + sizeof(Allocation);
#ifndef NDEBUG
	memset(addr, 0xCC, len);
#endif
	kmalloc_sanity(addr);
	return addr;
}

void
kfree(void *ptr)
{
	Allocation *hdr;
	size_t len;
	if (ptr == NULL) return;

	kmalloc_sanity(ptr);
	hdr = ptr - sizeof(Allocation);
	len = hdr->len;

	if (hdr->next)
		hdr->next->prev = hdr->prev;
	if (hdr->prev)
		hdr->prev->next = hdr->next;
	if (malloc_last == hdr)
		malloc_last = hdr->prev;
#ifndef NDEBUG
	memset(hdr, 0xC0, len);
#else
	hdr->magic = ~MALLOC_MAGIC; /* (hopefully) detect double frees */
#endif
	page_free(hdr, page_amt(len));
}