2026-03-29 10:40:09 -06:00
|
|
|
#include "arena.h"
|
2026-03-29 17:53:10 -06:00
|
|
|
#include <stdbool.h>
|
|
|
|
|
#include <stdlib.h>
|
2026-03-29 18:35:13 -06:00
|
|
|
#include <string.h>
|
2026-03-29 19:06:49 -06:00
|
|
|
#include <sys/types.h>
|
2026-03-29 17:53:10 -06:00
|
|
|
|
2026-03-29 18:00:57 -06:00
|
|
|
ArenaResult arena_init(size_t size) {
|
|
|
|
|
|
|
|
|
|
void *buffer = malloc(size);
|
2026-03-29 17:53:10 -06:00
|
|
|
if (buffer == NULL) {
|
|
|
|
|
ArenaResult err = {
|
|
|
|
|
.is_valid = false,
|
2026-03-29 18:00:57 -06:00
|
|
|
.err = ARENA_BAD_ALLOC,
|
2026-03-29 17:53:10 -06:00
|
|
|
};
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (size < 1) {
|
|
|
|
|
ArenaResult err = {
|
|
|
|
|
.is_valid = false,
|
|
|
|
|
.err = ARENA_INVALID_SIZE,
|
|
|
|
|
};
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Arena new_arena = {
|
2026-03-29 19:06:49 -06:00
|
|
|
.buffer = buffer,
|
2026-03-29 17:53:10 -06:00
|
|
|
.size = size,
|
|
|
|
|
.offset = 0,
|
|
|
|
|
};
|
|
|
|
|
ArenaResult val = {
|
|
|
|
|
.is_valid = true,
|
|
|
|
|
.arena = new_arena,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return val;
|
|
|
|
|
}
|
2026-03-29 18:00:57 -06:00
|
|
|
|
|
|
|
|
void arena_destroy(Arena *arena) {
|
|
|
|
|
if (arena == NULL) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
arena->offset = 0;
|
|
|
|
|
arena->size = 0;
|
2026-03-29 19:06:49 -06:00
|
|
|
free(arena->buffer);
|
2026-03-29 18:00:57 -06:00
|
|
|
}
|
2026-03-29 18:35:13 -06:00
|
|
|
|
|
|
|
|
ArenaPointer arena_alloc(Arena *arena, size_t size, size_t alignment) {
|
|
|
|
|
if (arena == NULL) {
|
|
|
|
|
ArenaPointer err = {
|
|
|
|
|
.is_valid = false,
|
|
|
|
|
.err = ARENA_NULL_ARG,
|
|
|
|
|
};
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t new_offset = align_arena_offset(arena, alignment);
|
|
|
|
|
if (new_offset + size >= arena->size) {
|
|
|
|
|
ArenaPointer err = {
|
|
|
|
|
.is_valid = false,
|
|
|
|
|
.err = ARENA_OUT_OF_SPACE,
|
|
|
|
|
};
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
arena->offset = new_offset;
|
|
|
|
|
|
|
|
|
|
ArenaPointer val = {
|
|
|
|
|
.is_valid = true,
|
2026-03-29 19:06:49 -06:00
|
|
|
.address = arena->buffer + arena->offset,
|
2026-03-29 18:35:13 -06:00
|
|
|
};
|
|
|
|
|
return val;
|
|
|
|
|
}
|
2026-03-29 19:06:49 -06:00
|
|
|
|
|
|
|
|
ArenaErr arena_push(Arena *arena, void *data, size_t size, size_t alignment) {
|
|
|
|
|
if (arena == NULL || data == NULL) {
|
|
|
|
|
return ARENA_NULL_ARG;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ArenaPointer pointer = arena_alloc(arena, size, alignment);
|
|
|
|
|
|
|
|
|
|
if (!pointer.is_valid) {
|
|
|
|
|
size_t new_size = arena->size * 2;
|
|
|
|
|
ArenaErr err = arena_realloc(arena, new_size);
|
|
|
|
|
if (err != ARENA_OK) {
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
memcpy(
|
|
|
|
|
pointer.address,
|
|
|
|
|
data,
|
|
|
|
|
size);
|
|
|
|
|
|
|
|
|
|
return ARENA_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ArenaErr arena_realloc(Arena *arena, size_t new_size) {
|
|
|
|
|
if (arena == NULL) {
|
|
|
|
|
return ARENA_NULL_ARG;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t *tmp = realloc(arena->buffer, new_size);
|
|
|
|
|
if (tmp == NULL) {
|
|
|
|
|
return ARENA_BAD_ALLOC;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
arena->buffer = tmp;
|
|
|
|
|
arena->size = new_size;
|
|
|
|
|
return ARENA_OK;
|
|
|
|
|
}
|