|
@@ -251,6 +251,26 @@ map_hash_is_valid :: #force_inline proc "contextless" (hash: Map_Hash) -> bool {
|
|
|
return (hash != 0) & (hash & TOMBSTONE_MASK == 0)
|
|
|
}
|
|
|
|
|
|
+@(require_results)
|
|
|
+map_seed :: #force_inline proc "contextless" (m: Raw_Map) -> uintptr {
|
|
|
+ return map_seed_from_map_data(map_data(m))
|
|
|
+}
|
|
|
+
|
|
|
+// splitmix for uintptr
|
|
|
+@(require_results)
|
|
|
+map_seed_from_map_data :: #force_inline proc "contextless" (data: uintptr) -> uintptr {
|
|
|
+ when size_of(uintptr) == size_of(u64) {
|
|
|
+ mix := data + 0x9e3779b97f4a7c15
|
|
|
+ mix = (mix ~ (mix >> 30)) * 0xbf58476d1ce4e5b9
|
|
|
+ mix = (mix ~ (mix >> 27)) * 0x94d049bb133111eb
|
|
|
+ return mix ~ (mix >> 31)
|
|
|
+ } else {
|
|
|
+ mix := data + 0x9e3779b9
|
|
|
+ mix = (mix ~ (mix >> 16)) * 0x21f0aaad
|
|
|
+ mix = (mix ~ (mix >> 15)) * 0x735a2d97
|
|
|
+ return mix ~ (mix >> 15)
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
// Computes the desired position in the array. This is just index % capacity,
|
|
|
// but a procedure as there's some math involved here to recover the capacity.
|
|
@@ -394,32 +414,71 @@ map_insert_hash_dynamic :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^
|
|
|
tk := map_cell_index_dynamic(sk, info.ks, 1)
|
|
|
tv := map_cell_index_dynamic(sv, info.vs, 1)
|
|
|
|
|
|
-
|
|
|
for {
|
|
|
hp := &hs[pos]
|
|
|
element_hash := hp^
|
|
|
|
|
|
if map_hash_is_empty(element_hash) {
|
|
|
- k_dst := map_cell_index_dynamic(ks, info.ks, pos)
|
|
|
- v_dst := map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
- intrinsics.mem_copy_non_overlapping(rawptr(k_dst), rawptr(k), size_of_k)
|
|
|
- intrinsics.mem_copy_non_overlapping(rawptr(v_dst), rawptr(v), size_of_v)
|
|
|
+ kp := map_cell_index_dynamic(ks, info.ks, pos)
|
|
|
+ vp := map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(kp), rawptr(k), size_of_k)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(vp), rawptr(v), size_of_v)
|
|
|
hp^ = h
|
|
|
|
|
|
- return result if result != 0 else v_dst
|
|
|
+ return result if result != 0 else vp
|
|
|
}
|
|
|
|
|
|
- if probe_distance := map_probe_distance(m^, element_hash, pos); distance > probe_distance {
|
|
|
- if map_hash_is_deleted(element_hash) {
|
|
|
- k_dst := map_cell_index_dynamic(ks, info.ks, pos)
|
|
|
- v_dst := map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
- intrinsics.mem_copy_non_overlapping(rawptr(k_dst), rawptr(k), size_of_k)
|
|
|
- intrinsics.mem_copy_non_overlapping(rawptr(v_dst), rawptr(v), size_of_v)
|
|
|
- hp^ = h
|
|
|
-
|
|
|
- return result if result != 0 else v_dst
|
|
|
+ if map_hash_is_deleted(element_hash) {
|
|
|
+ next_pos := (pos + 1) & mask
|
|
|
+
|
|
|
+ // backward shift
|
|
|
+ for !map_hash_is_empty(hs[next_pos]) {
|
|
|
+ probe_distance := map_probe_distance(m^, hs[next_pos], next_pos)
|
|
|
+ if probe_distance == 0 {
|
|
|
+ break
|
|
|
+ }
|
|
|
+ probe_distance -= 1
|
|
|
+
|
|
|
+ kp := map_cell_index_dynamic(ks, info.ks, pos)
|
|
|
+ vp := map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
+ kn := map_cell_index_dynamic(ks, info.ks, next_pos)
|
|
|
+ vn := map_cell_index_dynamic(vs, info.vs, next_pos)
|
|
|
+
|
|
|
+ if distance > probe_distance {
|
|
|
+ if result == 0 {
|
|
|
+ result = vp
|
|
|
+ }
|
|
|
+ // move stored into pos; store next
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(kp), rawptr(k), size_of_k)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(vp), rawptr(v), size_of_v)
|
|
|
+ hs[pos] = h
|
|
|
+
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(k), rawptr(kn), size_of_k)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(v), rawptr(vn), size_of_v)
|
|
|
+ h = hs[next_pos]
|
|
|
+ } else {
|
|
|
+ // move next back 1
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(kp), rawptr(kn), size_of_k)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(vp), rawptr(vn), size_of_v)
|
|
|
+ hs[pos] = hs[next_pos]
|
|
|
+ distance = probe_distance
|
|
|
+ }
|
|
|
+ hs[next_pos] = 0
|
|
|
+ pos = (pos + 1) & mask
|
|
|
+ next_pos = (next_pos + 1) & mask
|
|
|
+ distance += 1
|
|
|
}
|
|
|
|
|
|
+ kp := map_cell_index_dynamic(ks, info.ks, pos)
|
|
|
+ vp := map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(kp), rawptr(k), size_of_k)
|
|
|
+ intrinsics.mem_copy_non_overlapping(rawptr(vp), rawptr(v), size_of_v)
|
|
|
+ hs[pos] = h
|
|
|
+
|
|
|
+ return result if result != 0 else vp
|
|
|
+ }
|
|
|
+
|
|
|
+ if probe_distance := map_probe_distance(m^, element_hash, pos); distance > probe_distance {
|
|
|
if result == 0 {
|
|
|
result = map_cell_index_dynamic(vs, info.vs, pos)
|
|
|
}
|
|
@@ -503,6 +562,7 @@ map_reserve_dynamic :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_
|
|
|
}
|
|
|
k := map_cell_index_dynamic(ks, info.ks, i)
|
|
|
v := map_cell_index_dynamic(vs, info.vs, i)
|
|
|
+ hash = info.key_hasher(rawptr(k), map_seed(resized))
|
|
|
_ = map_insert_hash_dynamic(&resized, info, hash, k, v)
|
|
|
// Only need to do this comparison on each actually added pair, so do not
|
|
|
// fold it into the for loop comparator as a micro-optimization.
|
|
@@ -550,6 +610,7 @@ map_shrink_dynamic :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_I
|
|
|
|
|
|
k := map_cell_index_dynamic(ks, info.ks, i)
|
|
|
v := map_cell_index_dynamic(vs, info.vs, i)
|
|
|
+ hash = info.key_hasher(rawptr(k), map_seed(shrunk))
|
|
|
_ = map_insert_hash_dynamic(&shrunk, info, hash, k, v)
|
|
|
// Only need to do this comparison on each actually added pair, so do not
|
|
|
// fold it into the for loop comparator as a micro-optimization.
|
|
@@ -581,7 +642,7 @@ map_lookup_dynamic :: proc "contextless" (m: Raw_Map, #no_alias info: ^Map_Info,
|
|
|
if map_len(m) == 0 {
|
|
|
return 0, false
|
|
|
}
|
|
|
- h := info.key_hasher(rawptr(k), 0)
|
|
|
+ h := info.key_hasher(rawptr(k), map_seed(m))
|
|
|
p := map_desired_position(m, h)
|
|
|
d := uintptr(0)
|
|
|
c := (uintptr(1) << map_log2_cap(m)) - 1
|
|
@@ -604,7 +665,7 @@ map_exists_dynamic :: proc "contextless" (m: Raw_Map, #no_alias info: ^Map_Info,
|
|
|
if map_len(m) == 0 {
|
|
|
return false
|
|
|
}
|
|
|
- h := info.key_hasher(rawptr(k), 0)
|
|
|
+ h := info.key_hasher(rawptr(k), map_seed(m))
|
|
|
p := map_desired_position(m, h)
|
|
|
d := uintptr(0)
|
|
|
c := (uintptr(1) << map_log2_cap(m)) - 1
|
|
@@ -637,7 +698,6 @@ map_erase_dynamic :: #force_inline proc "contextless" (#no_alias m: ^Raw_Map, #n
|
|
|
|
|
|
{ // coalesce tombstones
|
|
|
// HACK NOTE(bill): This is an ugly bodge but it is coalescing the tombstone slots
|
|
|
- // TODO(bill): we should do backward shift deletion and not rely on tombstone slots
|
|
|
mask := (uintptr(1)<<map_log2_cap(m^)) - 1
|
|
|
curr_index := uintptr(index)
|
|
|
|
|
@@ -711,7 +771,7 @@ map_get :: proc "contextless" (m: $T/map[$K]$V, key: K) -> (stored_key: K, store
|
|
|
info := intrinsics.type_map_info(T)
|
|
|
key := key
|
|
|
|
|
|
- h := info.key_hasher(&key, 0)
|
|
|
+ h := info.key_hasher(&key, map_seed(m))
|
|
|
pos := map_desired_position(rm, h)
|
|
|
distance := uintptr(0)
|
|
|
mask := (uintptr(1) << map_log2_cap(rm)) - 1
|
|
@@ -762,15 +822,15 @@ __dynamic_map_get :: proc "contextless" (#no_alias m: ^Raw_Map, #no_alias info:
|
|
|
}
|
|
|
|
|
|
// IMPORTANT: USED WITHIN THE COMPILER
|
|
|
-__dynamic_map_check_grow :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_Info, loc := #caller_location) -> Allocator_Error {
|
|
|
+__dynamic_map_check_grow :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_Info, loc := #caller_location) -> (err: Allocator_Error, has_grown: bool) {
|
|
|
if m.len >= map_resize_threshold(m^) {
|
|
|
- return map_grow_dynamic(m, info, loc)
|
|
|
+ return map_grow_dynamic(m, info, loc), true
|
|
|
}
|
|
|
- return nil
|
|
|
+ return nil, false
|
|
|
}
|
|
|
|
|
|
__dynamic_map_set_without_hash :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_Info, key, value: rawptr, loc := #caller_location) -> rawptr {
|
|
|
- return __dynamic_map_set(m, info, info.key_hasher(key, 0), key, value, loc)
|
|
|
+ return __dynamic_map_set(m, info, info.key_hasher(key, map_seed(m^)), key, value, loc)
|
|
|
}
|
|
|
|
|
|
|
|
@@ -781,9 +841,14 @@ __dynamic_map_set :: proc "odin" (#no_alias m: ^Raw_Map, #no_alias info: ^Map_In
|
|
|
return found
|
|
|
}
|
|
|
|
|
|
- if __dynamic_map_check_grow(m, info, loc) != nil {
|
|
|
+ hash := hash
|
|
|
+ err, has_grown := __dynamic_map_check_grow(m, info, loc)
|
|
|
+ if err != nil {
|
|
|
return nil
|
|
|
}
|
|
|
+ if has_grown {
|
|
|
+ hash = info.key_hasher(key, map_seed(m^))
|
|
|
+ }
|
|
|
|
|
|
result := map_insert_hash_dynamic(m, info, hash, uintptr(key), uintptr(value))
|
|
|
m.len += 1
|