allocators.odin 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. #+private
  2. package os2
  3. import "base:runtime"
  4. @(require_results)
  5. file_allocator :: proc() -> runtime.Allocator {
  6. return heap_allocator()
  7. }
  8. temp_allocator_proc :: runtime.arena_allocator_proc
  9. @(private="file")
  10. MAX_TEMP_ARENA_COUNT :: 2
  11. @(private="file", thread_local)
  12. global_default_temp_allocator_arenas: [MAX_TEMP_ARENA_COUNT]runtime.Arena
  13. @(private="file", thread_local)
  14. global_default_temp_allocator_index: uint
  15. @(require_results)
  16. temp_allocator :: proc() -> runtime.Allocator {
  17. arena := &global_default_temp_allocator_arenas[global_default_temp_allocator_index]
  18. if arena.backing_allocator.procedure == nil {
  19. arena.backing_allocator = heap_allocator()
  20. }
  21. return runtime.Allocator{
  22. procedure = temp_allocator_proc,
  23. data = arena,
  24. }
  25. }
  26. @(require_results)
  27. temp_allocator_temp_begin :: proc(loc := #caller_location) -> (temp: runtime.Arena_Temp) {
  28. temp = runtime.arena_temp_begin(&global_default_temp_allocator_arenas[global_default_temp_allocator_index], loc)
  29. return
  30. }
  31. temp_allocator_temp_end :: proc(temp: runtime.Arena_Temp, loc := #caller_location) {
  32. runtime.arena_temp_end(temp, loc)
  33. }
  34. @(fini, private)
  35. temp_allocator_fini :: proc() {
  36. for &arena in global_default_temp_allocator_arenas {
  37. runtime.arena_destroy(&arena)
  38. }
  39. global_default_temp_allocator_arenas = {}
  40. }
  41. TEMP_ALLOCATOR_GUARD_END :: proc(temp: runtime.Arena_Temp, loc := #caller_location) {
  42. runtime.arena_temp_end(temp, loc)
  43. if temp.arena != nil {
  44. global_default_temp_allocator_index = (global_default_temp_allocator_index-1)%MAX_TEMP_ARENA_COUNT
  45. }
  46. }
  47. @(deferred_out=TEMP_ALLOCATOR_GUARD_END)
  48. TEMP_ALLOCATOR_GUARD :: #force_inline proc(loc := #caller_location) -> (runtime.Arena_Temp, runtime.Source_Code_Location) {
  49. global_default_temp_allocator_index = (global_default_temp_allocator_index+1)%MAX_TEMP_ARENA_COUNT
  50. tmp := temp_allocator_temp_begin(loc)
  51. return tmp, loc
  52. }
  53. @(init, private)
  54. init_thread_local_cleaner :: proc() {
  55. runtime.add_thread_local_cleaner(temp_allocator_fini)
  56. }