summaryrefslogtreecommitdiff
path: root/Kernel/Heap/kmalloc.h
diff options
context:
space:
mode:
authorTom <tomut@yahoo.com>2020-08-29 16:41:30 -0600
committerAndreas Kling <kling@serenityos.org>2020-08-30 11:39:38 +0200
commit4b66692a553df89d21d7709e47852e368d8eaacf (patch)
treeff4f2b47ee0f2263deeedae82122f0e71392da03 /Kernel/Heap/kmalloc.h
parent8ecc3d31d1072520e4df332c3fad9071a5f94dee (diff)
downloadserenity-4b66692a553df89d21d7709e47852e368d8eaacf.zip
Kernel: Make Heap implementation reusable, and make kmalloc expandable
Add an ExpandableHeap and switch kmalloc to use it, which allows for the kmalloc heap to grow as needed. In order to make heap expansion to work, we keep around a 1 MiB backup memory region, because creating a region would require space in the same heap. This means, the heap will grow as soon as the reported utilization is less than 1 MiB. It will also return memory if an entire subheap is no longer needed, although that is rarely possible.
Diffstat (limited to 'Kernel/Heap/kmalloc.h')
-rw-r--r--Kernel/Heap/kmalloc.h37
1 files changed, 29 insertions, 8 deletions
diff --git a/Kernel/Heap/kmalloc.h b/Kernel/Heap/kmalloc.h
index 75f0fab642..ced23e7146 100644
--- a/Kernel/Heap/kmalloc.h
+++ b/Kernel/Heap/kmalloc.h
@@ -36,17 +36,19 @@
void kmalloc_init();
[[gnu::malloc, gnu::returns_nonnull, gnu::alloc_size(1)]] void* kmalloc_impl(size_t);
[[gnu::malloc, gnu::returns_nonnull, gnu::alloc_size(1)]] void* kmalloc_eternal(size_t);
-[[gnu::malloc, gnu::returns_nonnull, gnu::alloc_size(1)]] void* kmalloc_page_aligned(size_t);
-[[gnu::malloc, gnu::returns_nonnull, gnu::alloc_size(1)]] void* kmalloc_aligned(size_t, size_t alignment);
+
void* krealloc(void*, size_t);
void kfree(void*);
-void kfree_aligned(void*);
-extern size_t g_kmalloc_bytes_allocated;
-extern size_t g_kmalloc_bytes_free;
-extern size_t g_kmalloc_bytes_eternal;
-extern size_t g_kmalloc_call_count;
-extern size_t g_kfree_call_count;
+struct kmalloc_stats {
+ size_t bytes_allocated;
+ size_t bytes_free;
+ size_t bytes_eternal;
+ size_t kmalloc_call_count;
+ size_t kfree_call_count;
+};
+void get_kmalloc_stats(kmalloc_stats&);
+
extern bool g_dump_kmalloc_stacks;
inline void* operator new(size_t, void* p) { return p; }
@@ -62,5 +64,24 @@ inline void* operator new[](size_t, void* p) { return p; }
return kmalloc_impl(size);
}
+template<size_t ALIGNMENT>
+[[gnu::malloc, gnu::returns_nonnull, gnu::alloc_size(1)]] inline void* kmalloc_aligned(size_t size)
+{
+ static_assert(ALIGNMENT > 1);
+ static_assert(ALIGNMENT < 255);
+ void* ptr = kmalloc(size + ALIGNMENT + sizeof(u8));
+ size_t max_addr = (size_t)ptr + ALIGNMENT;
+ void* aligned_ptr = (void*)(max_addr - (max_addr % ALIGNMENT));
+ ((u8*)aligned_ptr)[-1] = (u8)((u8*)aligned_ptr - (u8*)ptr);
+ return aligned_ptr;
+}
+
+inline void kfree_aligned(void* ptr)
+{
+ kfree((u8*)ptr - ((u8*)ptr)[-1]);
+}
+
+void kmalloc_enable_expand();
+
extern u8* const kmalloc_start;
extern u8* const kmalloc_end;