aboutsummaryrefslogtreecommitdiff
path: root/libc/mem.c
diff options
context:
space:
mode:
authorMarvin Borner2021-02-27 18:24:38 +0100
committerMarvin Borner2021-02-27 18:24:38 +0100
commit7304e20731980078a7bfe138a20a8d13653fed7b (patch)
treefbfce00a6a8017f842e8a165b563ab15a094eb84 /libc/mem.c
parent4309322f9d2b3e31421a3cc5399ab1f4368e0652 (diff)
Started basic paging port from skiftOS
Diffstat (limited to 'libc/mem.c')
-rw-r--r--libc/mem.c351
1 files changed, 2 insertions, 349 deletions
diff --git a/libc/mem.c b/libc/mem.c
index 971315a..95242e4 100644
--- a/libc/mem.c
+++ b/libc/mem.c
@@ -66,12 +66,13 @@ void *memcpy(void *dest, const void *src, u32 n)
void *memset(void *dest, int val, u32 n)
{
+ u32 uval = val;
u32 num_dwords = n / 4;
u32 num_bytes = n % 4;
u32 *dest32 = (u32 *)dest;
u8 *dest8 = ((u8 *)dest) + num_dwords * 4;
u8 val8 = (u8)val;
- u32 val32 = val | (val << 8) | (val << 16) | (val << 24);
+ u32 val32 = uval | (uval << 8) | (uval << 16) | (uval << 24);
// TODO: What's faster?
__asm__ volatile("rep stosl\n"
@@ -118,351 +119,3 @@ int mememp(const u8 *buf, u32 n)
{
return buf[0] == 0 && !memcmp(buf, buf + 1, n - 1);
}
-
-/**
- * Heap allocator
- * Inspired by SHMALL (MIT License)
- * Copyright (c) 2017 Chris Careaga
- * Copyright (c) 2021 Marvin Borner
- */
-
-#ifdef kernel
-
-#define HEAP_MAGIC 0x424242
-#define HEAP_INIT_SIZE 0xf000000
-#define HEAP_MIN_SIZE HEAP_INIT_SIZE
-#define MIN_ALLOC_SZ 4
-#define BIN_COUNT 9
-#define BIN_MAX_IDX (BIN_COUNT - 1)
-#define OVERHEAD (sizeof(struct h_footer) + sizeof(struct h_node))
-/* #define MIN_WILDERNESS 0x2000 */
-/* #define MAX_WILDERNESS 0x1000000 */
-
-struct h_node {
- u32 magic;
- u32 hole;
- u32 size;
- struct h_node *next;
- struct h_node *prev;
-};
-
-struct h_footer {
- struct h_node *header;
-};
-
-struct h_bin {
- struct h_node *head;
-};
-
-struct heap {
- u32 start;
- u32 end;
- struct h_bin bins[BIN_COUNT];
-};
-
-static void node_add(struct h_bin *bin, struct h_node *node)
-{
- node->magic = HEAP_MAGIC;
- node->next = NULL;
- node->prev = NULL;
- if (!bin->head) {
- bin->head = node;
- return;
- }
- struct h_node *curr = bin->head;
- struct h_node *prev = NULL;
- while (curr && curr->size <= node->size) {
- prev = curr;
- curr = curr->next;
- }
- if (!curr) {
- prev->next = node;
- node->prev = prev;
- } else if (prev) {
- node->next = curr;
- prev->next = node;
- node->prev = prev;
- curr->prev = node;
- } else {
- node->next = bin->head;
- bin->head->prev = node;
- bin->head = node;
- }
-}
-
-static void node_remove(struct h_bin *bin, struct h_node *node)
-{
- if (!bin->head)
- return;
- if (bin->head == node) {
- bin->head = bin->head->next;
- return;
- }
-
- struct h_node *temp = bin->head->next;
- while (temp) {
- if (temp == node) {
- if (!temp->next) {
- temp->prev->next = NULL;
- } else {
- temp->prev->next = temp->next;
- temp->next->prev = temp->prev;
- }
- return;
- }
- temp = temp->next;
- }
-}
-
-static struct h_node *node_best_fit(struct h_bin *bin, u32 size)
-{
- if (!bin->head)
- return NULL;
-
- struct h_node *temp = bin->head;
- while (temp) {
- if (temp->size >= size)
- return temp;
- temp = temp->next;
- }
- return NULL;
-}
-
-/* static struct h_node *node_last(struct h_bin *bin) */
-/* { */
-/* struct h_node *temp = bin->head; */
-/* while (temp->next) */
-/* temp = temp->next; */
-/* return temp; */
-/* } */
-
-static struct h_footer *node_foot(struct h_node *node)
-{
- return (struct h_footer *)((char *)node + sizeof(*node) + node->size);
-}
-
-static void node_create_foot(struct h_node *head)
-{
- struct h_footer *foot = node_foot(head);
- foot->header = head;
-}
-
-static u32 bin_index(u32 sz)
-{
- u32 index = 0;
- sz = sz < 4 ? 4 : sz;
- while (sz >>= 1)
- index++;
- index -= 2;
- if (index > BIN_MAX_IDX)
- index = BIN_MAX_IDX;
- return index;
-}
-
-/* struct h_node *wilderness_get(struct heap *heap) */
-/* { */
-/* struct h_footer *wild_foot = (struct h_footer *)((char *)heap->end - sizeof(*wild_foot)); */
-/* return wild_foot->header; */
-/* } */
-
-/* static u32 expand(struct heap *heap, u32 sz) */
-/* { */
-/* (void)heap; */
-/* (void)sz; */
-/* return 0; */
-/* } */
-
-/* static u32 contract(struct heap *heap, u32 sz) */
-/* { */
-/* (void)heap; */
-/* (void)sz; */
-/* return 0; */
-/* } */
-
-static struct heap heap = { 0 };
-void heap_init(u32 start)
-{
- struct h_node *init_region = (struct h_node *)start;
- init_region->hole = 1;
- init_region->size = HEAP_INIT_SIZE - OVERHEAD;
- node_create_foot(init_region);
- node_add(&heap.bins[bin_index(init_region->size)], init_region);
- heap.start = (u32)start;
- heap.end = (u32)start + HEAP_INIT_SIZE;
-}
-
-#define ALIGN sizeof(long)
-static void *_malloc(u32 size)
-{
- size = ((size + ALIGN - 1) / ALIGN) * ALIGN; // Alignment
- u32 index = bin_index(size);
- struct h_bin *temp = (struct h_bin *)&heap.bins[index];
- struct h_node *found = node_best_fit(temp, size);
-
- while (!found) {
- assert(index + 1 < BIN_COUNT);
-
- temp = &heap.bins[++index];
- found = node_best_fit(temp, size);
- }
-
- assert(found->magic == HEAP_MAGIC);
-
- if ((found->size - size) > (OVERHEAD + MIN_ALLOC_SZ)) {
- struct h_node *split = (struct h_node *)(((char *)found + OVERHEAD) + size);
- split->magic = HEAP_MAGIC;
- split->size = found->size - size - OVERHEAD;
- split->hole = 1;
-
- node_create_foot(split);
-
- u32 new_idx = bin_index(split->size);
-
- node_add(&heap.bins[new_idx], split);
-
- found->size = size;
- node_create_foot(found);
- }
-
- found->hole = 0;
- node_remove(&heap.bins[index], found);
-
- // TODO: Implement expand/contract
- /* struct h_node *wild = wilderness_get(&heap); */
- /* if (wild->size < MIN_WILDERNESS) { */
- /* assert(expand(&heap, 0x1000)); */
- /* } else if (wild->size > MAX_WILDERNESS) { */
- /* assert(contract(&heap, 0x1000)); */
- /* } */
-
- found->prev = NULL;
- found->next = NULL;
- return &found->next;
-}
-
-static void _free(void *p)
-{
- if (!p)
- return;
-
- struct h_bin *list;
- struct h_footer *new_foot, *old_foot;
-
- struct h_node *head = (struct h_node *)((char *)p - 12);
- assert(head->magic == HEAP_MAGIC && head->hole == 0);
- if (head == (struct h_node *)(u32 *)heap.start) {
- head->hole = 1;
- node_add(&heap.bins[bin_index(head->size)], head);
- return;
- }
-
- struct h_node *next = (struct h_node *)((char *)node_foot(head) + sizeof(struct h_footer));
- struct h_footer *f = (struct h_footer *)((char *)head - sizeof(struct h_footer));
- struct h_node *prev = f->header;
-
- if (prev->hole) {
- list = &heap.bins[bin_index(prev->size)];
- node_remove(list, prev);
-
- prev->size += OVERHEAD + head->size;
- new_foot = node_foot(head);
- new_foot->header = prev;
-
- head = prev;
- }
-
- if (next->hole) {
- list = &heap.bins[bin_index(next->size)];
- node_remove(list, next);
-
- head->size += OVERHEAD + next->size;
-
- old_foot = node_foot(next);
- old_foot->header = 0;
- next->size = 0;
- next->hole = 0;
-
- new_foot = node_foot(head);
- new_foot->header = head;
- }
-
- head->hole = 1;
- node_add(&heap.bins[bin_index(head->size)], head);
-}
-
-#elif defined(userspace)
-
-#define kmalloc(n) (void *)sys1(SYS_MALLOC, n)
-#define kfree(ptr) (void)(sys1(SYS_FREE, (int)ptr))
-
-static void *_malloc(u32 size)
-{
- return kmalloc(size);
-}
-
-static void _free(void *ptr)
-{
- kfree(ptr);
-}
-
-#endif
-
-#ifdef kernel
-#define PREFIX "K"
-#define FUNC printf
-#else
-#define PREFIX "U"
-#define FUNC log
-#endif
-
-void *zalloc(u32 size)
-{
- void *ret = malloc(size);
- memset(ret, 0, size);
- return ret;
-}
-
-// Naive realloc implementation - TODO!
-void *realloc(void *ptr, u32 size)
-{
- if (!ptr)
- return malloc(size);
-
- FUNC("Realloc not implemented!\n");
- return NULL;
- /* // This could work; untested
- struct h_node *node = (struct h_node *)((char *)ptr - 12);
- u32 old_size = node->size;
-
- void *new = malloc(size);
- memcpy(new, ptr, old_size);
-
- free(ptr);
- return new;
- */
-}
-
-void *malloc_debug(u32 size, const char *file, int line, const char *func, const char *inp)
-{
- assert(size < (100 << 20)); // Don't brag with memory pls
- void *ret = _malloc(size);
-
- (void)file;
- (void)line;
- (void)func;
- (void)inp;
- /* FUNC(PREFIX "MALLOC\t%s:%d: %s: 0x%x %dB (%s)\n", file, line, func, ret, size, inp); */
- return ret;
-}
-
-void free_debug(void *ptr, const char *file, int line, const char *func, const char *inp)
-{
- if (ptr)
- _free(ptr);
-
- (void)file;
- (void)line;
- (void)func;
- (void)inp;
- /* FUNC(PREFIX "FREE\t%s:%d: %s: 0x%x (%s)\n", file, line, func, ptr, inp); */
-}