package slice import "core:intrinsics" import "core:builtin" import "core:math/bits" import "core:runtime" _ :: intrinsics _ :: builtin _ :: bits _ :: runtime /* Turn a pointer and a length into a slice. */ @(require_results) from_ptr :: proc "contextless" (ptr: ^$T, count: int) -> []T { return ([^]T)(ptr)[:count] } /* Turn a pointer and a length into a byte slice. */ @(require_results) bytes_from_ptr :: proc "contextless" (ptr: rawptr, byte_count: int) -> []byte { return ([^]byte)(ptr)[:byte_count] } /* Turn a slice into a byte slice. See `slice.reinterpret` to go the other way. */ @(require_results) to_bytes :: proc "contextless" (s: []$T) -> []byte { return ([^]byte)(raw_data(s))[:len(s) * size_of(T)] } /* Turn a slice of one type, into a slice of another type. Only converts the type and length of the slice itself. The length is rounded down to the nearest whole number of items. ``` large_items := []i64{1, 2, 3, 4} small_items := slice.reinterpret([]i32, large_items) assert(len(small_items) == 8) ``` ``` small_items := []byte{1, 0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0} large_items := slice.reinterpret([]i64, small_items) assert(len(large_items) == 1) // only enough bytes to make 1 x i64; two would need at least 8 bytes. ``` */ @(require_results) reinterpret :: proc "contextless" ($T: typeid/[]$U, s: []$V) -> []U { when size_of(U) == 0 || size_of(V) == 0 { return nil } else { bytes := to_bytes(s) n := len(bytes) / size_of(U) return ([^]U)(raw_data(bytes))[:n] } } swap :: proc(array: $T/[]$E, a, b: int) { when size_of(E) > 8 { ptr_swap_non_overlapping(&array[a], &array[b], size_of(E)) } else { array[a], array[b] = array[b], array[a] } } swap_between :: proc(a, b: $T/[]$E) { n := builtin.min(len(a), len(b)) if n >= 0 { ptr_swap_overlapping(&a[0], &b[0], size_of(E)*n) } } reverse :: proc(array: $T/[]$E) { n := len(array)/2 for i in 0.. bool where intrinsics.type_is_comparable(E) { _, found := linear_search(array, value) return found } @(require_results) linear_search :: proc(array: $A/[]$T, key: T) -> (index: int, found: bool) where intrinsics.type_is_comparable(T) #no_bounds_check { for x, i in array { if x == key { return i, true } } return -1, false } @(require_results) linear_search_proc :: proc(array: $A/[]$T, f: proc(T) -> bool) -> (index: int, found: bool) #no_bounds_check { for x, i in array { if f(x) { return i, true } } return -1, false } @(require_results) binary_search :: proc(array: $A/[]$T, key: T) -> (index: int, found: bool) where intrinsics.type_is_ordered(T) #no_bounds_check { n := len(array) switch n { case 0: return -1, false case 1: if array[0] == key { return 0, true } return -1, false } lo, hi := 0, n-1 for array[hi] != array[lo] && key >= array[lo] && key <= array[hi] { when intrinsics.type_is_ordered_numeric(T) { // NOTE(bill): This is technically interpolation search m := lo + int((key - array[lo]) * T(hi - lo) / (array[hi] - array[lo])) } else { m := lo + (hi - lo)/2 } switch { case array[m] < key: lo = m + 1 case key < array[m]: hi = m - 1 case: return m, true } } if key == array[lo] { return lo, true } return -1, false } @(require_results) equal :: proc(a, b: $T/[]$E) -> bool where intrinsics.type_is_comparable(E) { if len(a) != len(b) { return false } when intrinsics.type_is_simple_compare(E) { return runtime.memory_compare(raw_data(a), raw_data(b), len(a)*size_of(E)) == 0 } else { for i in 0.. bool where intrinsics.type_is_simple_compare(E) { if len(a) != len(b) { return false } return runtime.memory_compare(raw_data(a), raw_data(b), len(a)*size_of(E)) == 0 } /* return the prefix length common between slices `a` and `b`. slice.prefix_length([]u8{1, 2, 3, 4}, []u8{1}) -> 1 slice.prefix_length([]u8{1, 2, 3, 4}, []u8{1, 2, 3}) -> 3 slice.prefix_length([]u8{1, 2, 3, 4}, []u8{2, 3, 4}) -> 0 */ @(require_results) prefix_length :: proc(a, b: $T/[]$E) -> (n: int) where intrinsics.type_is_comparable(E) { _len := builtin.min(len(a), len(b)) #no_bounds_check for n < _len && a[n] == b[n] { n += 1 } return } @(require_results) has_prefix :: proc(array: $T/[]$E, needle: E) -> bool where intrinsics.type_is_comparable(E) { n := len(needle) if len(array) >= n { return equal(array[:n], needle) } return false } @(require_results) has_suffix :: proc(array: $T/[]$E, needle: E) -> bool where intrinsics.type_is_comparable(E) { array := array m, n := len(array), len(needle) if m >= n { return equal(array[m-n:], needle) } return false } zero :: proc(array: $T/[]$E) #no_bounds_check { if len(array) > 0 { intrinsics.mem_zero(raw_data(array), size_of(E)*len(array)) } } fill :: proc(array: $T/[]$E, value: E) #no_bounds_check { if len(array) <= 0 { return } array[0] = value for i := 1; i < len(array); i *= 2 { copy(array[i:], array[:i]) } } rotate_left :: proc(array: $T/[]$E, mid: int) { n := len(array) m := mid %% n k := n - m // FIXME: (ap29600) this cast is a temporary fix for the compiler not matching // [^T] with $P/^$T p := cast(^E)raw_data(array) ptr_rotate(m, ptr_add(p, m), k) } rotate_right :: proc(array: $T/[]$E, k: int) { rotate_left(array, -k) } swap_with_slice :: proc(a, b: $T/[]$E, loc := #caller_location) { assert(len(a) == len(b), "miss matching slice lengths", loc) ptr_swap_non_overlapping(raw_data(a), raw_data(b), len(a)*size_of(E)) } @(require_results) concatenate :: proc(a: []$T/[]$E, allocator := context.allocator) -> (res: T, err: runtime.Allocator_Error) #optional_allocator_error { if len(a) == 0 { return } n := 0 for s in a { n += len(s) } res = make(T, n, allocator) or_return i := 0 for s in a { i += copy(res[i:], s) } return } // copies a slice into a new slice @(require_results) clone :: proc(a: $T/[]$E, allocator := context.allocator, loc := #caller_location) -> ([]E, runtime.Allocator_Error) #optional_allocator_error { d, err := make([]E, len(a), allocator, loc) copy(d[:], a) return d, err } // copies slice into a new dynamic array clone_to_dynamic :: proc(a: $T/[]$E, allocator := context.allocator, loc := #caller_location) -> ([dynamic]E, runtime.Allocator_Error) #optional_allocator_error { d, err := make([dynamic]E, len(a), allocator, loc) copy(d[:], a) return d, err } to_dynamic :: clone_to_dynamic // Converts slice into a dynamic array without cloning or allocating memory @(require_results) into_dynamic :: proc(a: $T/[]$E) -> [dynamic]E { s := transmute(runtime.Raw_Slice)a d := runtime.Raw_Dynamic_Array{ data = s.data, len = 0, cap = s.len, allocator = runtime.nil_allocator(), } return transmute([dynamic]E)d } @(require_results) length :: proc(a: $T/[]$E) -> int { return len(a) } @(require_results) is_empty :: proc(a: $T/[]$E) -> bool { return len(a) == 0 } @(require_results) split_at :: proc(array: $T/[]$E, index: int) -> (a, b: T) { return array[:index], array[index:] } @(require_results) split_first :: proc(array: $T/[]$E) -> (first: E, rest: T) { return array[0], array[1:] } @(require_results) split_last :: proc(array: $T/[]$E) -> (rest: T, last: E) { n := len(array)-1 return array[:n], array[n] } @(require_results) first :: proc(array: $T/[]$E) -> E { return array[0] } @(require_results) last :: proc(array: $T/[]$E) -> E { return array[len(array)-1] } @(require_results) first_ptr :: proc(array: $T/[]$E) -> ^E { if len(array) != 0 { return &array[0] } return nil } @(require_results) last_ptr :: proc(array: $T/[]$E) -> ^E { if len(array) != 0 { return &array[len(array)-1] } return nil } @(require_results) get :: proc(array: $T/[]$E, index: int) -> (value: E, ok: bool) { if uint(index) < len(array) { value = array[index] ok = true } return } @(require_results) get_ptr :: proc(array: $T/[]$E, index: int) -> (value: ^E, ok: bool) { if uint(index) < len(array) { value = &array[index] ok = true } return } @(require_results) as_ptr :: proc(array: $T/[]$E) -> [^]E { return raw_data(array) } @(require_results) mapper :: proc(s: $S/[]$U, f: proc(U) -> $V, allocator := context.allocator) -> (r: []V, err: runtime.Allocator_Error) #optional_allocator_error { r = make([]V, len(s), allocator) or_return for v, i in s { r[i] = f(v) } return } @(require_results) reduce :: proc(s: $S/[]$U, initializer: $V, f: proc(V, U) -> V) -> V { r := initializer for v in s { r = f(r, v) } return r } @(require_results) filter :: proc(s: $S/[]$U, f: proc(U) -> bool, allocator := context.allocator) -> S { r := make([dynamic]U, 0, 0, allocator) for v in s { if f(v) { append(&r, v) } } return r[:] } @(require_results) scanner :: proc (s: $S/[]$U, initializer: $V, f: proc(V, U) -> V, allocator := context.allocator) -> (res: []V, err: runtime.Allocator_Error) #optional_allocator_error { if len(s) == 0 { return } res = make([]V, len(s), allocator) or_return p := as_ptr(s) q := as_ptr(res) r := initializer for l := len(s); l > 0; l -= 1 { r = f(r, p[0]) q[0] = r p = p[1:] q = q[1:] } return } @(require_results) min :: proc(s: $S/[]$T) -> (res: T, ok: bool) where intrinsics.type_is_ordered(T) #optional_ok { if len(s) != 0 { res = s[0] ok = true for v in s[1:] { res = builtin.min(res, v) } } return } @(require_results) max :: proc(s: $S/[]$T) -> (res: T, ok: bool) where intrinsics.type_is_ordered(T) #optional_ok { if len(s) != 0 { res = s[0] ok = true for v in s[1:] { res = builtin.max(res, v) } } return } @(require_results) min_max :: proc(s: $S/[]$T) -> (min, max: T, ok: bool) where intrinsics.type_is_ordered(T) { if len(s) != 0 { min, max = s[0], s[0] ok = true for v in s[1:] { min = builtin.min(min, v) max = builtin.max(max, v) } } return } @(require_results) any_of :: proc(s: $S/[]$T, value: T) -> bool where intrinsics.type_is_comparable(T) { for v in s { if v == value { return true } } return false } @(require_results) none_of :: proc(s: $S/[]$T, value: T) -> bool where intrinsics.type_is_comparable(T) { for v in s { if v == value { return false } } return true } @(require_results) all_of :: proc(s: $S/[]$T, value: T) -> bool where intrinsics.type_is_comparable(T) { if len(s) == 0 { return false } for v in s { if v != value { return false } } return true } @(require_results) any_of_proc :: proc(s: $S/[]$T, f: proc(T) -> bool) -> bool { for v in s { if f(v) { return true } } return false } @(require_results) none_of_proc :: proc(s: $S/[]$T, f: proc(T) -> bool) -> bool { for v in s { if f(v) { return false } } return true } @(require_results) all_of_proc :: proc(s: $S/[]$T, f: proc(T) -> bool) -> bool { if len(s) == 0 { return false } for v in s { if !f(v) { return false } } return true } @(require_results) count :: proc(s: $S/[]$T, value: T) -> (n: int) where intrinsics.type_is_comparable(T) { for v in s { if v == value { n += 1 } } return } @(require_results) count_proc :: proc(s: $S/[]$T, f: proc(T) -> bool) -> (n: int) { for v in s { if f(v) { n += 1 } } return } @(require_results) dot_product :: proc(a, b: $S/[]$T) -> (r: T, ok: bool) where intrinsics.type_is_numeric(T) { if len(a) != len(b) { return } #no_bounds_check for _, i in a { r += a[i] * b[i] } return r, true } // Convert a pointer to an enumerated array to a slice of the element type @(require_results) enumerated_array :: proc(ptr: ^$T) -> []intrinsics.type_elem_type(T) where intrinsics.type_is_enumerated_array(T) { return ([^]intrinsics.type_elem_type(T))(ptr)[:len(T)] }