From 4b15459edaaa3ec047f34864d4fc6c53197e804a Mon Sep 17 00:00:00 2001 From: Son Ho Date: Fri, 11 Feb 2022 23:26:04 +0100 Subject: Prove some high-level lemmas about find after insert --- tests/hashmap/Hashmap.Properties.fst | 71 +++++++++++++++++++++++++++++++++++- 1 file changed, 69 insertions(+), 2 deletions(-) (limited to 'tests') 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 -- cgit v1.2.3