Exponential Array (Xar). A dynamically growing array using exponentially-sized chunks, providing stable memory addresses for all elements. Unlike `[dynamic]T`, elements are never moved once allocated, making it safe to hold pointers to elements. For more information: https://azmr.uk/dyn/#exponential-arrayxar Example: import "core:container/xar" example :: proc() { x: xar.Array(int, 4) defer xar.destroy(&x) xar.push_back(&x, 10) xar.push_back(&x, 20) xar.push_back(&x, 30) ptr := xar.get_ptr(&x, 1) // ptr remains valid after more push_backs xar.push_back(&x, 40) fmt.println(ptr^) // prints 20 }

Collection Info

View Source
Collection
core
Path
container/xar
Entries
71

Source Files

Constants

2

Types

4

Array #

Source
Array :: Array

An Exponential Array with stable element addresses. Unlike `[dynamic]T` which reallocates and moves elements when growing, `Array` allocates separate chunks of exponentially increasing size. This guarantees that pointers to elements remain valid for the lifetime of the container. Fields: - `chunks`: Fixed array of multi-pointers to allocated chunks - `len`: Number of elements currently stored - `allocator`: Allocator used for chunk allocations Type Parameters: - `T`: The element type - `SHIFT`: Controls initial chunk size (1 << SHIFT). Must be in range (0, MAX_SHIFT]. Larger values mean fewer, bigger chunks. Recommended: 4-8. Chunk sizes grow as: - `chunks[0]`: 1 << SHIFT elements - `chunks[1]`: 1 << SHIFT elements - `chunks[2]`: 1 << (SHIFT + 1) elements - `chunks[3]`: 1 << (SHIFT + 2) elements - `chunks[4]`: 1 << (SHIFT + 3) elements - ...and so on Example: import "core:container/xar" example :: proc() { // Xar with initial chunk size of 16 (1 << 4) x: xar.Array(My_Struct, 4) defer xar.destroy(&x) }

Array_Iterator #

Source
Array_Iterator :: Array_Iterator

Iterator state for traversing a `Xar`. Fields: - `xar`: Pointer to the exponential array being iterated - `idx`: Current iteration index

Procedures

38

append_and_get_ptr #

Source
@(require_results)
append_and_get_ptr :: proc(x: ^$X/Array($T, $SHIFT), value: $T, loc := #caller_location) -> (ptr: $$deferred_return, err: Allocator_Error) {…}

Append an element and return a stable pointer to it. This is useful when you need to initialize a complex struct in-place or retain a reference to the newly added element. **Inputs** - `x`: Pointer to the exponential array - `value`: The element to append **Returns** - a stable pointer to the newly added element - allocation error if chunk allocation failed Example: import "core:container/xar" push_back_and_get_ptr_example :: proc() { x: xar.Array(My_Struct, 4) defer xar.destroy(&x) ptr := xar.push_back_elem_and_get_ptr(&x, My_Struct{}) or_else panic("alloc failed") ptr.field = 42 // Initialize in-place }

array_append_and_get_ptr #

Source
@(require_results)
array_append_and_get_ptr :: proc(x: ^$X/Array($T, $SHIFT), value: $T, loc := #caller_location) -> (ptr: $$deferred_return, err: Allocator_Error) {…}

Append an element and return a stable pointer to it. This is useful when you need to initialize a complex struct in-place or retain a reference to the newly added element. **Inputs** - `x`: Pointer to the exponential array - `value`: The element to append **Returns** - a stable pointer to the newly added element - allocation error if chunk allocation failed Example: import "core:container/xar" push_back_and_get_ptr_example :: proc() { x: xar.Array(My_Struct, 4) defer xar.destroy(&x) ptr := xar.push_back_elem_and_get_ptr(&x, My_Struct{}) or_else panic("alloc failed") ptr.field = 42 // Initialize in-place }

array_cap #

Source
@(require_results)
array_cap :: proc "contextless" (x: $X/Array($T, $SHIFT)) -> int {…}

Returns the number of allocated elements

array_clear #

Source
array_clear :: proc "contextless" (x: ^$X/Array($T, $SHIFT)) {…}

Resets the array's length to zero without freeing memory. Allocated chunks are retained for reuse.

array_destroy #

Source
array_destroy :: proc(x: ^$X/Array($T, $SHIFT)) {…}

Frees all allocated chunks and resets the exponential array. **Inputs** - `x`: Pointer to the exponential array to destroy

array_get #

Source
@(require_results)
array_get :: proc(x: ^$X/Array($T, $SHIFT), #any_int index: int, loc := #caller_location) -> (val: $$deferred_return) {…}

Get a copy of the element at the specified index. **Inputs** - `x`: Pointer to the exponential array - `index`: Position of the element (0-indexed) **Returns** - a copy of the element

array_get_ptr #

Source
@(require_results)
array_get_ptr :: proc(x: ^$X/Array($T, $SHIFT), #any_int index: int, loc := #caller_location) -> (val: $$deferred_return) {…}

Get a pointer to the element at the specified index. The returned pointer remains valid even after additional elements are added, as long as the element is not removed and the array is not destroyed. **Inputs** - `x`: Pointer to the exponential array - `index`: Position of the element (0-indexed) **Returns** - a stable pointer to the element Example: import "core:container/xar" get_ptr_example :: proc() { x: xar.Array(int, 4) defer xar.destroy(&x) xar.push_back(&x, 100) ptr := xar.get_ptr(&x, 0) // Pointer remains valid after growing for i in 0..<1000 { xar.push_back(&x, i) } fmt.println(ptr^) // Still prints 100 }

array_get_ptr_unsafe #

Source
@(require_results)
array_get_ptr_unsafe :: proc "contextless" (x: ^$X/Array($T, $SHIFT), #any_int index: int) -> (val: $$deferred_return) {…}

No bounds checking

array_init #

Source
array_init :: proc(x: ^$X/Array($T, $SHIFT), allocator := context.allocator) {…}

Initializes an exponential array with the given allocator. **Inputs** - `x`: Pointer to the exponential array to initialize - `allocator`: Allocator to use for chunk allocations (defaults to context.allocator)

array_iterate_by_ptr #

Source
array_iterate_by_ptr :: proc(it: ^Array_Iterator($T, $SHIFT)) -> (val: $$deferred_return, idx: int, ok: bool) {…}

Advance the iterator and returns a pointer to the next element. **Inputs** - `it`: Pointer to the iterator **Returns** - pointer to the current element - `true` if an element was returned, `false` if iteration is complete

array_iterate_by_val #

Source
array_iterate_by_val :: proc(it: ^Array_Iterator($T, $SHIFT)) -> (val: $$deferred_return, idx: int, ok: bool) {…}

Advance the iterator and returns the next element. **Inputs** - `it`: Pointer to the iterator **Returns** - current element - `true` if an element was returned, `false` if iteration is complete

array_iterator #

Source
array_iterator :: proc(xar: ^$X/Array($T, $SHIFT)) -> $$deferred_return {…}

Create an iterator for traversing the exponential array. **Inputs** - `xar`: Pointer to the exponential array **Returns** - an iterator positioned at the start Example: import "core:container/xar" import "core:fmt" iterator_example :: proc() { x: xar.Array(int, 4) defer xar.destroy(&x) xar.push_back(&x, 10) xar.push_back(&x, 20) xar.push_back(&x, 30) it := xar.iterator(&x) for val in xar.iterate_by_ptr(&it) { fmt.println(val^) } } Output: 10 20 30

array_len #

Source
@(require_results)
array_len :: proc "contextless" (x: $X/Array($T, $SHIFT)) -> int {…}

Returns the length of the exponential-array

array_linear_search #

Source
@(require_results)
array_linear_search :: proc(x: ^$X/Array($T, $SHIFT), elem: $T) -> (index: int, found: bool) {…}

array_pop #

Source
array_pop :: proc(x: ^$X/Array($T, $SHIFT), loc := #caller_location) -> (val: $$deferred_return) {…}

`pop` will remove and return the end value of an exponential array `x` and reduces the length of the array by 1. Note: If the exponential array has no elements (`xar.len(x) == 0`), this procedure will panic.

array_pop_safe #

Source
@(require_results)
array_pop_safe :: proc(x: ^$X/Array($T, $SHIFT)) -> (val: $$deferred_return, ok: bool) {…}

`pop_safe` trys to remove and return the end value of dynamic array `x` and reduces the length of the array by 1. If the operation is not possible, it will return false.

array_push_back_elem #

Source
array_push_back_elem :: proc(x: ^$X/Array($T, $SHIFT), value: $T, loc := #caller_location) -> (n: int, err: Allocator_Error) {…}

Append an element to the end of the exponential array. Allocates a new chunk if necessary. Existing elements aren't moved, and their pointers remain stable. **Inputs** - `x`: Pointer to the exponential array - `value`: The element to append **Returns** - number of elements added (always 1 on success) - allocation error if chunk allocation failed Example: import "core:container/xar" push_back_example :: proc() { x: xar.Array(string, 4) defer xar.destroy(&x) xar.push_back(&x, "hello") xar.push_back(&x, "world") fmt.println(xar.get(&x, 0)) // hello fmt.println(xar.get(&x, 1)) // world }

array_push_back_elem_and_get_ptr #

Source
@(require_results)
array_push_back_elem_and_get_ptr :: proc(x: ^$X/Array($T, $SHIFT), value: $T, loc := #caller_location) -> (ptr: $$deferred_return, err: Allocator_Error) {…}

Append an element and return a stable pointer to it. This is useful when you need to initialize a complex struct in-place or retain a reference to the newly added element. **Inputs** - `x`: Pointer to the exponential array - `value`: The element to append **Returns** - a stable pointer to the newly added element - allocation error if chunk allocation failed Example: import "core:container/xar" push_back_and_get_ptr_example :: proc() { x: xar.Array(My_Struct, 4) defer xar.destroy(&x) ptr := xar.push_back_elem_and_get_ptr(&x, My_Struct{}) or_else panic("alloc failed") ptr.field = 42 // Initialize in-place }

array_push_back_elems #

Source
array_push_back_elems :: proc(x: ^$X/Array($T, $SHIFT), values: ..$T, loc := #caller_location) -> (n: int, err: Allocator_Error) {…}

Append multiple elements to the end of the exponential array. **Inputs** - `x`: Pointer to the exponential array - `values`: The elements to append **Returns** - number of elements successfully added - allocation error if chunk allocation failed (partial append possible)

array_set #

Source
array_set :: proc(x: ^$X/Array($T, $SHIFT), #any_int index: int, value: $T, loc := #caller_location) {…}

Set the element at the specified index to the given value. **Inputs** - `x`: Pointer to the exponential array - `index`: Position of the element (0-indexed) - `value`: The value to set

array_unordered_remove #

Source
array_unordered_remove :: proc(x: ^$X/Array($T, $SHIFT), #any_int index: int, loc := #caller_location) {…}

`unordered_remove` removed the element at the specified `index`. It does so by replacing the current end value with the old value, and reducing the length of the exponential array by 1. Note: This is an O(1) operation. Note: This is currently no procedure that is the equivalent of an "ordered_remove" Note: If the index is out of bounds, this procedure will panic. Note: Pointers to the last element become invalid (it gets moved). Pointers to other elements remain valid. Example: import "core:container/xar" unordered_remove_example :: proc() { x: xar.Array(int, 4) defer xar.destroy(&x) xar.push_back(&x, 10) xar.push_back(&x, 20) xar.push_back(&x, 30) xar.unordered_remove(&x, 0) // Removes 10, replaces with 30 // Array now contains [30, 20] fmt.println(xar.get(&x, 0)) // 30 fmt.println(xar.get(&x, 1)) // 20 }

freelist_get #

Source
@(require_results)
freelist_get :: proc(x: ^$X/Freelist_Array($T, $SHIFT), #any_int index: int, loc := #caller_location) -> $$deferred_return {…}

freelist_get_ptr #

Source
@(require_results)
freelist_get_ptr :: proc(x: ^$X/Freelist_Array($T, $SHIFT), #any_int index: int, loc := #caller_location) -> $$deferred_return {…}

Procedure Groups

27