287 lines
11 KiB
Plaintext
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_prod_case [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_prod_case [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 |