123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153 |
- package mem_virtual
- import "core:mem"
- Static_Arena :: struct {
- block: ^Memory_Block,
- total_used: uint,
- total_reserved: uint,
-
- minimum_block_size: uint,
- temp_count: int,
- }
- STATIC_ARENA_DEFAULT_COMMIT_SIZE :: 1<<20 // 1 MiB should be enough to start with
- // 1 GiB on 64-bit systems, 128 MiB on 32-bit systems by default
- STATIC_ARENA_DEFAULT_RESERVE_SIZE :: 1<<30 when size_of(uintptr) == 8 else 1<<27
- static_arena_init :: proc(arena: ^Static_Arena, reserved: uint, commit_size: uint = STATIC_ARENA_DEFAULT_COMMIT_SIZE) -> (err: Allocator_Error) {
- arena.block = memory_block_alloc(commit_size, reserved, {}) or_return
- arena.total_used = 0
- arena.total_reserved = arena.block.reserved
- return
- }
- static_arena_destroy :: proc(arena: ^Static_Arena) {
- memory_block_dealloc(arena.block)
- arena^ = {}
- }
- static_arena_alloc :: proc(arena: ^Static_Arena, size: int, alignment: int) -> (data: []byte, err: Allocator_Error) {
- align_forward :: #force_inline proc "contextless" (ptr: uint, align: uint) -> uint {
- mask := align-1
- return (ptr + mask) &~ mask
- }
-
- if arena.block == nil {
- reserve_size := max(arena.minimum_block_size, STATIC_ARENA_DEFAULT_RESERVE_SIZE)
- static_arena_init(arena, reserve_size, STATIC_ARENA_DEFAULT_COMMIT_SIZE) or_return
- }
-
- MINIMUM_ALIGN :: 2*align_of(uintptr)
-
- defer arena.total_used = arena.block.used
- return alloc_from_memory_block(arena.block, size, max(MINIMUM_ALIGN, alignment))
- }
- static_arena_reset_to :: proc(arena: ^Static_Arena, pos: uint) -> bool {
- if arena.block != nil {
- prev_pos := arena.block.used
- arena.block.used = clamp(pos, 0, arena.block.reserved)
-
- if prev_pos < pos {
- mem.zero_slice(arena.block.base[arena.block.used:][:pos-prev_pos])
- }
- return true
- } else if pos == 0 {
- return true
- }
- return false
- }
- static_arena_free_all :: proc(arena: ^Static_Arena) {
- static_arena_reset_to(arena, 0)
- }
- static_arena_bootstrap_new_by_offset :: proc($T: typeid, offset_to_arena: uintptr, reserved: uint) -> (ptr: ^T, err: Allocator_Error) {
- bootstrap: Static_Arena
- bootstrap.minimum_block_size = reserved
-
- data := static_arena_alloc(&bootstrap, size_of(T), align_of(T)) or_return
-
- ptr = (^T)(raw_data(data))
-
- (^Static_Arena)(uintptr(ptr) + offset_to_arena)^ = bootstrap
-
- return
- }
- static_arena_bootstrap_new_by_name :: proc($T: typeid, $field_name: string, reserved: uint) -> (ptr: ^T, err: Allocator_Error) {
- return static_arena_bootstrap_new_by_offset(T, offset_of_by_string(T, field_name), reserved)
- }
- static_arena_bootstrap_new :: proc{
- static_arena_bootstrap_new_by_offset,
- static_arena_bootstrap_new_by_name,
- }
- static_arena_allocator :: proc(arena: ^Static_Arena) -> mem.Allocator {
- return mem.Allocator{static_arena_allocator_proc, arena}
- }
- static_arena_allocator_proc :: proc(allocator_data: rawptr, mode: mem.Allocator_Mode,
- size, alignment: int,
- old_memory: rawptr, old_size: int,
- location := #caller_location) -> (data: []byte, err: Allocator_Error) {
- arena := (^Static_Arena)(allocator_data)
-
- switch mode {
- case .Alloc:
- return static_arena_alloc(arena, size, alignment)
- case .Free:
- err = .Mode_Not_Implemented
- return
- case .Free_All:
- static_arena_free_all(arena)
- return
- case .Resize:
- return mem.default_resize_bytes_align(mem.byte_slice(old_memory, old_size), size, alignment, static_arena_allocator(arena), location)
-
- case .Query_Features, .Query_Info:
- err = .Mode_Not_Implemented
- return
- }
-
- err = .Mode_Not_Implemented
- return
- }
- Static_Arena_Temp :: struct {
- arena: ^Static_Arena,
- used: uint,
- }
- static_arena_temp_begin :: proc(arena: ^Static_Arena) -> (temp: Static_Arena_Temp) {
- temp.arena = arena
- temp.used = arena.block.used if arena.block != nil else 0
- arena.temp_count += 1
- return
- }
- static_arena_temp_end :: proc(temp: Static_Arena_Temp, loc := #caller_location) {
- assert(temp.arena != nil, "nil arena", loc)
- arena := temp.arena
-
- used := arena.block.used if arena.block != nil else 0
-
- assert(temp.used >= used, "invalid Static_Arena_Temp", loc)
-
- static_arena_reset_to(arena, temp.used)
-
- assert(arena.temp_count > 0, "double-use of static_arena_temp_end", loc)
- arena.temp_count -= 1
- }
- static_arena_check_temp :: proc(arena: ^Static_Arena, loc := #caller_location) {
- assert(arena.temp_count == 0, "Static_Arena_Temp not been ended", loc)
- }
|