2024-02-14 02:29:28 -05:00
|
|
|
// Based on gencpp's and thus zpl's Array implementation
|
|
|
|
// Made becasue of the map issue with fonts during hot-reload.
|
2024-05-04 11:47:21 -04:00
|
|
|
// I didn't want to make the HMapZPL impl with the [dynamic] array for now to isolate the hot-reload issue (when I was diagnoising)
|
2024-02-14 02:29:28 -05:00
|
|
|
package sectr
|
|
|
|
|
|
|
|
import "core:c/libc"
|
|
|
|
import "core:mem"
|
2024-02-22 23:15:29 -05:00
|
|
|
import "core:slice"
|
2024-02-14 02:29:28 -05:00
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
// Array :: struct ( $ Type : typeid ) {
|
2024-03-12 02:32:16 -04:00
|
|
|
// bakcing : Allocator,
|
2024-03-05 10:40:28 -05:00
|
|
|
// capacity : u64,
|
|
|
|
// num : u64,
|
|
|
|
// data : [^]Type,
|
|
|
|
// }
|
|
|
|
|
|
|
|
ArrayHeader :: struct ( $ Type : typeid ) {
|
2024-03-12 02:32:16 -04:00
|
|
|
backing : Allocator,
|
2024-05-06 23:29:47 -04:00
|
|
|
dbg_name : string,
|
|
|
|
fixed_cap : b32,
|
2024-02-14 02:29:28 -05:00
|
|
|
capacity : u64,
|
|
|
|
num : u64,
|
|
|
|
data : [^]Type,
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
Array :: struct ( $ Type : typeid ) {
|
|
|
|
using header : ^ArrayHeader(Type),
|
|
|
|
}
|
|
|
|
|
|
|
|
array_underlying_slice :: proc(slice: []($ Type)) -> Array(Type)
|
|
|
|
{
|
2024-03-02 18:19:27 -05:00
|
|
|
if len(slice) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
array_size := size_of( Array(Type))
|
|
|
|
raw_data := & slice[0]
|
|
|
|
array_ptr := cast( ^Array(Type)) ( uintptr(first_element_ptr) - uintptr(array_size))
|
|
|
|
return array_ptr ^
|
|
|
|
}
|
|
|
|
|
2024-05-14 11:47:44 -04:00
|
|
|
array_to_slice :: proc( using self : Array($ Type) ) -> []Type {
|
2024-03-02 18:19:27 -05:00
|
|
|
return slice_ptr( data, int(num) )
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-05-14 11:47:44 -04:00
|
|
|
array_to_slice_capacity :: proc( using self : Array($ Type) ) -> []Type {
|
2024-03-02 20:22:28 -05:00
|
|
|
return slice_ptr( data, int(capacity))
|
|
|
|
}
|
|
|
|
|
2024-02-14 02:29:28 -05:00
|
|
|
array_grow_formula :: proc( value : u64 ) -> u64 {
|
2024-03-20 22:34:18 -04:00
|
|
|
result := (2 * value) + 8
|
|
|
|
return result
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
array_init :: proc( $ Type : typeid, allocator : Allocator ) -> ( Array(Type), AllocatorError ) {
|
|
|
|
return array_init_reserve( Type, allocator, array_grow_formula(0) )
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_init_reserve :: proc
|
2024-05-06 23:29:47 -04:00
|
|
|
( $ Type : typeid, allocator : Allocator, capacity : u64, fixed_cap : b32 = false, dbg_name : string = "" ) -> ( result : Array(Type), alloc_error : AllocatorError )
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-03-19 23:25:48 -04:00
|
|
|
header_size := size_of(ArrayHeader(Type))
|
|
|
|
array_size := header_size + int(capacity) * size_of(Type)
|
2024-03-05 10:40:28 -05:00
|
|
|
|
|
|
|
raw_mem : rawptr
|
2024-03-19 23:25:48 -04:00
|
|
|
raw_mem, alloc_error = alloc( array_size, allocator = allocator )
|
2024-03-20 13:34:47 -04:00
|
|
|
// log( str_fmt_tmp("array reserved: %d", header_size + int(capacity) * size_of(Type) ))
|
2024-03-05 10:40:28 -05:00
|
|
|
if alloc_error != AllocatorError.None do return
|
|
|
|
|
2024-05-14 11:47:44 -04:00
|
|
|
result.header = cast( ^ArrayHeader(Type)) raw_mem
|
2024-03-12 02:32:16 -04:00
|
|
|
result.backing = allocator
|
2024-05-06 23:29:47 -04:00
|
|
|
// result.dbg_name = dbg_name
|
|
|
|
result.fixed_cap = fixed_cap
|
2024-02-14 02:29:28 -05:00
|
|
|
result.capacity = capacity
|
2024-03-05 10:40:28 -05:00
|
|
|
result.data = cast( [^]Type ) (cast( [^]ArrayHeader(Type)) result.header)[ 1:]
|
|
|
|
return
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-10 10:31:21 -04:00
|
|
|
array_append :: proc( self : ^Array( $ Type), value : Type ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-03-11 02:05:18 -04:00
|
|
|
// profile(#procedure)
|
2024-03-10 10:31:21 -04:00
|
|
|
if self.header.num == self.header.capacity
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-03-10 10:31:21 -04:00
|
|
|
grow_result := array_grow( self, self.header.capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
if grow_result != AllocatorError.None {
|
|
|
|
return grow_result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-10 10:31:21 -04:00
|
|
|
self.header.data[ self.header.num ] = value
|
|
|
|
self.header.num += 1
|
2024-02-14 02:29:28 -05:00
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_append_slice :: proc( using self : ^Array( $ Type ), items : []Type ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
if num + len(items) > capacity
|
|
|
|
{
|
2024-02-27 07:50:57 -05:00
|
|
|
grow_result := array_grow( self, capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
if grow_result != AllocatorError.None {
|
|
|
|
return grow_result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note(Ed) : Original code from gencpp
|
|
|
|
// libc.memcpy( ptr_offset(data, num), raw_data(items), len(items) * size_of(Type) )
|
|
|
|
|
|
|
|
// TODO(Ed) : VERIFY VIA DEBUG THIS COPY IS FINE.
|
|
|
|
target := ptr_offset( data, num )
|
|
|
|
copy( slice_ptr(target, capacity - num), items )
|
|
|
|
|
|
|
|
num += len(items)
|
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_append_at :: proc( using self : ^Array( $ Type ), item : Type, id : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
id := id
|
|
|
|
if id >= num {
|
|
|
|
id = num - 1
|
|
|
|
}
|
|
|
|
if id < 0 {
|
|
|
|
id = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if capacity < num + 1
|
|
|
|
{
|
2024-02-27 07:50:57 -05:00
|
|
|
grow_result := array_grow( self, capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
if grow_result != AllocatorError.None {
|
|
|
|
return grow_result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
target := & data[id]
|
2024-05-06 23:29:47 -04:00
|
|
|
libc.memmove( ptr_offset(target, 1), target, uint(num - id) * size_of(Type) )
|
2024-02-14 02:29:28 -05:00
|
|
|
|
|
|
|
data[id] = item
|
|
|
|
num += 1
|
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_append_at_slice :: proc( using self : ^Array( $ Type ), items : []Type, id : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
id := id
|
|
|
|
if id >= num {
|
|
|
|
return array_append_slice( items )
|
|
|
|
}
|
|
|
|
if len(items) > capacity
|
|
|
|
{
|
2024-02-27 07:50:57 -05:00
|
|
|
grow_result := array_grow( self, capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
if grow_result != AllocatorError.None {
|
|
|
|
return grow_result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note(Ed) : Original code from gencpp
|
|
|
|
// target := ptr_offset( data, id + len(items) )
|
|
|
|
// src := ptr_offset( data, id )
|
|
|
|
// libc.memmove( target, src, num - id * size_of(Type) )
|
|
|
|
// libc.memcpy ( src, raw_data(items), len(items) * size_of(Type) )
|
|
|
|
|
|
|
|
// TODO(Ed) : VERIFY VIA DEBUG THIS COPY IS FINE
|
|
|
|
target := & data[id + len(items)]
|
|
|
|
dst := slice_ptr( target, num - id - len(items) )
|
|
|
|
src := slice_ptr( & data[id], num - id )
|
|
|
|
copy( dst, src )
|
|
|
|
copy( src, items )
|
|
|
|
|
|
|
|
num += len(items)
|
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-20 00:44:43 -04:00
|
|
|
// array_back :: proc( )
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_push_back :: proc( using self : Array( $ Type)) -> b32 {
|
|
|
|
if num == capacity {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
data[ num ] = value
|
|
|
|
num += 1
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2024-05-06 23:29:47 -04:00
|
|
|
array_clear :: proc "contextless" ( using self : Array( $ Type ), zero_data : b32 = false ) {
|
2024-02-27 07:50:57 -05:00
|
|
|
if zero_data {
|
2024-05-06 23:29:47 -04:00
|
|
|
mem.set( data, 0, int(num * size_of(Type)) )
|
2024-02-27 07:50:57 -05:00
|
|
|
}
|
2024-05-06 23:29:47 -04:00
|
|
|
header.num = 0
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_fill :: proc( using self : Array( $ Type ), begin, end : u64, value : Type ) -> b32
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
if begin < 0 || end >= num {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2024-02-22 23:15:29 -05:00
|
|
|
// data_slice := slice_ptr( ptr_offset( data, begin ), end - begin )
|
|
|
|
// slice.fill( data_slice, cast(int) value )
|
|
|
|
|
2024-02-14 02:29:28 -05:00
|
|
|
for id := begin; id < end; id += 1 {
|
|
|
|
data[ id ] = value
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_free :: proc( using self : Array( $ Type ) ) {
|
2024-05-06 23:29:47 -04:00
|
|
|
free( self.header, backing )
|
2024-03-05 10:40:28 -05:00
|
|
|
self.data = nil
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_grow :: proc( using self : ^Array( $ Type ), min_capacity : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-03-12 02:32:16 -04:00
|
|
|
// profile(#procedure)
|
2024-02-22 23:15:29 -05:00
|
|
|
new_capacity := array_grow_formula( capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
|
|
|
|
if new_capacity < min_capacity {
|
|
|
|
new_capacity = min_capacity
|
|
|
|
}
|
2024-02-27 07:50:57 -05:00
|
|
|
return array_set_capacity( self, new_capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_pop :: proc( using self : Array( $ Type ) ) {
|
2024-02-27 07:50:57 -05:00
|
|
|
verify( num != 0, "Attempted to pop an array with no elements" )
|
|
|
|
num -= 1
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_remove_at :: proc( using self : Array( $ Type ), id : u64 )
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-05-06 23:29:47 -04:00
|
|
|
verify( id < header.num, "Attempted to remove from an index larger than the array" )
|
2024-02-14 02:29:28 -05:00
|
|
|
|
2024-05-06 23:29:47 -04:00
|
|
|
left := & data[id]
|
|
|
|
right := & data[id + 1]
|
|
|
|
libc.memmove( left, right, uint(num - id) * size_of(Type) )
|
2024-02-14 02:29:28 -05:00
|
|
|
|
2024-05-06 23:29:47 -04:00
|
|
|
header.num -= 1
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_reserve :: proc( using self : ^Array( $ Type ), new_capacity : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
if capacity < new_capacity {
|
2024-02-27 07:50:57 -05:00
|
|
|
return array_set_capacity( self, new_capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_resize :: proc( array : ^Array( $ Type ), num : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
|
|
|
if array.capacity < num
|
|
|
|
{
|
2024-02-22 23:15:29 -05:00
|
|
|
grow_result := array_grow( array, array.capacity )
|
2024-02-14 02:29:28 -05:00
|
|
|
if grow_result != AllocatorError.None {
|
|
|
|
return grow_result
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
array.num = num
|
|
|
|
return AllocatorError.None
|
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
array_set_capacity :: proc( self : ^Array( $ Type ), new_capacity : u64 ) -> AllocatorError
|
2024-02-14 02:29:28 -05:00
|
|
|
{
|
2024-03-05 10:40:28 -05:00
|
|
|
if new_capacity == self.capacity {
|
2024-02-22 23:15:29 -05:00
|
|
|
return AllocatorError.None
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
2024-03-05 10:40:28 -05:00
|
|
|
if new_capacity < self.num {
|
|
|
|
self.num = new_capacity
|
2024-02-22 23:15:29 -05:00
|
|
|
return AllocatorError.None
|
2024-02-14 02:29:28 -05:00
|
|
|
}
|
|
|
|
|
2024-03-05 10:40:28 -05:00
|
|
|
header_size :: size_of(ArrayHeader(Type))
|
|
|
|
|
2024-03-19 23:25:48 -04:00
|
|
|
new_size := header_size + (cast(int) new_capacity ) * size_of(Type)
|
|
|
|
old_size := header_size + (cast(int) self.capacity) * size_of(Type)
|
2024-03-05 10:40:28 -05:00
|
|
|
|
2024-03-19 12:18:39 -04:00
|
|
|
new_mem, result_code := resize_non_zeroed( self.header, old_size, new_size, mem.DEFAULT_ALIGNMENT, allocator = self.backing )
|
2024-03-12 02:32:16 -04:00
|
|
|
|
2024-03-02 10:24:09 -05:00
|
|
|
if result_code != AllocatorError.None {
|
|
|
|
ensure( false, "Failed to allocate for new array capacity" )
|
|
|
|
return result_code
|
|
|
|
}
|
2024-03-12 02:32:16 -04:00
|
|
|
if new_mem == nil {
|
|
|
|
ensure(false, "new_mem is nil but no allocation error")
|
|
|
|
return result_code
|
|
|
|
}
|
2024-03-05 10:40:28 -05:00
|
|
|
|
2024-03-19 23:25:48 -04:00
|
|
|
self.header = cast( ^ArrayHeader(Type)) raw_data(new_mem);
|
|
|
|
self.header.data = cast( [^]Type ) (cast( [^]ArrayHeader(Type)) self.header)[ 1:]
|
|
|
|
self.header.capacity = new_capacity
|
|
|
|
self.header.num = self.num
|
2024-02-14 02:29:28 -05:00
|
|
|
return result_code
|
|
|
|
}
|
2024-05-06 23:29:47 -04:00
|
|
|
|
|
|
|
array_block_size :: proc "contextless" ( self : Array( $Type ) ) -> u64 {
|
|
|
|
header_size :: size_of(ArrayHeader(Type))
|
|
|
|
block_size := cast(u64) (header_size + self.capacity * size_of(Type))
|
|
|
|
return block_size
|
|
|
|
}
|
|
|
|
|
|
|
|
array_memtracker_entry :: proc( self : Array( $Type ), name : string ) -> MemoryTrackerEntry {
|
|
|
|
header_size :: size_of(ArrayHeader(Type))
|
|
|
|
block_size := cast(uintptr) (header_size + (cast(uintptr) self.capacity) * size_of(Type))
|
|
|
|
|
|
|
|
block_start := transmute(^u8) self.header
|
|
|
|
block_end := ptr_offset( block_start, block_size )
|
|
|
|
|
|
|
|
tracker_entry := MemoryTrackerEntry { name, block_start, block_end }
|
|
|
|
return tracker_entry
|
|
|
|
}
|