summaryrefslogtreecommitdiff
path: root/tests/hashmap
diff options
context:
space:
mode:
authorSon Ho2022-02-11 23:26:04 +0100
committerSon Ho2022-02-11 23:26:04 +0100
commit4b15459edaaa3ec047f34864d4fc6c53197e804a (patch)
tree33d76b0d38540af06bbd2d7684db31ec76b4039b /tests/hashmap
parent8265d379754eef7be20000bb2b2f4a4686371a22 (diff)
Prove some high-level lemmas about find after insert
Diffstat (limited to '')
-rw-r--r--tests/hashmap/Hashmap.Properties.fst71
1 files changed, 69 insertions, 2 deletions
diff --git a/tests/hashmap/Hashmap.Properties.fst b/tests/hashmap/Hashmap.Properties.fst
index 6cec0921..7436ec5f 100644
--- a/tests/hashmap/Hashmap.Properties.fst
+++ b/tests/hashmap/Hashmap.Properties.fst
@@ -466,6 +466,15 @@ let slot_t_inv_from_funs_lem
()
*)
+// TODO: use more
+type slot_s (t : Type0) = list (binding t)
+type slots_s (t : Type0) = list (slot_s t)
+
+let slots_inv (#t : Type0) (slots : slots_s t{length slots <= usize_max}) : Type0 =
+ forall(i:nat{i < length slots}).
+ {:pattern index slots i}
+ slot_inv (length slots) i (index slots i)
+
let slots_t_inv (#t : Type0) (slots : slots_t t{length slots <= usize_max}) : Type0 =
forall(i:nat{i < length slots}).
{:pattern index slots i}
@@ -494,6 +503,12 @@ let slots_t_from_fun
()
*)
+// TODO: hash_map_slots -> hash_map_slots_s
+let hash_map_slots_inv (#t : Type0) (hm : hash_map_slots t) : Type0 =
+ length hm <= usize_max /\
+ length hm > 0 /\
+ slots_inv hm
+
/// Base invariant for the hashmap (the complete invariant can be temporarily
/// broken between the moment we inserted an element and the moment we resize)
let hash_map_t_base_inv (#t : Type0) (hm : hash_map_t t) : Type0 =
@@ -1186,6 +1201,7 @@ let hash_map_insert_in_list_back_lem t len key value ls =
(**** insert_no_resize *)
+(***** Refinement proof *)
/// Same strategy as for [insert_in_list]: we introduce a high-level version of
/// the function, and reason about it.
/// We work on [hash_map_slots] (we use a higher-level view of the hash-map, but
@@ -1229,7 +1245,6 @@ val hash_map_insert_no_resize_fwd_back_lem
| _ -> False
end))
-#push-options "--z3rlimit 200"
let hash_map_insert_no_resize_fwd_back_lem t self key value =
begin match hash_key_fwd key with
| Fail -> ()
@@ -1298,7 +1313,59 @@ let hash_map_insert_no_resize_fwd_back_lem t self key value =
end
end
end
-#pop-options
+
+(***** find after insert *)
+/// Lemmas about what happens if we call [find] after an insertion
+
+val hash_map_insert_no_resize_s_get_same_lem
+ (#t : Type0) (hm : hash_map_slots t)
+ (key : usize) (value : t) :
+ Lemma (requires (hash_map_slots_inv hm))
+ (ensures (
+ match hash_map_insert_no_resize_s hm key value with
+ | Fail -> True
+ | Return hm' ->
+ hash_map_slots_find_s hm' key == Some value))
+
+let hash_map_insert_no_resize_s_get_same_lem #t hm key value =
+ let num_entries = length (flatten hm) in
+ if None? (hash_map_slots_find_s hm key) && num_entries = usize_max then ()
+ else
+ begin
+ let hm' = Return?.v (hash_map_insert_no_resize_s hm key value) in
+ let len = length hm in
+ let i = hash_mod_key key len in
+ let slot = index hm i in
+ hash_map_insert_in_list_s_lem t len key value slot
+ end
+
+val hash_map_insert_no_resize_s_get_diff_lem
+ (#t : Type0) (hm : hash_map_slots t)
+ (key : usize) (value : t) (key' : usize{key' <> key}) :
+ Lemma (requires (hash_map_slots_inv hm))
+ (ensures (
+ match hash_map_insert_no_resize_s hm key value with
+ | Fail -> True
+ | Return hm' ->
+ hash_map_slots_find_s hm' key' == hash_map_slots_find_s hm key'))
+
+let hash_map_insert_no_resize_s_get_diff_lem #t hm key value key' =
+ let num_entries = length (flatten hm) in
+ if None? (hash_map_slots_find_s hm key) && num_entries = usize_max then ()
+ else
+ begin
+ let hm' = Return?.v (hash_map_insert_no_resize_s hm key value) in
+ let len = length hm in
+ let i = hash_mod_key key len in
+ let slot = index hm i in
+ hash_map_insert_in_list_s_lem t len key value slot;
+ let i' = hash_mod_key key' len in
+ if i <> i' then ()
+ else
+ begin
+ ()
+ end
+ end
(*
/// We also do a disjunction over the cases