lh-l4v/lib/OptionMonadWP.thy

287 lines
11 KiB
Plaintext

(*
* Copyright 2014, NICTA
*
* This software may be distributed and modified according to the terms of
* the BSD 2-Clause license. Note that NO WARRANTY is provided.
* See "LICENSE_BSD2.txt" for details.
*
* @TAG(NICTA_BSD)
*)
(*
Hoare reasoning and WP (weakest-precondition) generator rules for the option monad.
This list is almost certainly incomplete; add rules here as they are needed.
*)
theory OptionMonadWP
imports
OptionMonadND
"wp/WP"
begin
declare K_def [simp]
(* Hoare triples.
TODO: design a sensible syntax for them. *)
(* Partial correctness. *)
definition ovalid :: "('s \<Rightarrow> bool) \<Rightarrow> ('s \<Rightarrow> 'a option) \<Rightarrow> ('a \<Rightarrow> 's \<Rightarrow> bool) \<Rightarrow> bool" where
"ovalid P f Q \<equiv> \<forall>s r. P s \<and> f s = Some r \<longrightarrow> Q r s"
(* Total correctness. *)
definition ovalidNF :: "('s \<Rightarrow> bool) \<Rightarrow> ('s \<Rightarrow> 'a option) \<Rightarrow> ('a \<Rightarrow> 's \<Rightarrow> bool) \<Rightarrow> bool" where
"ovalidNF P f Q \<equiv> \<forall>s. P s \<longrightarrow> (f s \<noteq> None \<and> (\<forall>r. f s = Some r \<longrightarrow> Q r s))"
(* Termination. *)
definition no_ofail where "no_ofail P f \<equiv> \<forall>s. P s \<longrightarrow> f s \<noteq> None"
(*
This rule lets us apply ovalidNF machinery for proving no_ofail.
However, we ought to eventually write working wp rules for no_ofail (see below).
*)
lemma no_ofail_is_ovalidNF: "no_ofail P f \<equiv> ovalidNF P f (\<lambda>_ _. True)"
by (simp add: no_ofail_def ovalidNF_def)
lemma ovalidNF_combine: "\<lbrakk> ovalid P f Q; no_ofail P f \<rbrakk> \<Longrightarrow> ovalidNF P f Q"
by (auto simp: ovalidNF_def ovalid_def no_ofail_def)
(* Annotating programs with loop invariant and measure. *)
definition owhile_inv ::
"('a \<Rightarrow> 's \<Rightarrow> bool) \<Rightarrow> ('a \<Rightarrow> 's \<Rightarrow> 'a option) \<Rightarrow> 'a
\<Rightarrow> ('a \<Rightarrow> 's \<Rightarrow> bool) \<Rightarrow> ('a \<Rightarrow> 's \<Rightarrow> nat) \<Rightarrow> 's \<Rightarrow> 'a option"
where "owhile_inv C B x I M = owhile C B x"
lemmas owhile_add_inv = owhile_inv_def[symmetric]
(* WP rules for ovalid. *)
lemma obind_wp [wp]:
"\<lbrakk> \<And>r. ovalid (R r) (g r) Q; ovalid P f R \<rbrakk> \<Longrightarrow> ovalid P (obind f g) Q"
by (simp add: ovalid_def obind_def split: option.splits, fast)
lemma oreturn_wp [wp]:
"ovalid (P x) (oreturn x) P"
by (simp add: ovalid_def oreturn_def K_def)
lemma ocondition_wp [wp]:
"\<lbrakk> ovalid L l Q; ovalid R r Q \<rbrakk>
\<Longrightarrow> ovalid (\<lambda>s. if C s then L s else R s) (ocondition C l r) Q"
by (auto simp: ovalid_def ocondition_def)
lemma ofail_wp [wp]:
"ovalid (\<lambda>_. True) ofail Q"
by (simp add: ovalid_def ofail_def)
lemma ovalid_K_bind_wp [wp]:
"ovalid P f Q \<Longrightarrow> ovalid P (K_bind f x) Q"
by simp
lemma ogets_wp [wp]: "ovalid (\<lambda>s. P (f s) s) (ogets f) P"
by (simp add: ovalid_def ogets_def)
lemma oguard_wp [wp]: "ovalid (\<lambda>s. f s \<longrightarrow> P () s) (oguard f) P"
by (simp add: ovalid_def oguard_def)
lemma oskip_wp [wp]:
"ovalid (\<lambda>s. P () s) oskip P"
by (simp add: ovalid_def oskip_def)
lemma ovalid_case_prod [wp]:
assumes "(\<And>x y. ovalid (P x y) (B x y) Q)"
shows "ovalid (case v of (x, y) \<Rightarrow> P x y) (case v of (x, y) \<Rightarrow> B x y) Q"
using assms unfolding ovalid_def by auto
lemma owhile_ovalid [wp]:
"\<lbrakk>\<And>a. ovalid (\<lambda>s. I a s \<and> C a s) (B a) I;
\<And>a s. \<lbrakk>I a s; \<not> C a s\<rbrakk> \<Longrightarrow> Q a s\<rbrakk>
\<Longrightarrow> ovalid (I a) (owhile_inv C B a I M) Q"
unfolding owhile_inv_def owhile_def ovalid_def
apply clarify
apply (frule_tac I = "\<lambda>a. I a s" in option_while_rule)
apply auto
done
definition ovalid_property where "ovalid_property P x = (\<lambda>s f. (\<forall>r. Some r = x s f \<longrightarrow> P r s))"
lemma ovalid_is_triple [wp_trip]:
"ovalid P f Q = triple_judgement P f (ovalid_property Q (\<lambda>s f. f s))"
by (auto simp: triple_judgement_def ovalid_def ovalid_property_def)
lemma ovalid_wp_comb1 [wp_comb]:
"\<lbrakk> ovalid P' f Q; ovalid P f Q'; \<And>s. P s \<Longrightarrow> P' s \<rbrakk> \<Longrightarrow> ovalid P f (\<lambda>r s. Q r s \<and> Q' r s)"
by (simp add: ovalid_def)
lemma ovalid_wp_comb2 [wp_comb]:
"\<lbrakk> ovalid P f Q; \<And>s. P' s \<Longrightarrow> P s \<rbrakk> \<Longrightarrow> ovalid P' f Q"
by (auto simp: ovalid_def)
lemma ovalid_wp_comb3 [wp_comb]:
"\<lbrakk> ovalid P f Q; ovalid P' f Q' \<rbrakk> \<Longrightarrow> ovalid (\<lambda>s. P s \<and> P' s) f (\<lambda>r s. Q r s \<and> Q' r s)"
by (auto simp: ovalid_def)
(* WP rules for ovalidNF. *)
lemma obind_NF_wp [wp]:
"\<lbrakk> \<And>r. ovalidNF (R r) (g r) Q; ovalidNF P f R \<rbrakk> \<Longrightarrow> ovalidNF P (obind f g) Q"
by (auto simp: ovalidNF_def obind_def split: option.splits)
lemma oreturn_NF_wp [wp]:
"ovalidNF (P x) (oreturn x) P"
by (simp add: ovalidNF_def oreturn_def)
lemma ocondition_NF_wp [wp]:
"\<lbrakk> ovalidNF L l Q; ovalidNF R r Q \<rbrakk>
\<Longrightarrow> ovalidNF (\<lambda>s. if C s then L s else R s) (ocondition C l r) Q"
by (simp add: ovalidNF_def ocondition_def)
lemma ofail_NF_wp [wp]:
"ovalidNF (\<lambda>_. False) ofail Q"
by (simp add: ovalidNF_def ofail_def)
lemma ovalidNF_K_bind_wp [wp]:
"ovalidNF P f Q \<Longrightarrow> ovalidNF P (K_bind f x) Q"
by simp
lemma ogets_NF_wp [wp]:
"ovalidNF (\<lambda>s. P (f s) s) (ogets f) P"
by (simp add: ovalidNF_def ogets_def)
lemma oguard_NF_wp [wp]:
"ovalidNF (\<lambda>s. f s \<and> P () s) (oguard f) P"
by (simp add: ovalidNF_def oguard_def)
lemma oskip_NF_wp [wp]:
"ovalidNF (\<lambda>s. P () s) oskip P"
by (simp add: ovalidNF_def oskip_def)
lemma ovalid_NF_case_prod [wp]:
assumes "(\<And>x y. ovalidNF (P x y) (B x y) Q)"
shows "ovalidNF (case v of (x, y) \<Rightarrow> P x y) (case v of (x, y) \<Rightarrow> B x y) Q"
using assms unfolding ovalidNF_def by auto
lemma owhile_NF [wp]:
"\<lbrakk>\<And>a. ovalidNF (\<lambda>s. I a s \<and> C a s) (B a) I;
\<And>a m. ovalid (\<lambda>s. I a s \<and> C a s \<and> M a s = m) (B a) (\<lambda>r s. M r s < m);
\<And>a s. \<lbrakk>I a s; \<not> C a s\<rbrakk> \<Longrightarrow> Q a s\<rbrakk>
\<Longrightarrow> ovalidNF (I a) (owhile_inv C B a I M) Q"
unfolding owhile_inv_def ovalidNF_def ovalid_def
apply clarify
apply (rule_tac I = I and M = "measure (\<lambda>r. M r s)" in owhile_rule)
apply fastforce
apply fastforce
apply fastforce
apply blast+
done
definition ovalidNF_property where "ovalidNF_property P x = (\<lambda>s f. (x s f \<noteq> None \<and> (\<forall>r. Some r = x s f \<longrightarrow> P r s)))"
lemma ovalidNF_is_triple [wp_trip]:
"ovalidNF P f Q = triple_judgement P f (ovalidNF_property Q (\<lambda>s f. f s))"
by (auto simp: triple_judgement_def ovalidNF_def ovalidNF_property_def)
lemma ovalidNF_wp_comb1 [wp_comb]:
"\<lbrakk> ovalidNF P' f Q; ovalidNF P f Q'; \<And>s. P s \<Longrightarrow> P' s \<rbrakk> \<Longrightarrow> ovalidNF P f (\<lambda>r s. Q r s \<and> Q' r s)"
by (simp add: ovalidNF_def)
lemma ovalidNF_wp_comb2 [wp_comb]:
"\<lbrakk> ovalidNF P f Q; \<And>s. P' s \<Longrightarrow> P s \<rbrakk> \<Longrightarrow> ovalidNF P' f Q"
by (simp add: ovalidNF_def)
lemma ovalidNF_wp_comb3 [wp_comb]:
"\<lbrakk> ovalidNF P f Q; ovalidNF P' f Q' \<rbrakk> \<Longrightarrow> ovalidNF (\<lambda>s. P s \<and> P' s) f (\<lambda>r s. Q r s \<and> Q' r s)"
by (simp add: ovalidNF_def)
(* FIXME: WP rules for no_ofail, which might not be correct. *)
lemma no_ofail_ofail [wp]: "no_ofail (\<lambda>_. False) ofail"
by (simp add: no_ofail_def ofail_def)
lemma no_ofail_ogets [wp]: "no_ofail (\<lambda>_. True) (ogets f)"
by (simp add: no_ofail_def ogets_def)
lemma no_ofail_obind [wp]:
"\<lbrakk> \<And>r. no_ofail (P r) (g r); no_ofail Q f; ovalid Q f P \<rbrakk> \<Longrightarrow> no_ofail Q (obind f g)"
by (auto simp: no_ofail_def obind_def ovalid_def)
lemma no_ofail_K_bind [wp]:
"no_ofail P f \<Longrightarrow> no_ofail P (K_bind f x)"
by simp
lemma no_ofail_oguard [wp]:
"no_ofail (\<lambda>s. f s) (oguard f)"
by (auto simp: no_ofail_def oguard_def)
lemma no_ofail_ocondition [wp]:
"\<lbrakk> no_ofail L l; no_ofail R r \<rbrakk>
\<Longrightarrow> no_ofail (\<lambda>s. if C s then L s else R s) (ocondition C l r)"
by (simp add: no_ofail_def ocondition_def)
lemma no_ofail_oreturn [wp]:
"no_ofail (\<lambda>_. True) (oreturn x)"
by (simp add: no_ofail_def oreturn_def)
lemma no_ofail_oskip [wp]:
"no_ofail (\<lambda>_. True) oskip"
by (simp add: no_ofail_def oskip_def)
lemma no_ofail_is_triple [wp_trip]:
"no_ofail P f = triple_judgement P f (\<lambda>s f. f s \<noteq> None)"
by (auto simp: triple_judgement_def no_ofail_def)
lemma no_ofail_wp_comb1 [wp_comb]:
"\<lbrakk> no_ofail P f; \<And>s. P' s \<Longrightarrow> P s \<rbrakk> \<Longrightarrow> no_ofail P' f"
by (simp add: no_ofail_def)
lemma no_ofail_wp_comb2 [wp_comb]:
"\<lbrakk> no_ofail P f; no_ofail P' f \<rbrakk> \<Longrightarrow> no_ofail (\<lambda>s. P s \<and> P' s) f"
by (simp add: no_ofail_def)
(* Some extra lemmas for our predicates. *)
lemma ovalid_grab_asm:
"(G \<Longrightarrow> ovalid P f Q) \<Longrightarrow> ovalid (\<lambda>s. G \<and> P s) f Q"
by (simp add: ovalid_def)
lemma ovalidNF_grab_asm:
"(G \<Longrightarrow> ovalidNF P f Q) \<Longrightarrow> ovalidNF (\<lambda>s. G \<and> P s) f Q"
by (simp add: ovalidNF_def)
lemma no_ofail_grab_asm:
"(G \<Longrightarrow> no_ofail P f) \<Longrightarrow> no_ofail (\<lambda>s. G \<and> P s) f"
by (simp add: no_ofail_def)
lemma ovalid_assume_pre:
"(\<And>s. P s \<Longrightarrow> ovalid P f Q) \<Longrightarrow> ovalid P f Q"
by (auto simp: ovalid_def)
lemma ovalidNF_assume_pre:
"(\<And>s. P s \<Longrightarrow> ovalidNF P f Q) \<Longrightarrow> ovalidNF P f Q"
by (simp add: ovalidNF_def)
lemma no_ofail_assume_pre:
"(\<And>s. P s \<Longrightarrow> no_ofail P f) \<Longrightarrow> no_ofail P f"
by (simp add: no_ofail_def)
lemma ovalid_pre_imp:
"\<lbrakk> \<And>s. P' s \<Longrightarrow> P s; ovalid P f Q \<rbrakk> \<Longrightarrow> ovalid P' f Q"
by (simp add: ovalid_def)
lemma ovalidNF_pre_imp:
"\<lbrakk> \<And>s. P' s \<Longrightarrow> P s; ovalidNF P f Q \<rbrakk> \<Longrightarrow> ovalidNF P' f Q"
by (simp add: ovalidNF_def)
lemma no_ofail_pre_imp:
"\<lbrakk> \<And>s. P' s \<Longrightarrow> P s; no_ofail P f \<rbrakk> \<Longrightarrow> no_ofail P' f"
by (simp add: no_ofail_def)
lemma ovalid_post_imp:
"\<lbrakk> \<And>r s. Q r s \<Longrightarrow> Q' r s; ovalid P f Q \<rbrakk> \<Longrightarrow> ovalid P f Q'"
by (simp add: ovalid_def)
lemma ovalidNF_post_imp:
"\<lbrakk> \<And>r s. Q r s \<Longrightarrow> Q' r s; ovalidNF P f Q \<rbrakk> \<Longrightarrow> ovalidNF P f Q'"
by (simp add: ovalidNF_def)
lemma ovalid_post_imp_assuming_pre:
"\<lbrakk> \<And>r s. \<lbrakk> P s; Q r s \<rbrakk> \<Longrightarrow> Q' r s; ovalid P f Q \<rbrakk> \<Longrightarrow> ovalid P f Q'"
by (simp add: ovalid_def)
lemma ovalidNF_post_imp_assuming_pre:
"\<lbrakk> \<And>r s. \<lbrakk> P s; Q r s \<rbrakk> \<Longrightarrow> Q' r s; ovalidNF P f Q \<rbrakk> \<Longrightarrow> ovalidNF P f Q'"
by (simp add: ovalidNF_def)
end