123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340 |
- package slice
- import "core:intrinsics"
- import "core:builtin"
- import "core:math/bits"
- import "core:mem"
- _ :: intrinsics
- _ :: builtin
- _ :: bits
- _ :: mem
- 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..<n {
- a, b := i, len(array)-i-1
- array[a], array[b] = array[b], array[a]
- }
- }
- contains :: proc(array: $T/[]$E, value: E) -> bool where intrinsics.type_is_comparable(E) {
- _, found := linear_search(array, value)
- return found
- }
- 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
- }
- 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
- }
- 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
- }
- 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 mem.compare_ptrs(raw_data(a), raw_data(b), len(a)*size_of(E)) == 0
- } else {
- for i in 0..<len(a) {
- if a[i] != b[i] {
- return false
- }
- }
- return true
- }
- }
- simple_equal :: proc(a, b: $T/[]$E) -> bool where intrinsics.type_is_simple_compare(E) {
- if len(a) != len(b) {
- return false
- }
- return mem.compare_ptrs(raw_data(a), raw_data(b), len(a)*size_of(E)) == 0
- }
- 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
- }
- 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
- }
- 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
- p := raw_data(array)
- ptr_rotate(mid, ptr_add(p, mid), 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))
- }
- concatenate :: proc(a: []$T/[]$E, allocator := context.allocator) -> (res: T) {
- if len(a) == 0 {
- return
- }
- n := 0
- for s in a {
- n += len(s)
- }
- res = make(T, n, allocator)
- i := 0
- for s in a {
- i += copy(res[i:], s)
- }
- return
- }
- // copies slice into a new dynamic array
- clone :: proc(a: $T/[]$E, allocator := context.allocator) -> []E {
- d := make([]E, len(a), allocator)
- copy(d[:], a)
- return d
- }
- // copies slice into a new dynamic array
- to_dynamic :: proc(a: $T/[]$E, allocator := context.allocator) -> [dynamic]E {
- d := make([dynamic]E, len(a), allocator)
- copy(d[:], a)
- return d
- }
- // Converts slice into a dynamic array without cloning or allocating memory
- into_dynamic :: proc(a: $T/[]$E) -> [dynamic]E {
- s := transmute(mem.Raw_Slice)a
- d := mem.Raw_Dynamic_Array{
- data = s.data,
- len = 0,
- cap = s.len,
- allocator = mem.nil_allocator(),
- }
- return transmute([dynamic]E)d
- }
- length :: proc(a: $T/[]$E) -> int {
- return len(a)
- }
- is_empty :: proc(a: $T/[]$E) -> bool {
- return len(a) == 0
- }
- split_at :: proc(array: $T/[]$E, index: int) -> (a, b: T) {
- return array[:index], array[index:]
- }
- split_first :: proc(array: $T/[]$E) -> (first: E, rest: T) {
- return array[0], array[1:]
- }
- split_last :: proc(array: $T/[]$E) -> (rest: T, last: E) {
- n := len(array)-1
- return array[:n], array[n]
- }
- first :: proc(array: $T/[]$E) -> E {
- return array[0]
- }
- last :: proc(array: $T/[]$E) -> E {
- return array[len(array)-1]
- }
- first_ptr :: proc(array: $T/[]$E) -> ^E {
- if len(array) != 0 {
- return &array[0]
- }
- return nil
- }
- last_ptr :: proc(array: $T/[]$E) -> ^E {
- if len(array) != 0 {
- return &array[len(array)-1]
- }
- return nil
- }
- get :: proc(array: $T/[]$E, index: int) -> (value: E, ok: bool) {
- if 0 <= index && index < len(array) {
- value = array[index]
- ok = true
- }
- return
- }
- get_ptr :: proc(array: $T/[]$E, index: int) -> (value: ^E, ok: bool) {
- if 0 <= index && index < len(array) {
- value = &array[index]
- ok = true
- }
- return
- }
- as_ptr :: proc(array: $T/[]$E) -> ^E {
- return raw_data(array)
- }
- mapper :: proc(s: $S/[]$U, f: proc(U) -> $V, allocator := context.allocator) -> []V {
- r := make([]V, len(s), allocator)
- for v, i in s {
- r[i] = f(v)
- }
- return r
- }
- 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
- }
- 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[:]
- }
- 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
- }
- 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
- }
- dot_product :: proc(a, b: $S/[]$T) -> T
- where intrinsics.type_is_numeric(T) {
- if len(a) != len(b) {
- panic("slice.dot_product: slices of unequal length")
- }
- r: T
- #no_bounds_check for _, i in a {
- r += a[i] * b[i]
- }
- return r
- }
|