Theory Heap_Hash_Map
subsection ‹Heap Hash Map›
theory Heap_Hash_Map
imports
Separation_Logic_Imperative_HOL.Sep_Main Separation_Logic_Imperative_HOL.Sep_Examples
Refine_Imperative_HOL.IICF
begin
no_notation Ref.update ("_ := _" 62)
definition big_star :: "assn multiset ⇒ assn" ("⋀* _" [60] 90) where
"big_star S ≡ fold_mset (*) emp S"
interpretation comp_fun_commute_mult:
comp_fun_commute "(*) :: ('a :: ab_semigroup_mult ⇒ _ ⇒ _)"
by standard (auto simp: ab_semigroup_mult_class.mult.left_commute)
lemma sep_big_star_insert [simp]: "⋀* (add_mset x S) = (x * ⋀* S)"
by (auto simp: big_star_def)
lemma sep_big_star_union [simp]: "⋀* (S + T) = (⋀* S) * (⋀* T)"
by (auto simp: big_star_def comp_fun_commute_mult.fold_mset_fun_left_comm)
lemma sep_big_star_empty [simp]: "⋀* {#} = emp"
by (simp add: big_star_def)
lemma big_star_entatilst_mono:
"⋀* T ⟹⇩t ⋀* S" if "S ⊆# T"
using that
proof (induction T arbitrary: S)
case empty
then show ?case
by auto
next
case (add x T)
then show ?case
proof (cases "x ∈# S")
case True
then obtain S' where "S' ⊆# T" "S = add_mset x S'"
by (metis add.prems mset_add mset_subset_eq_add_mset_cancel)
with add.IH[of S'] add.prems have "⋀* T ⟹⇩t ⋀* S'"
by auto
then show ?thesis
unfolding ‹S = _› by (sep_auto intro: entt_star_mono)
next
case False
with add.prems have "S ⊆# T"
by (metis add_mset_remove_trivial diff_single_trivial mset_le_subtract)
then have "⋀* T ⟹⇩t ⋀* S"
by (rule add.IH)
then show ?thesis
using entt_fr_drop star_aci(2) by fastforce
qed
qed
definition "map_assn V m mi ≡
↑ (dom mi = dom m ∧ finite (dom m)) *
(⋀* {# V (the (m k)) (the (mi k)) . k ∈# mset_set (dom m)#})
"
lemma map_assn_empty_map[simp]:
"map_assn A Map.empty Map.empty = emp"
unfolding map_assn_def by auto
lemma in_mset_union_split:
"mset_set S = mset_set (S - {k}) + {#k#}" if "k ∈ S" "finite S"
using that by (auto simp: mset_set.remove)
lemma in_mset_dom_union_split:
"mset_set (dom m) = mset_set (dom m - {k}) + {#k#}" if "m k = Some v" "finite (dom m)"
apply (rule in_mset_union_split)
using that by (auto)
lemma dom_remove_not_in_dom_simp[simp]:
"dom m - {k} = dom m" if "m k = None"
using that unfolding dom_def by auto
lemma map_assn_delete:
"map_assn A m mh ⟹⇩A
map_assn A (m(k := None)) (mh(k := None)) * option_assn A (m k) (mh k)"
unfolding map_assn_def
apply sep_auto
apply (sep_auto cong: multiset.map_cong_simp)
apply (cases "m k"; cases "mh k"; simp)
apply (solves auto)+
by (subst in_mset_dom_union_split, assumption+, sep_auto)
lemma in_mset_set_iff_in_set[simp]:
"z ∈# mset_set S ⟷ z ∈ S" if "finite S"
using that by auto
lemma ent_refl':
"a = b ⟹ a ⟹⇩A b"
by auto
lemma map_assn_update_aux:
"map_assn A m mh * A v vi ⟹⇩A map_assn A (m(k ↦ v)) (mh(k ↦ vi))"
if "k ∉ dom m"
unfolding map_assn_def
apply (sep_auto simp: that cong: multiset.map_cong_simp)
apply (subst mult.commute)
apply (rule ent_star_mono)
apply simp
apply (rule ent_refl')
apply (rule arg_cong[where f = "big_star"])
apply (rule image_mset_cong)
using that by auto
lemma map_assn_update:
"map_assn A m mh * A v vi ⟹⇩A
map_assn A (m(k ↦ v)) (mh(k ↦ vi)) * true"
apply (rule ent_frame_fwd[OF map_assn_delete[where k = k]], frame_inference)
apply (rule ent_frame_fwd[OF map_assn_update_aux[where k = k and A = A], rotated], frame_inference)
by sep_auto+
definition (in imp_map) "hms_assn A m mi ≡ ∃⇩Amh. is_map mh mi * map_assn A m mh"
definition (in imp_map) "hms_assn' K A = hr_comp (hms_assn A) (⟨the_pure K, Id⟩map_rel)"
declare (in imp_map) hms_assn'_def[symmetric, fcomp_norm_unfold]
definition (in imp_map_empty) [code_unfold]: "hms_empty ≡ empty"
lemma (in imp_map_empty) hms_empty_rule [sep_heap_rules]:
"<emp> hms_empty <hms_assn A Map.empty>⇩t"
unfolding hms_empty_def hms_assn_def
by (sep_auto eintros: exI[where x = Map.empty])
definition (in imp_map_update) [code_unfold]: "hms_update = update"
lemma (in imp_map_update) hms_update_rule [sep_heap_rules]:
"<hms_assn A m mi * A v vi> hms_update k vi mi <hms_assn A (m(k ↦ v))>⇩t"
unfolding hms_update_def hms_assn_def
apply (sep_auto eintros del: exI)
subgoal for mh mi
apply (rule exI[where x = "mh(k ↦ vi)"])
apply (rule ent_frame_fwd[OF map_assn_update[where A = A]], frame_inference)
by sep_auto
done
lemma restrict_not_in_dom_simp[simp]:
"m |` (- {k}) = m" if "m k = None"
using that by (auto simp: restrict_map_def)
[code]:
" lookup delete k m =
do {
vo ← lookup k m;
case vo of
None ⇒ return (None, m) |
Some v ⇒ do {
m ← delete k m;
return (Some v, m)
}
}
"
definition [code]:
"hms_lookup lookup copy k m =
do {
vo ← lookup k m;
case vo of
None ⇒ return None |
Some v ⇒ do {
v' ← copy v;
return (Some v')
}
}
"
= imp_map_delete + imp_map_lookup
begin
lemma map_assn_domain_simps[simp]:
assumes "vassn_tag (map_assn A m mh)"
shows "mh k = None ⟷ m k = None" "dom mh = dom m" "finite (dom m)"
using assms unfolding map_assn_def vassn_tag_def by auto
lemma [sep_heap_rules]:
"<hms_assn A m mi>
hms_extract lookup delete k mi
<λ (vi, mi'). option_assn A (m k) vi * hms_assn A (m(k := None)) mi'>⇩t"
unfolding hms_extract_def hms_assn_def
apply (sep_auto eintros del: exI)
subgoal
unfolding map_assn_def by auto
subgoal for mh
apply (rule exI[where x = "mh(k := None)"])
apply (rule fr_refl)
apply (rule ent_star_mono, simp add: fun_upd_idem)
apply (rule entails_preI, simp add: fun_upd_idem)
done
apply (sep_auto eintros del: exI)
subgoal for mh
apply (rule exI[where x = "mh(k := None)"])
apply (rule ent_frame_fwd[OF map_assn_delete[where A = A]], frame_inference)
by (sep_auto simp add: map_upd_eq_restrict)+
done
lemma [sep_heap_rules]:
assumes
"(copy, RETURN o COPY) ∈ A⇧k →⇩a A"
shows
"<hms_assn A m mi>
hms_lookup lookup copy k mi
<λ vi. hms_assn A m mi * option_assn A (m k) vi>⇩t"
proof -
have 0: "
<is_map mh mi * map_assn A (m(k := None)) (mh(k := None)) * option_assn A (m k) (mh k) * true>
copy x'
<λr. ∃⇩Ax.
is_map mh mi * map_assn A (m(k := None)) (mh(k := None)) * option_assn A (m k) (mh k)
* A x r * ↑(m k = Some x)
* true>
" if "mh k = Some x'" for mh x'
supply [sep_heap_rules] = assms[to_hnr, unfolded hn_refine_def hn_ctxt_def, simplified]
by (sep_auto simp add: that option_assn_alt_def split: option.splits)
have 1: "
<is_map mh mi * map_assn A m mh * true>
copy x'
<λr. ∃⇩Ax. is_map mh mi * map_assn A m mh * A x r * ↑(m k = Some x) * true>"
if "mh k = Some x'" for mh x'
apply (rule cons_rule[rotated 2], rule 0, rule that)
apply (rule ent_frame_fwd[OF map_assn_delete[where A = A]], frame_inference, frame_inference)
apply sep_auto
apply (fr_rot 4)
apply (fr_rot_rhs 3)
apply (rule fr_refl)
apply (fr_rot 1)
apply (fr_rot_rhs 1)
apply (rule fr_refl)
apply (fr_rot 2)
apply (fr_rot_rhs 1)
unfolding option_assn_alt_def
apply (sep_auto split: option.split)
subgoal for x x'
apply (subgoal_tac "m(k ↦ x) = m")
apply (subgoal_tac "mh(k ↦ x') = mh")
using map_assn_update[of A "m(k := None)" "mh(k := None)" x x' k]
by (auto simp add: ent_true_drop(1))
done
show ?thesis
unfolding hms_lookup_def hms_assn_def
apply (sep_auto eintros del: exI)
subgoal
unfolding map_assn_def by auto
subgoal for mh
by (rule exI[where x = mh]) sep_auto
by (sep_auto intro: 1)
qed
end
context imp_map_update
begin
lemma hms_update_hnr:
"(uncurry2 hms_update, uncurry2 (RETURN ooo op_map_update)) ∈
id_assn⇧k *⇩a A⇧d *⇩a (hms_assn A)⇧d →⇩a hms_assn A"
by sepref_to_hoare sep_auto
sepref_decl_impl update: hms_update_hnr uses op_map_update.fref[where V = Id] .
end
context imp_map_empty
begin
lemma hms_empty_hnr:
"(uncurry0 hms_empty, uncurry0 (RETURN op_map_empty)) ∈ unit_assn⇧k →⇩a hms_assn A"
by sepref_to_hoare sep_auto
sepref_decl_impl (no_register) empty: hms_empty_hnr uses op_map_empty.fref[where V = Id] .
definition "op_hms_empty ≡ IICF_Map.op_map_empty"
sublocale hms: map_custom_empty op_hms_empty
by unfold_locales (simp add: op_hms_empty_def)
lemmas [sepref_fr_rules] = empty_hnr[folded op_hms_empty_def]
lemmas hms_fold_custom_empty = hms.fold_custom_empty
end
:
"λk m. (m k, m(k := None))" :: "K → ⟨K,V⟩map_rel → ⟨V⟩option_rel ×⇩r ⟨K,V⟩map_rel"
where "single_valued K" "single_valued (K¯)"
apply (rule fref_ncI)
apply parametricity
unfolding map_rel_def
apply (elim IntE)
apply parametricity
apply (elim IntE; rule IntI)
apply parametricity
apply (simp add: pres_eq_iff_svb; fail)
by auto
context imp_map_extract_derived
begin
lemma :
"(uncurry (hms_extract lookup delete), uncurry (RETURN oo op_map_extract)) ∈
id_assn⇧k *⇩a (hms_assn A)⇧d →⇩a prod_assn (option_assn A) (hms_assn A)"
by sepref_to_hoare sep_auto
lemma :
"(uncurry (hms_lookup lookup copy), uncurry (RETURN oo op_map_lookup)) ∈
id_assn⇧k *⇩a (hms_assn A)⇧k →⇩a option_assn A" if "(copy, RETURN o COPY) ∈ A⇧k →⇩a A"
using that by sepref_to_hoare sep_auto
sepref_decl_impl "": hms_extract_hnr uses op_map_extract.fref[where V = Id] .
end
hms_hm: imp_map_extract_derived is_hashmap hm_delete hm_lookup by standard
end