Theory Sepref_Rules
section ‹Refinement Rule Management›
theory Sepref_Rules
imports Sepref_Basic Sepref_Constraints
begin
text ‹This theory contains tools for managing the refinement rules used by Sepref›
text ‹The theories are based on uncurried functions, i.e.,
every function has type @{typ "'a⇒'b"}, where @{typ 'a} is the
tuple of parameters, or unit if there are none.
›
definition "non_dep R ≡ ∀b. R b = R undefined"
lemma non_dep_simp: "non_dep R ⟹ NO_MATCH undefined x ⟹ R x = R undefined"
by (auto simp: non_dep_def)
lemma non_dep_K[simp, intro!]: "non_dep (λ_. c)"
by (auto simp: non_dep_def)
definition "non_dep2 R ≡ ∀a b. R a b = R undefined undefined"
lemma non_dep2_simp:
"non_dep2 R ⟹ NO_MATCH undefined x ⟹ R x y = R undefined undefined"
"non_dep2 R ⟹ NO_MATCH undefined y ⟹ R x y = R undefined undefined"
by (auto simp: non_dep2_def)
lemma non_dep2_K[simp, intro!]: "non_dep2 (λ_ _. c)"
by (auto simp: non_dep2_def)
subsection ‹Assertion Interface Binding›
text ‹Binding of interface types to refinement assertions›
definition intf_of_assn :: "('a ⇒ _ ⇒ assn) ⇒ 'b itself ⇒ bool" where
[simp]: "intf_of_assn a b = True"
lemma intf_of_assnI: "intf_of_assn R TYPE('a)" by simp
named_theorems_rev intf_of_assn ‹Links between refinement assertions and interface types›
lemma intf_of_assn_fallback: "intf_of_assn (R :: 'a ⇒ _ ⇒ assn) TYPE('a)" by simp
subsection ‹Function Refinement with Precondition›
definition fref :: "('c ⇒ bool) ⇒ ('a × 'c) set ⇒ ('c ⇒ ('b × 'd) set)
⇒ (('a ⇒ 'b) × ('c ⇒ 'd)) set"
("[_]⇩f⇩d _ → _" [0,60,60] 60)
where "[P]⇩f⇩d R → S ≡ {(f,g). ∀x y. P y ∧ (x,y)∈R ⟶ (f x, g y)∈S y}"
abbreviation freft ("_ →⇩f⇩d _" [60,60] 60) where "R →⇩f⇩d S ≡ ([λ_. True]⇩f⇩d R → S)"
abbreviation freftnd ("_ →⇩f _" [60,60] 60) where "R →⇩f S ≡ ([λ_. True]⇩f⇩d R → (λ_. S))"
abbreviation frefnd ("[_]⇩f _ → _" [0,60,60] 60) where "[P]⇩f R → S ≡ [P]⇩f⇩d R → (λ_. S)"
lemma rel2p_fref[rel2p]: "rel2p (fref P R S)
= (λf g. (∀x y. P y ⟶ rel2p R x y ⟶ rel2p (S y) (f x) (g y)))"
by (auto simp: fref_def rel2p_def[abs_def])
lemma fref_cons:
assumes "(f,g) ∈ [P]⇩f⇩d R → S"
assumes "⋀c a. (c,a)∈R' ⟹ Q a ⟹ P a"
assumes "R' ⊆ R"
assumes "⋀c a. ⟦(c,a)∈R'; Q a⟧ ⟹ S a ⊆ S' a"
shows "(f,g) ∈ [Q]⇩f⇩d R' → S'"
using assms
unfolding fref_def
by fastforce
lemmas fref_cons' = fref_cons[OF _ _ order_refl order_refl]
lemma frefI[intro?]:
assumes "⋀x y. ⟦P y; (x,y)∈R⟧ ⟹ (f x, g y)∈S y"
shows "(f,g)∈fref P R S"
using assms
unfolding fref_def
by auto
lemma fref_ncI: "(f,g)∈R→S ⟹ (f,g)∈R→⇩fS"
apply (rule frefI)
apply parametricity
done
lemma frefD:
assumes "(f,g)∈fref P R S"
shows "⟦P y; (x,y)∈R⟧ ⟹ (f x, g y)∈S y"
using assms
unfolding fref_def
by auto
lemma fref_ncD: "(f,g)∈R→⇩fS ⟹ (f,g)∈R→S"
apply (rule fun_relI)
apply (drule frefD)
apply simp
apply assumption+
done
definition "rr_comp S R1 R2 x ≡ if non_dep R1 then R1 undefined O R2 x else (⋃{R1 y | y. (y,x)∈S}) O R2 x"
lemma rr_comp_K[simp]: "rr_comp S (λ_. R1) R2 = (λx. R1 O R2 x)"
by (auto simp: rr_comp_def fun_eq_iff)
lemma rr_comp_nondep: "rr_comp T (λ_. A) R = (λx. A O (R x))"
unfolding rr_comp_def
by (auto simp: fun_eq_iff)
lemma fref_compI:
"fref P R1 R2 O fref Q S1 S2 ⊆
fref (λx. Q x ∧ (∀y. (y,x)∈S1 ⟶ P y)) (R1 O S1) (rr_comp S1 R2 S2)"
unfolding fref_def rr_comp_def
apply (cases "non_dep R2"; simp)
subgoal by (fastforce simp: non_dep_simp[of R2])
subgoal by fastforce
done
lemma fref_compI':
"⟦ (f,g)∈fref P R1 R2; (g,h)∈fref Q S1 S2 ⟧
⟹ (f,h) ∈ fref (λx. Q x ∧ (∀y. (y,x)∈S1 ⟶ P y)) (R1 O S1) (rr_comp S1 R2 S2)"
using fref_compI[of P R1 R2 Q S1 S2]
by auto
lemma fref_unit_conv:
"(λ_. c, λ_. a) ∈ fref P unit_rel S ⟷ (P () ⟶ (c,a)∈S ())"
by (auto simp: fref_def)
lemma fref_uncurry_conv:
"(uncurry c, uncurry a) ∈ fref P (R1×⇩rR2) S
⟷ (∀x1 y1 x2 y2. P (y1,y2) ⟶ (x1,y1)∈R1 ⟶ (x2,y2)∈R2 ⟶ (c x1 x2, a y1 y2) ∈ S (y1,y2))"
by (auto simp: fref_def)
lemma fref_mono: "⟦ ⋀x. P' x ⟹ P x; R' ⊆ R; ⋀x y. P' x ⟹ S x ⊆ S' x ⟧
⟹ fref P R S ⊆ fref P' R' S'"
unfolding fref_def
by auto blast
lemma fref_composeI:
assumes FR1: "(f,g)∈fref P R1 R2"
assumes FR2: "(g,h)∈fref Q S1 S2"
assumes C1: "⋀x. P' x ⟹ Q x"
assumes C2: "⋀x y. ⟦P' x; (y,x)∈S1⟧ ⟹ P y"
assumes R1: "R' ⊆ R1 O S1"
assumes R2: "⋀x. ⟦ P' x ⟧ ⟹ rr_comp S1 R2 S2 x ⊆ S' x"
assumes FH: "f'=f" "h'=h"
shows "(f',h') ∈ fref P' R' S'"
unfolding FH
apply (rule set_mp[OF fref_mono fref_compI'[OF FR1 FR2]])
using C1 C2 apply blast
using R1 apply blast
using R2 apply blast
done
lemma fref_triv: "A⊆Id ⟹ (f,f)∈[P]⇩f A → Id"
by (auto simp: fref_def)
subsection ‹Heap-Function Refinement›
text ‹
The following relates a heap-function with a pure function.
It contains a precondition, a refinement assertion for the arguments
before and after execution, and a refinement relation for the result.
›
definition hfref
:: "
('a ⇒ bool)
⇒ ('ai ⇒ bool)
⇒ (('a ⇒ 'ai ⇒ assn) × ('a ⇒ 'ai ⇒ assn))
⇒ ('a ⇒ 'b ⇒ 'bi ⇒ assn)
⇒ ('ai ⇒ 'bi ⇒ bool)
⇒ (('ai ⇒ 'bi llM) × ('a⇒'b nres)) set"
("[_]⇩a [_]⇩c _ →⇩d _ [_]⇩c" [0,0,60,60,0] 60)
where
"[P]⇩a [C]⇩c RS →⇩d T [CP]⇩c ≡ { (f,g) . ∀c a. P a ∧ C c ⟶ hn_refine (fst RS a c) (f c) (snd RS a c) (T a) (CP c) (g a)}"
abbreviation hfrefcpt ("[_]⇩a _ →⇩d _" [0,60,60] 60) where "[P]⇩a RS →⇩d T ≡ ([P]⇩a [λ_. True]⇩c RS →⇩d T [λ_ _. True]⇩c)"
abbreviation hfrefpt ("[_]⇩c _ →⇩d _ [_]⇩c" [0,60,60,0] 60) where "[C]⇩c RS →⇩d T [CP]⇩c ≡ ([λ_. True]⇩a [C]⇩c RS →⇩d T [CP]⇩c)"
abbreviation hfreftt ("_ →⇩a⇩d _" [60,60] 60) where "RS →⇩a⇩d T ≡ ([λ_. True]⇩a RS →⇩d T)"
abbreviation hfrefcptnd ("[_]⇩a _ → _" [0,60,60] 60) where "[P]⇩a RS → T ≡ [P]⇩a RS →⇩d (λ_. T)"
abbreviation hfrefptnd ("[_]⇩c _ → _ [_]⇩c" [0,60,60,0] 60) where "[C]⇩c RS → T [CP]⇩c ≡ [C]⇩c RS →⇩d (λ_. T) [CP]⇩c"
abbreviation hfrefttnd ("_ →⇩a _" [60,60] 60) where "RS →⇩a T ≡ RS →⇩a⇩d (λ_. T)"
lemma hfrefI[intro?]:
assumes "⋀c a. P a ⟹ C c ⟹ hn_refine (fst RS a c) (f c) (snd RS a c) (T a) (CP c) (g a)"
shows "(f,g)∈hfref P C RS T CP"
using assms unfolding hfref_def by blast
lemma hfrefD:
assumes "(f,g)∈hfref P C RS T CP"
shows "⋀c a. P a ⟹ C c ⟹ hn_refine (fst RS a c) (f c) (snd RS a c) (T a) (CP c) (g a)"
using assms unfolding hfref_def by blast
lemma hfref_to_ASSERT_conv:
"NO_MATCH (λ_. True) P ⟹ (a,b)∈[P]⇩a [C]⇩c R →⇩d S [CP]⇩c ⟷ (a,λx. doN {ASSERT (P x); b x}) ∈ [C]⇩c R →⇩d S [CP]⇩c"
unfolding hfref_def
apply (clarsimp; safe; clarsimp?)
apply (rule hn_refine_nofailI)
apply (simp add: refine_pw_simps)
subgoal for xc xa
apply (drule spec[of _ xc])
apply simp
apply (drule spec[of _ xa])
by simp
done
text ‹
A pair of argument refinement assertions can be created by the
input assertion and the information whether the parameter is kept or destroyed
by the function.
›
primrec hf_pres
:: "('a ⇒ 'b ⇒ assn) ⇒ bool ⇒ ('a ⇒ 'b ⇒ assn)×('a ⇒ 'b ⇒ assn)"
where
"hf_pres R True = (R,R)" | "hf_pres R False = (R,invalid_assn R)"
abbreviation hfkeep
:: "('a ⇒ 'b ⇒ assn) ⇒ ('a ⇒ 'b ⇒ assn)×('a ⇒ 'b ⇒ assn)"
("(_⇧k)" [1000] 999)
where "R⇧k ≡ hf_pres R True"
abbreviation hfdrop
:: "('a ⇒ 'b ⇒ assn) ⇒ ('a ⇒ 'b ⇒ assn)×('a ⇒ 'b ⇒ assn)"
("(_⇧d)" [1000] 999)
where "R⇧d ≡ hf_pres R False"
abbreviation "hn_kede R kd ≡ hn_ctxt (snd (hf_pres R kd))"
abbreviation "hn_keep R ≡ hn_kede R True"
abbreviation "hn_dest R ≡ hn_kede R False"
lemma keep_drop_sels[simp]:
"fst (R⇧k) = R"
"snd (R⇧k) = R"
"fst (R⇧d) = R"
"snd (R⇧d) = invalid_assn R"
by auto
lemma hf_pres_fst[simp]: "fst (hf_pres R k) = R" by (cases k) auto
text ‹
The following operator combines multiple argument assertion-pairs to
argument assertion-pairs for the product. It is required to state
argument assertion-pairs for uncurried functions.
›
definition hfprod :: "
(('a ⇒ 'b ⇒ assn)×('a ⇒ 'b ⇒ assn))
⇒ (('c ⇒ 'd ⇒ assn)×('c ⇒ 'd ⇒ assn))
⇒ ((('a×'c) ⇒ ('b × 'd) ⇒ assn) × (('a×'c) ⇒ ('b × 'd) ⇒ assn))"
(infixl "*⇩a" 65)
where "RR *⇩a SS ≡ (prod_assn (fst RR) (fst SS), prod_assn (snd RR) (snd SS))"
lemma hfprod_fst_snd[simp]:
"fst (A *⇩a B) = prod_assn (fst A) (fst B)"
"snd (A *⇩a B) = prod_assn (snd A) (snd B)"
unfolding hfprod_def by auto
subsubsection ‹Conversion from fref to hfref›
lemma fref_to_pure_hfref':
assumes "(f,g) ∈ [P]⇩f R→(⟨S⟩nres_rel)"
assumes "⋀x. x∈Domain R ∩ R¯``Collect P ⟹ f x = RETURN (f' x)"
shows "(Mreturn o f', g) ∈ [P]⇩a (pure R)⇧k→⇩d(λ_. pure S)"
proof -
{
fix c a
assume A: "P a" "nofail (g a)" "(c, a) ∈ R"
hence "∃x. (f' c, x) ∈ S ∧ RETURN x ≤ g a"
using assms
by (fastforce simp: fref_def pw_le_iff pw_nres_rel_iff refine_pw_simps)
} note AUX=this
show ?thesis
apply (rule hfrefI) apply (rule hn_refineI)
unfolding pure_def
apply vcg
apply (frule (2) AUX)
apply vcg
.
qed
subsubsection ‹Conversion from hfref to hnr›
text ‹This section contains the lemmas. The ML code is further down. ›
lemma hf2hnr:
assumes "(f,g) ∈ [P]⇩a [C]⇩c R →⇩d S [CP]⇩c"
shows "∀x xi. P x ∧ C xi ⟶ hn_refine (hn_ctxt (fst R) x xi ** □) (f$xi) (hn_ctxt (snd R) x xi ** □) (S x) (CP xi) (g$x)"
using assms
unfolding hfref_def
by (auto simp: hn_ctxt_def)
definition [simp]: "to_hnr_prod ≡ prod_assn"
lemma to_hnr_prod_fst_snd:
"fst (A *⇩a B) = to_hnr_prod (fst A) (fst B)"
"snd (A *⇩a B) = to_hnr_prod (snd A) (snd B)"
unfolding hfprod_def by auto
lemma hnr_uncurry_unfold: "
(∀x xi. P x ⟶
hn_refine
(hn_ctxt (to_hnr_prod A B) x xi ** Γ)
(fi xi)
(hn_ctxt (to_hnr_prod A' B') x xi ** Γ')
(R x) (CP xi)
(f x))
⟷ (∀b bi a ai. P (a,b) ⟶
hn_refine
(hn_ctxt A a ai ** hn_ctxt B b bi ** Γ)
(fi (ai,bi))
(hn_ctxt A' a ai ** hn_ctxt B' b bi ** Γ')
(R (a,b)) (CP (ai,bi))
(f (a,b))
)"
by (auto simp: hn_ctxt_def prod_assn_def sep_conj_c)
lemma hn_ctxt_ctxt_fix_conv: "hn_ctxt (hn_ctxt R) = hn_ctxt R"
by (simp add: hn_ctxt_def[abs_def])
lemma uncurry_APP: "uncurry f$(a,b) = f$a$b" by auto
lemma norm_RETURN_o:
"⋀f. (RETURN o f)$x = (RETURN$(f$x))"
"⋀f. (RETURN oo f)$x$y = (RETURN$(f$x$y))"
"⋀f. (RETURN ooo f)$x$y$z = (RETURN$(f$x$y$z))"
"⋀f. (λx. RETURN ooo f x)$x$y$z$a = (RETURN$(f$x$y$z$a))"
"⋀f. (λx y. RETURN ooo f x y)$x$y$z$a$b = (RETURN$(f$x$y$z$a$b))"
by auto
lemma norm_return_o:
"⋀f. (return o f)$x = (return$(f$x))"
"⋀f. (return oo f)$x$y = (return$(f$x$y))"
"⋀f. (return ooo f)$x$y$z = (return$(f$x$y$z))"
"⋀f. (λx. return ooo f x)$x$y$z$a = (return$(f$x$y$z$a))"
"⋀f. (λx y. return ooo f x y)$x$y$z$a$b = (return$(f$x$y$z$a$b))"
by auto
lemma hn_val_unit_conv_emp[simp]: "hn_val unit_rel x y = □"
by (auto simp: hn_ctxt_def pure_def sep_algebra_simps)
subsubsection ‹Conversion from hnr to hfref›
text ‹This section contains the lemmas. The ML code is further down. ›
abbreviation "id_assn ≡ pure Id"
abbreviation "unit_assn ≡ id_assn :: unit ⇒ _"
lemma pure_unit_rel_eq_empty: "unit_assn x y = □"
by (auto simp: pure_def sep_algebra_simps)
lemma uc_hfprod_sel:
"fst (A *⇩a B) a c = (case (a,c) of ((a1,a2),(c1,c2)) ⇒ fst A a1 c1 ** fst B a2 c2)"
"snd (A *⇩a B) a c = (case (a,c) of ((a1,a2),(c1,c2)) ⇒ snd A a1 c1 ** snd B a2 c2)"
by auto
subsubsection ‹Conversion from relation to fref›
text ‹This section contains the lemmas. The ML code is further down. ›
definition "CURRY R ≡ { (f,g). (uncurry f, uncurry g) ∈ R }"
lemma fref_param1: "R→S = R→⇩fS"
by (auto simp: fref_def fun_relD)
lemma fref_nest: "[P1]⇩f R1 → ([P2]⇩f R2 → S)
≡ CURRY ([(λ(a,b). P1 a ∧ P2 b)]⇩f (R1×⇩rR2) → S)"
apply (rule eq_reflection)
by (auto simp: fref_def CURRY_def)
lemma in_CURRY_conv: "(f,g) ∈ CURRY R ⟷ (uncurry f, uncurry g) ∈ R"
unfolding CURRY_def by auto
lemma uncurry0_APP[simp]: "uncurry0 c $ x = c" by auto
lemma fref_param0I: "(c,a)∈R ⟹ (uncurry0 c, uncurry0 a) ∈ unit_rel →⇩f R"
by (auto simp: fref_def)
subsubsection ‹Composition›
definition hr_comp :: "('b ⇒ 'c ⇒ assn) ⇒ ('b × 'a) set ⇒ 'a ⇒ 'c ⇒ assn"
where "hr_comp R1 R2 a c ≡ EXS b. R1 b c ** ↑((b,a)∈R2)"
definition hrp_comp
:: "('d ⇒ 'b ⇒ assn) × ('d ⇒ 'c ⇒ assn)
⇒ ('d × 'a) set ⇒ ('a ⇒ 'b ⇒ assn) × ('a ⇒ 'c ⇒ assn)"
where "hrp_comp RR' S ≡ (hr_comp (fst RR') S, hr_comp (snd RR') S) "
lemma hr_compI: "(b,a)∈R2 ⟹ R1 b c ⊢ hr_comp R1 R2 a c"
unfolding hr_comp_def
by (auto simp: sep_algebra_simps entails_def)
lemma hr_comp_Id1[simp]: "hr_comp (pure Id) R = pure R"
unfolding hr_comp_def[abs_def] pure_def
apply (intro ext)
by (auto simp: pred_lift_extract_simps)
lemma hr_comp_Id2[simp]: "hr_comp R Id = R"
unfolding hr_comp_def[abs_def]
apply (intro ext)
by (auto simp: sep_algebra_simps)
lemma hr_comp_emp[simp]: "hr_comp (λa c. □) R a c = ↑(∃b. (b,a)∈R)"
unfolding hr_comp_def[abs_def]
apply (intro ext)
by (auto simp: sep_algebra_simps)
lemma hr_comp_prod_conv[simp]:
"hr_comp (prod_assn Ra Rb) (Ra' ×⇩r Rb')
= prod_assn (hr_comp Ra Ra') (hr_comp Rb Rb')"
unfolding hr_comp_def[abs_def] prod_assn_def[abs_def]
apply (intro ext)
apply (auto 0 3 simp: sep_algebra_simps)
done
lemma hr_comp_pure: "hr_comp (pure R) S = pure (R O S)"
apply (intro ext)
unfolding hr_comp_def[abs_def] pure_def
by (auto simp: sep_algebra_simps)
lemma hr_comp_is_pure[safe_constraint_rules]: "is_pure A ⟹ is_pure (hr_comp A B)"
by (auto simp: hr_comp_pure is_pure_conv)
lemma hr_comp_the_pure: "is_pure A ⟹ the_pure (hr_comp A B) = the_pure A O B"
unfolding is_pure_conv
by (clarsimp simp: hr_comp_pure)
lemma rdomp_hrcomp_conv[simp]: "rdomp (hr_comp A R) x ⟷ (∃y. rdomp A y ∧ (y,x)∈R)"
by (auto simp: rdomp_def hr_comp_def sep_algebra_simps)
lemma hn_rel_compI:
"⟦nofail a; (b,a)∈⟨R2⟩nres_rel⟧ ⟹ hn_rel R1 b c ⊢ hn_rel (hr_comp R1 R2) a c"
unfolding hr_comp_def hn_rel_def nres_rel_def entails_def
apply (auto simp: sep_algebra_simps)
apply (drule (1) order_trans)
apply (auto simp add: ret_le_down_conv)
done
lemma hr_comp_assoc: "hr_comp (hr_comp R S) T = hr_comp R (S O T)"
apply (intro ext)
unfolding hr_comp_def
by (auto simp: sep_algebra_simps)
lemma hrp_comp_Id1[simp]: "hrp_comp (hf_pres (pure Id) pp) R = hf_pres (pure R) pp"
unfolding hrp_comp_def apply (cases pp) apply auto
by (auto simp: hr_comp_def[abs_def] invalid_assn_def[abs_def] fun_eq_iff sep_algebra_simps)
lemma hrp_comp_Id2[simp]: "hrp_comp A Id = A"
unfolding hrp_comp_def by auto
lemma hnr_comp_aux:
assumes "RETURN x ≤ m" "nofail m'" "(m,m')∈⟨R⟩nres_rel"
obtains x' where "(x,x')∈R" "RETURN x' ≤ m'"
by (meson assms(1) assms(2) assms(3) inres_def nres_relD pwD2 pw_conc_inres)
definition "hrr_comp R R1 R2 x a c ≡
if non_dep R1 then
hr_comp (R1 undefined) (R2 x) a c
else
EXS b. ↑((b,x)∈R) ** hr_comp (R1 b) (R2 x) a c"
lemma hnr_comp:
assumes R: "⋀b1. P b1 ⟹ hn_refine (R1 b1 c1 ** Γ) (c c1) (R1p b1 c1 ** Γ') (R b1) (CP c1) (b b1)"
assumes S: "⋀a1 b1. ⟦Q a1; (b1,a1)∈R1'⟧ ⟹ (b b1,a a1)∈⟨R' a1⟩nres_rel"
assumes PQ: "⋀a1 b1. ⟦Q a1; (b1,a1)∈R1'⟧ ⟹ P b1"
assumes Q: "Q a1"
shows "hn_refine
(hr_comp R1 R1' a1 c1 ** Γ)
(c c1)
(hr_comp R1p R1' a1 c1 ** Γ')
(hrr_comp R1' R R' a1)
(CP c1)
(a a1)"
proof -
note [vcg_rules] = R[THEN hn_refineD]
have [simp]: "nofail (b x)" if "nofail (a a1)" "(x, a1) ∈ R1'" for x
using that Q S nres_rel_def pw_ref_iff by fastforce
show ?thesis
unfolding hn_refine_alt
unfolding hr_comp_def hn_rel_def hrr_comp_def
apply (cases "non_dep R"; simp)
subgoal premises prems
apply (auto simp: sep_algebra_simps simp del: pred_lift_extract_simps)
using PQ[OF Q]
supply [simp] = non_dep_simp[OF prems]
apply vcg
apply (frule S[OF Q])
apply (erule (2) hnr_comp_aux)
by vcg_try_solve
subgoal
apply (auto simp: sep_algebra_simps simp del: pred_lift_extract_simps)
using PQ[OF Q]
apply vcg
apply (frule S[OF Q])
apply (erule (2) hnr_comp_aux)
by vcg_try_solve
done
qed
lemma hnr_comp1_aux:
assumes R: "⋀b1. P b1 ⟹ hn_refine (hn_ctxt R1 b1 c1) (c c1) (hn_ctxt R1p b1 c1) (R b1) (CP c1) (b$b1)"
assumes S: "⋀a1 b1. ⟦Q a1; (b1,a1)∈R1'⟧ ⟹ (b$b1,a$a1)∈⟨R' a1⟩nres_rel"
assumes PQ: "⋀a1 b1. ⟦Q a1; (b1,a1)∈R1'⟧ ⟹ P b1"
assumes Q: "Q a1"
shows "hn_refine
(hr_comp R1 R1' a1 c1)
(c c1)
(hr_comp R1p R1' a1 c1)
(hrr_comp R1' R R' a1)
(CP c1)
(a a1)"
using assms hnr_comp[where Γ=□ and Γ'=□ and a=a and b=b and c=c and P=P and Q=Q]
unfolding hn_ctxt_def
by auto
lemma hfcomp:
assumes A: "(f,g) ∈ [P]⇩a [C]⇩c RR' →⇩d S [CP]⇩c"
assumes B: "(g,h) ∈ [Q]⇩f⇩d T → (λx. ⟨U x⟩nres_rel)"
shows "(f,h) ∈ [λa. Q a ∧ (∀a'. (a',a)∈T ⟶ P a')]⇩a [C]⇩c
hrp_comp RR' T →⇩d hrr_comp T S U [CP]⇩c"
using assms
unfolding fref_def hfref_def hrp_comp_def
apply clarsimp
subgoal for c a
apply (rule hnr_comp1_aux[of
P "fst RR'" c f "snd RR'" S _ g "λa. Q a ∧ (∀a'. (a',a)∈T ⟶ P a')" T h U])
apply (auto simp: hn_ctxt_def)
done
done
lemma hrr_comp_nondep: "hrr_comp T (λ_. A) R = (λx. hr_comp A (R x))"
unfolding hrr_comp_def
by (auto simp: fun_eq_iff)
lemma hrr_comp_Id_R_Id: "hrr_comp Id R1 (λ_. Id) = R1"
by (auto simp: hrr_comp_def fun_eq_iff pred_lift_extract_simps non_dep_simp[of R1])
lemma hrr_comp_id_conv[simp]: "hrr_comp Id R1 R2 = (λx. hr_comp (R1 x) (R2 x))"
unfolding hrr_comp_def
by (auto simp: fun_eq_iff pred_lift_extract_simps non_dep_simp[of R1])
lemma hfref_weaken_pre_nofail:
assumes "(f,g) ∈ [P]⇩a [C]⇩c R →⇩d S [CP]⇩c"
shows "(f,g) ∈ [λx. nofail (g x) ⟶ P x]⇩a [C]⇩c R →⇩d S [CP]⇩c"
using assms
unfolding hfref_def hn_refine_def
by auto
lemma hfref_cons:
assumes "(f,g) ∈ [P]⇩a [C]⇩c R →⇩d S [CP]⇩c"
assumes "⋀x. P' x ⟹ P x"
assumes "⋀x. C' x ⟹ C x"
assumes "⋀x y. fst R' x y ⊢ fst R x y"
assumes "⋀x y. snd R x y ⊢ snd R' x y"
assumes "⋀x y a. P' a ⟹ S a x y ⊢ S' a x y"
assumes "⋀x y. CP x y ⟹ CP' x y"
shows "(f,g) ∈ [P']⇩a [C']⇩c R' →⇩d S' [CP']⇩c"
unfolding hfref_def
apply clarsimp
apply (rule hn_refine_cons_cp)
apply (rule assms(4))
defer
apply (rule entails_trans[OF assms(5) entails_refl])
apply (erule assms(6))
apply (erule assms(7))
apply (frule assms(2))
apply (frule assms(3))
using assms(1)
unfolding hfref_def
apply auto
done
subsubsection ‹Composition Automation›
text ‹This section contains the lemmas. The ML code is further down. ›
lemma prod_hrp_comp:
"hrp_comp (A *⇩a B) (C ×⇩r D) = hrp_comp A C *⇩a hrp_comp B D"
unfolding hrp_comp_def hfprod_def by simp
lemma hrp_comp_keep: "hrp_comp (A⇧k) B = (hr_comp A B)⇧k"
by (auto simp: hrp_comp_def)
lemma hr_comp_invalid: "hr_comp (invalid_assn R1) R2 = invalid_assn (hr_comp R1 R2)"
apply (intro ext)
unfolding invalid_assn_def hr_comp_def
apply (auto simp: sep_algebra_simps)
apply (auto simp: pure_part_def)
done
lemma hrp_comp_dest: "hrp_comp (A⇧d) B = (hr_comp A B)⇧d"
by (auto simp: hrp_comp_def hr_comp_invalid)
definition "hrp_imp RR RR' ≡
∀a b. (fst RR' a b ⊢ fst RR a b) ∧ (snd RR a b ⊢ snd RR' a b)"
lemma hfref_imp: "hrp_imp RR RR' ⟹ [P]⇩a [C]⇩c RR →⇩d S [CP]⇩c ⊆ [P]⇩a [C]⇩c RR' →⇩d S [CP]⇩c"
apply clarsimp
apply (erule hfref_cons)
apply (simp_all add: hrp_imp_def)
done
lemma hrp_imp_refl: "hrp_imp RR RR"
unfolding hrp_imp_def by auto
lemma hrp_imp_reflI: "RR = RR' ⟹ hrp_imp RR RR'"
unfolding hrp_imp_def by auto
lemma hrp_comp_cong: "hrp_imp A A' ⟹ B=B' ⟹ hrp_imp (hrp_comp A B) (hrp_comp A' B')"
by (auto simp: hrp_imp_def hrp_comp_def hr_comp_def entails_def sep_algebra_simps)
lemma hrp_prod_cong: "hrp_imp A A' ⟹ hrp_imp B B' ⟹ hrp_imp (A*⇩aB) (A'*⇩aB')"
by (auto simp: hrp_imp_def prod_assn_def sep_algebra_simps
intro: conj_entails_mono
)
lemma hrp_imp_trans: "hrp_imp A B ⟹ hrp_imp B C ⟹ hrp_imp A C"
unfolding hrp_imp_def
by (fastforce intro: entails_trans)
lemma fcomp_norm_dflt_init: "x∈[P]⇩a [C]⇩c R →⇩d T [CP]⇩c ⟹ hrp_imp R S ⟹ x∈[P]⇩a [C]⇩c S →⇩d T [CP]⇩c"
apply (erule set_rev_mp)
by (rule hfref_imp)
definition "comp_PRE R P Q S ≡ λx. S x ⟶ (P x ∧ (∀y. (y,x)∈R ⟶ Q x y))"
lemma comp_PRE_cong[cong]:
assumes "R≡R'"
assumes "⋀x. P x ≡ P' x"
assumes "⋀x. S x ≡ S' x"
assumes "⋀x y. ⟦P x; (y,x)∈R; y∈Domain R; S' x ⟧ ⟹ Q x y ≡ Q' x y"
shows "comp_PRE R P Q S ≡ comp_PRE R' P' Q' S'"
using assms
by (fastforce simp: comp_PRE_def fun_eq_iff intro!: eq_reflection)
lemma fref_compI_PRE:
"⟦ (f,g)∈fref P R1 R2; (g,h)∈fref Q S1 S2 ⟧
⟹ (f,h) ∈ fref (comp_PRE S1 Q (λ_. P) (λ_. True)) (R1 O S1) (rr_comp S1 R2 S2)"
using fref_compI[of P R1 R2 Q S1 S2]
unfolding comp_PRE_def
by auto
lemma PRE_D1: "(Q x ∧ P x) ⟶ comp_PRE S1 Q (λx _. P x) S x"
by (auto simp: comp_PRE_def)
lemma PRE_D2: "(Q x ∧ (∀y. (y,x)∈S1 ⟶ S x ⟶ P x y)) ⟶ comp_PRE S1 Q P S x"
by (auto simp: comp_PRE_def)
lemma fref_weaken_pre:
assumes "⋀x. P x ⟶ P' x"
assumes "(f,h) ∈ fref P' R S"
shows "(f,h) ∈ fref P R S"
apply (rule set_rev_mp[OF assms(2) fref_mono])
using assms(1) by auto
lemma fref_PRE_D1:
assumes "(f,h) ∈ fref (comp_PRE S1 Q (λx _. P x) X) R S"
shows "(f,h) ∈ fref (λx. Q x ∧ P x) R S"
by (rule fref_weaken_pre[OF PRE_D1 assms])
lemma fref_PRE_D2:
assumes "(f,h) ∈ fref (comp_PRE S1 Q P X) R S"
shows "(f,h) ∈ fref (λx. Q x ∧ (∀y. (y,x)∈S1 ⟶ X x ⟶ P x y)) R S"
by (rule fref_weaken_pre[OF PRE_D2 assms])
lemmas fref_PRE_D = fref_PRE_D1 fref_PRE_D2
lemma hfref_weaken_pre:
assumes "⋀x. P x ⟶ P' x"
assumes "(f,h) ∈ hfref P' C R S CP"
shows "(f,h) ∈ hfref P C R S CP"
using assms
by (auto simp: hfref_def)
lemma hfref_weaken_pre':
assumes "⋀x. ⟦P x; rdomp (fst R) x⟧ ⟹ P' x"
assumes "(f,h) ∈ hfref P' C R S CP"
shows "(f,h) ∈ hfref P C R S CP"
apply (rule hfrefI)
apply (rule hn_refine_preI)
using assms
by (auto simp: hfref_def rdomp_def)
lemma hfref_weaken_pre_nofail':
assumes "(f,g) ∈ [P]⇩a [C]⇩c R →⇩d S [CP]⇩c"
assumes "⋀x. ⟦nofail (g x); Q x⟧ ⟹ P x"
shows "(f,g) ∈ [Q]⇩a [C]⇩c R →⇩d S [CP]⇩c"
apply (rule hfref_weaken_pre[OF _ assms(1)[THEN hfref_weaken_pre_nofail]])
using assms(2)
by blast
lemma hfref_with_rdomI:
assumes "(c,a)∈[λx. P x ∧ rdomp (fst A) x]⇩a [C]⇩c A →⇩d R [CP]⇩c"
shows "(c,a)∈[P]⇩a [C]⇩c A →⇩d R [CP]⇩c"
by (metis (no_types, lifting) assms hfref_weaken_pre')
lemma hfref_compI_PRE_aux:
assumes A: "(f,g) ∈ [P]⇩a [C]⇩c RR' →⇩d S [CP]⇩c"
assumes B: "(g,h) ∈ [Q]⇩f⇩d T → (λx. ⟨U x⟩nres_rel)"
shows "(f,h) ∈ [comp_PRE T Q (λ_. P) (λ_. True)]⇩a [C]⇩c
hrp_comp RR' T →⇩d hrr_comp T S U [CP]⇩c"
apply (rule hfref_weaken_pre[OF _ hfcomp[OF A B]])
by (auto simp: comp_PRE_def)
lemma hfref_compI_PRE:
assumes A: "(f,g) ∈ [P]⇩a [C]⇩c RR' →⇩d S [CP]⇩c"
assumes B: "(g,h) ∈ [Q]⇩f⇩d T → (λx. ⟨U x⟩nres_rel)"
shows "(f,h) ∈ [comp_PRE T Q (λx y. P y) (λx. nofail (h x))]⇩a [C]⇩c
hrp_comp RR' T →⇩d hrr_comp T S U [CP]⇩c"
using hfref_compI_PRE_aux[OF A B, THEN hfref_weaken_pre_nofail]
apply (rule hfref_weaken_pre[rotated])
apply (auto simp: comp_PRE_def)
done
lemma hfref_PRE_D1:
assumes "(f,h) ∈ hfref (comp_PRE S1 Q (λx _. P x) X) C R S CP"
shows "(f,h) ∈ hfref (λx. Q x ∧ P x) C R S CP"
by (rule hfref_weaken_pre[OF PRE_D1 assms])
lemma hfref_PRE_D2:
assumes "(f,h) ∈ hfref (comp_PRE S1 Q P X) C R S CP"
shows "(f,h) ∈ hfref (λx. Q x ∧ (∀y. (y,x)∈S1 ⟶ X x ⟶ P x y)) C R S CP"
by (rule hfref_weaken_pre[OF PRE_D2 assms])
lemma hfref_PRE_D3:
assumes "(f,h) ∈ hfref (comp_PRE S1 Q P X) C R S CP"
shows "(f,h) ∈ hfref (comp_PRE S1 Q P X) C R S CP"
using assms .
lemmas hfref_PRE_D = hfref_PRE_D1 hfref_PRE_D3
subsection ‹Automation›
text ‹Purity configuration for constraint solver›
lemmas [safe_constraint_rules] = pure_pure
lemma invalid_pure[safe_constraint_rules]: "is_pure (invalid_assn A)"
unfolding invalid_assn_def is_pure_def by auto
text ‹Configuration for hfref to hnr conversion›
named_theorems to_hnr_post ‹to_hnr converter: Postprocessing unfold rules›
lemma uncurry0_add_app_tag: "uncurry0 (RETURN c) = uncurry0 (RETURN$c)" by simp
thm sep_conj_empty[of "P::ll_assn" for P]
lemmas [to_hnr_post] = norm_RETURN_o norm_return_o
uncurry0_add_app_tag uncurry0_apply uncurry0_APP hn_val_unit_conv_emp
sep_conj_empty[of "P::ll_assn" for P] sep_conj_empty'[of "P::ll_assn" for P]
named_theorems to_hfref_post ‹to_hfref converter: Postprocessing unfold rules›
lemma prod_casesK[to_hfref_post]: "case_prod (λ_ _. k) = (λ_. k)" by auto
lemma uncurry0_hfref_post[to_hfref_post]: "hfref (uncurry0 True) R S = hfref (λ_. True) R S"
apply (fo_rule arg_cong fun_cong)+ by auto
text ‹Configuration for relation normalization after composition›
named_theorems fcomp_norm_unfold ‹fcomp-normalizer: Unfold theorems›
named_theorems fcomp_norm_simps ‹fcomp-normalizer: Simplification theorems›
named_theorems fcomp_norm_init "fcomp-normalizer: Initialization rules"
named_theorems fcomp_norm_trans "fcomp-normalizer: Transitivity rules"
named_theorems fcomp_norm_cong "fcomp-normalizer: Congruence rules"
named_theorems fcomp_norm_norm "fcomp-normalizer: Normalization rules"
named_theorems fcomp_norm_refl "fcomp-normalizer: Reflexivity rules"
text ‹Default Setup›
lemmas [fcomp_norm_unfold] = prod_rel_comp nres_rel_comp Id_O_R R_O_Id
lemmas [fcomp_norm_unfold] = hr_comp_Id1 hr_comp_Id2 hrp_comp_Id1 hrp_comp_Id2
find_theorems rr_comp
lemmas [fcomp_norm_unfold] = hrr_comp_nondep rr_comp_nondep
lemmas [fcomp_norm_unfold] = hr_comp_prod_conv
lemmas [fcomp_norm_unfold] = prod_hrp_comp hrp_comp_keep hrp_comp_dest hr_comp_pure
lemma [fcomp_norm_simps]: "CONSTRAINT is_pure P ⟹ pure (the_pure P) = P" by simp
lemmas [fcomp_norm_simps] = True_implies_equals
lemmas [fcomp_norm_init] = fcomp_norm_dflt_init
lemmas [fcomp_norm_trans] = hrp_imp_trans
lemmas [fcomp_norm_cong] = hrp_comp_cong hrp_prod_cong
lemmas [fcomp_norm_refl] = refl hrp_imp_refl
lemma ensure_fref_nresI: "(f,g)∈[P]⇩f⇩d R→S ⟹ (RETURN o f, RETURN o g)∈[P]⇩f⇩d R→(λx. ⟨S x⟩nres_rel)"
by (auto intro: nres_relI simp: fref_def)
lemma ensure_fref_nres_unfold:
"⋀f. RETURN o (uncurry0 f) = uncurry0 (RETURN f)"
"⋀f. RETURN o (uncurry f) = uncurry (RETURN oo f)"
"⋀f. (RETURN ooo uncurry) f = uncurry (RETURN ooo f)"
by auto
text ‹Composed precondition normalizer›
named_theorems fcomp_prenorm_simps ‹fcomp precondition-normalizer: Simplification theorems›
text ‹Support for preconditions of the form @{text "_∈Domain R"},
where @{text R} is the relation of the next more abstract level.›
declare DomainI[fcomp_prenorm_simps]
lemma auto_weaken_pre_init_hf:
assumes "⋀x. PROTECT P x ⟶ P' x"
assumes "(f,h) ∈ hfref P' C R S CP"
shows "(f,h) ∈ hfref P C R S CP"
using assms
by (auto simp: hfref_def)
lemma auto_weaken_pre_init_f:
assumes "⋀x. PROTECT P x ⟶ P' x"
assumes "(f,h) ∈ fref P' R S"
shows "(f,h) ∈ fref P R S"
using assms
by (auto simp: fref_def)
lemmas auto_weaken_pre_init = auto_weaken_pre_init_hf auto_weaken_pre_init_f
lemma auto_weaken_pre_uncurry_step:
assumes "PROTECT f a ≡ f'"
shows "PROTECT (λ(x,y). f x y) (a,b) ≡ f' b"
using assms
by (auto simp: curry_def dest!: meta_eq_to_obj_eq intro!: eq_reflection)
lemma auto_weaken_pre_uncurry_finish:
"PROTECT f x ≡ f x" by (auto)
lemma auto_weaken_pre_uncurry_start:
assumes "P ≡ P'"
assumes "P'⟶Q"
shows "P⟶Q"
using assms by (auto)
lemma auto_weaken_pre_comp_PRE_I:
assumes "S x ⟹ P x"
assumes "⋀y. ⟦(y,x)∈R; P x; S x⟧ ⟹ Q x y"
shows "comp_PRE R P Q S x"
using assms by (auto simp: comp_PRE_def)
lemma auto_weaken_pre_to_imp_nf:
"(A⟶B⟶C) = (A∧B ⟶ C)"
"((A∧B)∧C) = (A∧B∧C)"
by auto
lemma auto_weaken_pre_add_dummy_imp:
"P ⟹ True ⟶ P" by simp
text ‹Synthesis for hfref statements›
definition hfsynth_ID_R :: "('a ⇒ _ ⇒ assn) ⇒ 'a ⇒ bool" where
[simp]: "hfsynth_ID_R _ _ ≡ True"
lemma hfsynth_ID_R_D:
fixes I :: "'a itself"
assumes "hfsynth_ID_R R a"
assumes "intf_of_assn R I"
shows "a ::⇩i I"
by simp
lemma hfsynth_hnr_from_hfI:
assumes "∀x xi. P x ∧ C xi ∧ hfsynth_ID_R (fst R) x ⟶ hn_refine (hn_ctxt (fst R) x xi ** □) (f$xi) (hn_ctxt (snd R) x xi ** □) (S x) (CP xi) (g$x)"
shows "(f,g) ∈ [P]⇩a [C]⇩c R →⇩d S [CP]⇩c"
using assms
unfolding hfref_def
by (auto simp: hn_ctxt_def)
lemma hfsynth_ID_R_uncurry_unfold:
"hfsynth_ID_R (to_hnr_prod R S) (a,b) ≡ hfsynth_ID_R R a ∧ hfsynth_ID_R S b"
"hfsynth_ID_R (fst (hf_pres R k)) ≡ hfsynth_ID_R R"
by (auto intro!: eq_reflection)
ML ‹
signature SEPREF_RULES = sig
val binder_rels: term -> term list
val body_rel: term -> term
val analyze_rel: term -> term option * term list * term
val mk_triv_precond: term list -> term
val mk_rel: term option * term list * term -> term
val strip_rel: term -> term list * term
val mk_hfprod : term * term -> term
val mk_hfprods : term list -> term
val intf_of_assn : Proof.context -> term -> typ
val to_fref : Proof.context -> thm -> thm
val to_foparam : Proof.context -> thm -> thm
val prepare_hfref_synth_tac : Proof.context -> tactic'
val to_hnr : Proof.context -> thm -> thm
val to_hfref: Proof.context -> thm -> thm
val ensure_fref : Proof.context -> thm -> thm
val ensure_fref_nres : Proof.context -> thm -> thm
val ensure_hfref : Proof.context -> thm -> thm
val ensure_hnr : Proof.context -> thm -> thm
type hnr_analysis = {
thm: thm,
precond: term,
ccond: term,
prems : term list,
ahead: term * bool,
chead: term * bool,
argrels: (term * bool) list,
result_rel: term,
cpcond: term
}
val analyze_hnr: Proof.context -> thm -> hnr_analysis
val pretty_hnr_analysis: Proof.context -> hnr_analysis -> Pretty.T
val mk_hfref_thm: Proof.context -> hnr_analysis -> thm
val simplify_precond: Proof.context -> thm -> thm
val norm_fcomp_rule: Proof.context -> thm -> thm
val add_pure_constraints_rule: Proof.context -> thm -> thm
val gen_compose : Proof.context -> thm -> thm -> thm
val fcomp_attrib: attribute context_parser
end
structure Sepref_Rules: SEPREF_RULES = struct
local open Refine_Util Relators in
fun binder_rels @{mpat "?F → ?G"} = F::binder_rels G
| binder_rels @{mpat "fref _ ?F _"} = strip_prodrel_left F
| binder_rels _ = []
local
fun br_aux @{mpat "_ → ?G"} = br_aux G
| br_aux R = R
in
fun body_rel @{mpat "fref _ _ ?G"} = G
| body_rel R = br_aux R
end
fun strip_rel R = (binder_rels R, body_rel R)
fun analyze_rel @{mpat "fref (λ_. True) ?R ?S"} = (NONE,strip_prodrel_left R,S)
| analyze_rel @{mpat "fref ?P ?R ?S"} = (SOME P,strip_prodrel_left R,S)
| analyze_rel R = let
val (args,res) = strip_rel R
val argTuc =
map (fastype_of #> HOLogic.dest_setT #> HOLogic.dest_prodT #> snd) args
|> list_prodT_left
val res = Term.absdummy argTuc res
in
(NONE,args,res)
end
fun mk_triv_precond Rs = absdummy (map rel_absT Rs |> list_prodT_left) @{term True}
fun mk_rel (P,Rs,S) = let
val R = list_prodrel_left Rs
val P = case P of
SOME P => P
| NONE => mk_triv_precond Rs
in
@{mk_term "fref ?P ?R ?S"}
end
end
fun mk_hfprod (a, b) = @{mk_term "?a*⇩a?b"}
local
fun mk_hfprods_rev [] = @{mk_term "unit_assn⇧k"}
| mk_hfprods_rev [Rk] = Rk
| mk_hfprods_rev (Rkn::Rks) = mk_hfprod (mk_hfprods_rev Rks, Rkn)
in
val mk_hfprods = mk_hfprods_rev o rev
end
fun intf_of_assn ctxt t = let
val orig_ctxt = ctxt
val (t,ctxt) = yield_singleton (Variable.import_terms false) t ctxt
val v = TVar (("T",0),Proof_Context.default_sort ctxt ("T",0)) |> Logic.mk_type
val goal = @{mk_term "Trueprop (intf_of_assn ?t ?v)"}
val i_of_assn_rls =
Named_Theorems_Rev.get ctxt @{named_theorems_rev intf_of_assn}
@ @{thms intf_of_assn_fallback}
fun tac ctxt = REPEAT_ALL_NEW (resolve_tac ctxt i_of_assn_rls)
val thm = Goal.prove ctxt [] [] goal (fn {context,...} => ALLGOALS (tac context))
val intf = case Thm.concl_of thm of
@{mpat "Trueprop (intf_of_assn _ (?v AS⇩p TYPE (_)))"} => v
| _ => raise THM("Intf_of_assn: Proved a different theorem?",~1,[thm])
val intf = singleton (Variable.export_terms ctxt orig_ctxt) intf
|> Logic.dest_type
in
intf
end
datatype rthm_type =
RT_HOPARAM
| RT_FREF
| RT_HNR
| RT_HFREF
| RT_OTHER
fun rthm_type thm =
case Thm.concl_of thm |> HOLogic.dest_Trueprop of
@{mpat "(_,_) ∈ fref _ _ _"} => RT_FREF
| @{mpat "(_,_) ∈ hfref _ _ _ _ _"} => RT_HFREF
| @{mpat "hn_refine _ _ _ _ _ _"} => RT_HNR
| @{mpat "(_,_) ∈ _"} => RT_HOPARAM
| _ => RT_OTHER
fun to_fref ctxt thm = let
open Conv
in
case Thm.concl_of thm |> HOLogic.dest_Trueprop of
@{mpat "(_,_)∈_→_"} =>
Local_Defs.unfold0 ctxt @{thms fref_param1} thm
|> fconv_rule (repeat_conv (Refine_Util.ftop_conv (K (rewr_conv @{thm fref_nest})) ctxt))
|> Local_Defs.unfold0 ctxt @{thms in_CURRY_conv}
| @{mpat "(_,_)∈_"} => thm RS @{thm fref_param0I}
| _ => raise THM ("to_fref: Expected theorem of form (_,_)∈_",~1,[thm])
end
fun to_foparam ctxt thm = let
val unf_thms = @{thms
split_tupled_all prod_rel_simp uncurry_apply cnv_conj_to_meta Product_Type.split}
in
case Thm.concl_of thm of
@{mpat "Trueprop ((_,_) ∈ fref _ _ _)"} =>
(@{thm frefD} OF [thm])
|> Thm.forall_intr_vars
|> Local_Defs.unfold0 ctxt unf_thms
|> Variable.gen_all ctxt
| @{mpat "Trueprop ((_,_) ∈ _)"} =>
Parametricity.fo_rule thm
| _ => raise THM("Expected parametricity or fref theorem",~1,[thm])
end
fun to_hnr ctxt thm =
(thm RS @{thm hf2hnr})
|> Local_Defs.unfold0 ctxt @{thms to_hnr_prod_fst_snd keep_drop_sels}
|> Local_Defs.unfold0 ctxt @{thms hnr_uncurry_unfold}
|> Local_Defs.unfold0 ctxt @{thms uncurry_apply uncurry_APP sep_conj_empty' split}
|> Local_Defs.unfold0 ctxt @{thms hn_ctxt_ctxt_fix_conv}
|> Local_Defs.unfold0 ctxt @{thms all_to_meta imp_to_meta HOL.True_implies_equals HOL.implies_True_equals Pure.triv_forall_equality cnv_conj_to_meta}
|> Local_Defs.unfold0 ctxt (Named_Theorems.get ctxt @{named_theorems to_hnr_post})
|> Goal.norm_result ctxt
|> Conv.fconv_rule Thm.eta_conversion
fun prepare_hfref_synth_tac ctxt = let
val i_of_assn_rls =
Named_Theorems_Rev.get ctxt @{named_theorems_rev intf_of_assn}
@ @{thms intf_of_assn_fallback}
val to_hnr_post_rls =
Named_Theorems.get ctxt @{named_theorems to_hnr_post}
val i_of_assn_tac = (
REPEAT' (
DETERM o dresolve_tac ctxt @{thms hfsynth_ID_R_D}
THEN' DETERM o SOLVED' (REPEAT_ALL_NEW (resolve_tac ctxt i_of_assn_rls))
)
)
in
resolve_tac ctxt @{thms hfsynth_hnr_from_hfI} THEN_ELSE' (
SELECT_GOAL (
unfold_tac ctxt @{thms to_hnr_prod_fst_snd keep_drop_sels hf_pres_fst}
THEN unfold_tac ctxt @{thms hnr_uncurry_unfold hfsynth_ID_R_uncurry_unfold}
THEN unfold_tac ctxt @{thms uncurry_apply uncurry_APP sep_conj_empty' split}
THEN unfold_tac ctxt @{thms all_to_meta imp_to_meta HOL.True_implies_equals HOL.implies_True_equals Pure.triv_forall_equality cnv_conj_to_meta}
THEN ALLGOALS i_of_assn_tac
THEN unfold_tac ctxt to_hnr_post_rls
THEN unfold_tac ctxt @{thms APP_def}
)
,
K all_tac
)
end
structure Termtab2 = Table(
type key = term * term
val ord = prod_ord Term_Ord.fast_term_ord Term_Ord.fast_term_ord);
type hnr_analysis = {
thm: thm,
precond: term,
ccond: term,
prems : term list,
ahead: term * bool,
chead: term * bool,
argrels: (term * bool) list,
result_rel: term,
cpcond: term
}
fun analyze_hnr (ctxt:Proof.context) thm = let
val dbg = Unsynchronized.ref []
fun add_dbg msg ts = (
dbg := (msg,ts) :: !dbg;
()
)
fun pretty_dbg (msg,ts) = Pretty.block [
Pretty.str msg,
Pretty.str ":",
Pretty.brk 1,
Pretty.list "[" "]" (map (Syntax.pretty_term ctxt) ts)
]
fun pretty_dbgs l = map pretty_dbg l |> Pretty.fbreaks |> Pretty.block
fun trace_dbg msg = Pretty.block [Pretty.str msg, Pretty.fbrk, pretty_dbgs (rev (!dbg))] |> Pretty.string_of |> tracing
fun fail msg = (trace_dbg msg; raise THM(msg,~1,[thm]))
fun assert cond msg = cond orelse fail msg;
fun check_strip_leading args t f =
if Termtab.defined args f then (t,false) else (f,true)
fun strip_leading_RETURN args (t as @{mpat "RETURN$(?f)"}) = check_strip_leading args t f
| strip_leading_RETURN args (t as @{mpat "RETURN ?f"}) = check_strip_leading args t f
| strip_leading_RETURN _ t = (t,false)
fun strip_leading_return args (t as @{mpat "Mreturn$(?f)"}) = check_strip_leading args t f
| strip_leading_return args (t as @{mpat "Mreturn ?f"}) = check_strip_leading args t f
| strip_leading_return _ t = (t,false)
fun strip_fun _ (t as @{mpat "PR_CONST _"}) = (t,[])
| strip_fun s (t as @{mpat "?f$?x"}) = check_arg s t f x
| strip_fun s (t as @{mpat "?f ?x"}) = check_arg s t f x
| strip_fun _ f = (f,[])
and check_arg s t f x =
if Termtab.defined s x then
strip_fun s f |> apsnd (curry op :: x)
else (t,[])
fun dest_hn_ctxt @{mpat "hn_ctxt ?R ?a ?c"} = ((a,c),R)
| dest_hn_ctxt _ = fail "Invalid hn_ctxt parameter in pre or postcondition"
fun is_emp @{mpat □} = true | is_emp _ = false
val strip_star' = Sepref_Basic.strip_star #> filter (not o is_emp)
fun pairs_eq pairs1 pairs2 =
Termtab2.forall (Termtab2.defined pairs1 o fst) pairs2
andalso Termtab2.forall (Termtab2.defined pairs2 o fst) pairs1
fun atomize_prem @{mpat "Trueprop ?p"} = p
| atomize_prem _ = fail "Non-atomic premises"
fun mk_conjs [] = @{const True}
| mk_conjs [p] = p
| mk_conjs (p::ps) = HOLogic.mk_binop @{const_name "HOL.conj"} (p,mk_conjs ps)
val _ = add_dbg "thm" [Thm.prop_of thm]
val prems = Thm.prems_of thm
val concl = Thm.concl_of thm |> HOLogic.dest_Trueprop
val (G,c,G',R,CP,a) = Sepref_Basic.dest_hn_refine concl
val pre_pairs = G
|> strip_star'
|> tap (add_dbg "precondition")
|> map dest_hn_ctxt
|> Termtab2.make
val post_pairs = G'
|> strip_star'
|> tap (add_dbg "postcondition")
|> map dest_hn_ctxt
|> Termtab2.make
val _ = assert (pairs_eq pre_pairs post_pairs)
"Parameters in precondition do not match postcondition"
val aa_set = pre_pairs |> Termtab2.keys |> map fst |> Termtab.make_set
val ca_set = pre_pairs |> Termtab2.keys |> map snd |> Termtab.make_set
val (a,leading_RETURN) = strip_leading_RETURN aa_set a
val (c,leading_return) = strip_leading_return ca_set c
val _ = add_dbg "stripped abstract term" [a]
val _ = add_dbg "stripped concrete term" [c]
val (ahead,aargs) = strip_fun aa_set a;
val (chead,cargs) = strip_fun ca_set c;
val _ = add_dbg "abstract head" [ahead]
val _ = add_dbg "abstract args" aargs
val _ = add_dbg "concrete head" [chead]
val _ = add_dbg "concrete args" cargs
val _ = assert (length cargs = length aargs) "Different number of abstract and concrete arguments";
val _ = assert (not (has_duplicates op aconv aargs)) "Duplicate abstract arguments"
val _ = assert (not (has_duplicates op aconv cargs)) "Duplicate concrete arguments"
val argpairs = aargs ~~ cargs
val ap_set = Termtab2.make_set argpairs
val _ = assert (pairs_eq pre_pairs ap_set) "Arguments from pre/postcondition do not match operation's arguments"
val pre_rels = map (the o (Termtab2.lookup pre_pairs)) argpairs
val post_rels = map (the o (Termtab2.lookup post_pairs)) argpairs
val _ = add_dbg "pre-rels" pre_rels
val _ = add_dbg "post-rels" post_rels
fun adjust_hf_pres @{mpat "snd (?R⇧k)"} = R
| adjust_hf_pres t = t
val post_rels = map adjust_hf_pres post_rels
fun is_invalid R @{mpat "invalid_assn ?R'"} = R aconv R'
| is_invalid _ @{mpat "snd (_⇧d)"} = true
| is_invalid _ _ = false
fun is_keep (R,R') =
if R aconv R' then true
else if is_invalid R R' then false
else fail "Mismatch between pre and post relation for argument"
val keep = map is_keep (pre_rels ~~ post_rels)
val argrels = pre_rels ~~ keep
val aa_set = Termtab.make_set aargs
val ca_set = Termtab.make_set cargs
datatype PCLS = pc_PREM | pc_ABS | pc_CONC
fun classify_precond t =
case (exists_subterm (Termtab.defined ca_set) t, exists_subterm (Termtab.defined aa_set) t) of
(false,false) => (pc_PREM,t)
| (true,false) => (pc_CONC,t)
| (false,true) => (pc_ABS,t)
| _ => fail "Premise contains abstract and concrete argument"
val prems = map classify_precond prems
val preconds = filter (fn (x,_) => x=pc_ABS) prems |> map snd
val cconds = filter (fn (x,_) => x=pc_CONC) prems |> map snd
val prems = filter (fn (x,_) => x=pc_PREM) prems |> map snd
val precond =
map atomize_prem preconds
|> mk_conjs
|> fold lambda aargs
val ccond =
map atomize_prem cconds
|> mk_conjs
|> fold lambda cargs
val _ = add_dbg "precond" [precond]
val _ = add_dbg "ccond" [ccond]
val _ = add_dbg "prems" prems
val cpcond = fold lambda cargs CP
val R = fold lambda aargs R
val _ = add_dbg "cpcond" [cpcond]
in
{
thm = thm,
precond = precond,
ccond = ccond,
prems = prems,
ahead = (ahead,leading_RETURN),
chead = (chead,leading_return),
argrels = argrels,
result_rel = R,
cpcond = cpcond
}
end
fun pretty_hnr_analysis
ctxt
({thm,precond,ccond,ahead,chead,argrels,result_rel,cpcond,...})
: Pretty.T =
let
val _ = thm
fun pretty_argrel (R,k) = Pretty.block [
Syntax.pretty_term ctxt R,
if k then Pretty.str "⇧k" else Pretty.str "⇧d"
]
val pretty_chead = case chead of
(t,false) => Syntax.pretty_term ctxt t
| (t,true) => Pretty.block [Pretty.str "return ", Syntax.pretty_term ctxt t]
val pretty_ahead = case ahead of
(t,false) => Syntax.pretty_term ctxt t
| (t,true) => Pretty.block [Pretty.str "RETURN ", Syntax.pretty_term ctxt t]
in
Pretty.fbreaks [
Pretty.block [
Pretty.enclose "[" "]" [pretty_chead, pretty_ahead],
Pretty.enclose "[" "]" [Syntax.pretty_term ctxt precond],
Pretty.enclose "[" "]" [Syntax.pretty_term ctxt ccond],
Pretty.brk 1,
Pretty.block (Pretty.separate " →" (map pretty_argrel argrels @ [Syntax.pretty_term ctxt result_rel])),
Pretty.brk 1,
Pretty.enclose "[" "]" [Syntax.pretty_term ctxt cpcond]
]
] |> Pretty.block
end
fun mk_hfref_thm
ctxt
({thm,precond,ccond,prems,ahead,chead,argrels,result_rel,cpcond}) =
let
fun mk_keep (R,true) = @{mk_term "?R⇧k"}
| mk_keep (R,false) = @{mk_term "?R⇧d"}
fun mk_uncurry f = @{mk_term "uncurry ?f"}
fun rpt_uncurry n t =
if n=0 then @{mk_term "uncurry0 ?t"}
else if n=1 then t
else funpow (n-1) mk_uncurry t
fun rew_uncurry_lambda t = let
val rr = map (Logic.dest_equals o Thm.prop_of) @{thms uncurry_def uncurry0_def}
val thy = Proof_Context.theory_of ctxt
in
Pattern.rewrite_term_top thy rr [] t
end
fun gsimp_only ctxt sec = let
val ss = put_simpset HOL_basic_ss ctxt |> sec
in asm_full_simp_tac ss end
fun simp_only ctxt thms = gsimp_only ctxt (fn ctxt => ctxt addsimps thms)
val num_args = length argrels
val precond = precond
|> rpt_uncurry num_args
|> rew_uncurry_lambda
val ccond = ccond
|> rpt_uncurry num_args
|> rew_uncurry_lambda
val cpcond = cpcond
|> rpt_uncurry num_args
|> rew_uncurry_lambda
val result_rel = result_rel
|> rpt_uncurry num_args
|> rew_uncurry_lambda
fun mk_RETURN (t,r) = if r then
let
val T = funpow num_args range_type (fastype_of (fst ahead))
val tRETURN = Const (@{const_name RETURN}, T --> Type(@{type_name nres},[T]))
in
Refine_Util.mk_compN num_args tRETURN t
end
else t
fun mk_return (t,r) = if r then @{mk_term "Mreturn ?t :: _ llM"}
else t
fun certify_inst ctxt (instT, inst) =
(TVars.map (K (Thm.ctyp_of ctxt)) instT,
Vars.map (K (Thm.cterm_of ctxt)) inst);
val ahead = ahead |> mk_RETURN |> rpt_uncurry num_args
val chead = chead |> mk_return |> rpt_uncurry num_args
val argrel = map mk_keep argrels |> rev |> mk_hfprods
val result = @{mk_term "Trueprop ((?chead,?ahead) ∈ [?precond]⇩a [?ccond]⇩c ?argrel →⇩d ?result_rel [?cpcond]⇩c)"}
val result = Logic.list_implies (prems,result)
val orig_ctxt = ctxt
val (insts, ctxt) = Variable.import_inst true [result] ctxt
val insts' = certify_inst ctxt insts
val result = Term_Subst.instantiate insts result
val thm = Thm.instantiate insts' thm
val thm = Local_Defs.unfold0 ctxt @{thms APP_def} thm
fun tac ctxt =
resolve_tac ctxt @{thms hfrefI}
THEN' gsimp_only ctxt (fn c => c
addsimps @{thms uncurry_def hn_ctxt_def uncurry0_def
keep_drop_sels uc_hfprod_sel o_apply
APP_def}
|> Splitter.add_split @{thm prod.split}
)
THEN' TRY o (
REPEAT_ALL_NEW (match_tac ctxt @{thms allI impI})
THEN' simp_only ctxt @{thms Product_Type.split prod.inject})
THEN' TRY o REPEAT_ALL_NEW (ematch_tac ctxt @{thms conjE})
THEN' TRY o hyp_subst_tac ctxt
THEN' simp_only ctxt @{thms triv_forall_equality}
THEN' (
resolve_tac ctxt @{thms hn_refine_cons[rotated]}
THEN' (resolve_tac ctxt [thm] THEN_ALL_NEW assume_tac ctxt))
THEN_ALL_NEW simp_only ctxt
@{thms hn_ctxt_def entails_refl pure_unit_rel_eq_empty
mult_ac mult_1 mult_1_right keep_drop_sels}
val result = Thm.cterm_of ctxt result
val rthm = Goal.prove_internal ctxt [] result (fn _ => ALLGOALS (tac ctxt))
val rthm = singleton (Variable.export ctxt orig_ctxt) rthm
val rthm = Local_Defs.unfold0 ctxt (Named_Theorems.get ctxt @{named_theorems to_hfref_post}) rthm
in
rthm
end
fun to_hfref ctxt = analyze_hnr ctxt #> mk_hfref_thm ctxt
local
fun norm_set_of ctxt = {
trans_rules = Named_Theorems.get ctxt @{named_theorems fcomp_norm_trans},
cong_rules = Named_Theorems.get ctxt @{named_theorems fcomp_norm_cong},
norm_rules = Named_Theorems.get ctxt @{named_theorems fcomp_norm_norm},
refl_rules = Named_Theorems.get ctxt @{named_theorems fcomp_norm_refl}
}
fun init_rules_of ctxt = Named_Theorems.get ctxt @{named_theorems fcomp_norm_init}
fun unfold_rules_of ctxt = Named_Theorems.get ctxt @{named_theorems fcomp_norm_unfold}
fun simp_rules_of ctxt = Named_Theorems.get ctxt @{named_theorems fcomp_norm_simps}
in
fun norm_fcomp_rule ctxt = let
open PO_Normalizer Refine_Util
val norm1 = gen_norm_rule (init_rules_of ctxt) (norm_set_of ctxt) ctxt
val norm2 = Local_Defs.unfold0 ctxt (unfold_rules_of ctxt)
val norm3 = Conv.fconv_rule (
Simplifier.asm_full_rewrite
(put_simpset HOL_basic_ss ctxt addsimps simp_rules_of ctxt))
val norm = changed_rule (try_rule norm1 o try_rule norm2 o try_rule norm3)
in
repeat_rule norm
end
end
fun add_pure_constraints_rule ctxt thm = let
val orig_ctxt = ctxt
val t = Thm.prop_of thm
fun
cnv (@{mpat (typs) "pure (mpaq_STRUCT (mpaq_Var ?x _) :: (?'v_c×?'v_a) set)"}) =
let
val T = a --> c --> @{typ assn}
val t = Var (x,T)
val t = @{mk_term "(the_pure ?t)"}
in
[(x,T,t)]
end
| cnv (t$u) = union op= (cnv t) (cnv u)
| cnv (Abs (_,_,t)) = cnv t
| cnv _ = []
val pvars = cnv t
val _ = (pvars |> map #1 |> has_duplicates op=)
andalso raise TERM ("Duplicate indexname with different type",[t])
val substs = map (fn (x,_,t) => (x,t)) pvars
val t' = subst_Vars substs t
fun mk_asm (x,T,_) = let
val t = Var (x,T)
val t = @{mk_term "Trueprop (CONSTRAINT is_pure ?t)"}
in
t
end
val assms = map mk_asm pvars
fun add_prems prems t = let
val prems' = Logic.strip_imp_prems t
val concl = Logic.strip_imp_concl t
in
Logic.list_implies (prems@prems', concl)
end
val t' = add_prems assms t'
val (t',ctxt) = yield_singleton (Variable.import_terms true) t' ctxt
val thm' = Goal.prove_internal ctxt [] (Thm.cterm_of ctxt t') (fn _ =>
ALLGOALS (resolve_tac ctxt [thm] THEN_ALL_NEW assume_tac ctxt))
val thm' = norm_fcomp_rule ctxt thm'
val thm' = singleton (Variable.export ctxt orig_ctxt) thm'
in
thm'
end
val cfg_simp_precond =
Attrib.setup_config_bool @{binding fcomp_simp_precond} (K true)
local
fun mk_simp_thm ctxt t = let
val st = t
|> HOLogic.mk_Trueprop
|> Thm.cterm_of ctxt
|> Goal.init
val ctxt = Context_Position.set_visible false ctxt
val ctxt = ctxt addsimps (
refine_pw_simps.get ctxt
@ Named_Theorems.get ctxt @{named_theorems fcomp_prenorm_simps}
@ @{thms split_tupled_all cnv_conj_to_meta}
)
val trace_incomplete_transfer_tac =
COND (Thm.prems_of #> exists (strip_all_body #> Logic.strip_imp_concl #> Term.is_open))
(print_tac ctxt "Failed transfer from intermediate level:") all_tac
val tac =
ALLGOALS (resolve_tac ctxt @{thms auto_weaken_pre_comp_PRE_I} )
THEN ALLGOALS (Simplifier.asm_full_simp_tac ctxt)
THEN trace_incomplete_transfer_tac
THEN ALLGOALS (TRY o filter_prems_tac ctxt (K false))
THEN Local_Defs.unfold0_tac ctxt [Drule.triv_forall_equality]
val st' = tac st |> Seq.take 1 |> Seq.list_of
val thm = case st' of [st'] => Goal.conclude st' | _ => raise THM("Simp_Precond: Simp-Tactic failed",~1,[st])
val _ = exists (Logic.is_all) (Thm.prems_of thm)
andalso raise THM("Simp_Precond: Transfer from intermediate level failed",~1,[thm])
val thm =
thm
|> Conv.fconv_rule (Object_Logic.atomize ctxt)
|> Local_Defs.unfold0 ctxt @{thms auto_weaken_pre_to_imp_nf}
val thm = case Thm.concl_of thm of
@{mpat "Trueprop (_ ⟶ _)"} => thm
| @{mpat "Trueprop _"} => thm RS @{thm auto_weaken_pre_add_dummy_imp}
| _ => raise THM("Simp_Precond: Generated odd theorem, expected form 'P⟶Q'",~1,[thm])
in
thm
end
in
fun simplify_precond ctxt thm = let
val orig_ctxt = ctxt
val thm = Refine_Util.OF_fst @{thms auto_weaken_pre_init} [asm_rl,thm]
val thm =
Local_Defs.unfold0 ctxt @{thms split_tupled_all} thm
OF @{thms auto_weaken_pre_uncurry_start}
fun rec_uncurry thm =
case try (fn () => thm OF @{thms auto_weaken_pre_uncurry_step}) () of
NONE => thm OF @{thms auto_weaken_pre_uncurry_finish}
| SOME thm => rec_uncurry thm
val thm = rec_uncurry thm
|> Conv.fconv_rule Thm.eta_conversion
val t = case Thm.prems_of thm of
t::_ => t | _ => raise THM("Simp-Precond: Expected at least one premise",~1,[thm])
val (t,ctxt) = yield_singleton (Variable.import_terms false) t ctxt
val ((_,t),ctxt) = Variable.focus NONE t ctxt
val t = case t of
@{mpat "Trueprop (_ ⟶ ?t)"} => t | _ => raise TERM("Simp_Precond: Expected implication",[t])
val simpthm = mk_simp_thm ctxt t
|> singleton (Variable.export ctxt orig_ctxt)
val thm = thm OF [simpthm]
val thm = Local_Defs.unfold0 ctxt @{thms prod_casesK} thm
in
thm
end
fun simplify_precond_if_cfg ctxt =
if Config.get ctxt cfg_simp_precond then
simplify_precond ctxt
else I
end
fun compose_ff ctxt A B =
(@{thm fref_compI_PRE} OF [A,B])
|> norm_fcomp_rule ctxt
|> simplify_precond_if_cfg ctxt
|> Conv.fconv_rule Thm.eta_conversion
fun compose_hf ctxt A B =
(@{thm hfref_compI_PRE} OF [A,B])
|> norm_fcomp_rule ctxt
|> simplify_precond_if_cfg ctxt
|> Conv.fconv_rule Thm.eta_conversion
|> add_pure_constraints_rule ctxt
|> Conv.fconv_rule Thm.eta_conversion
fun ensure_fref ctxt thm = case rthm_type thm of
RT_HOPARAM => to_fref ctxt thm
| RT_FREF => thm
| _ => raise THM("Expected parametricity or fref theorem",~1,[thm])
fun ensure_fref_nres ctxt thm = let
val thm = ensure_fref ctxt thm
in
case Thm.concl_of thm of
@{mpat (typs) "Trueprop (_∈fref _ _ (_::_ ⇒ (_ nres×_)set))"} => thm
| @{mpat "Trueprop ((_,_)∈fref _ _ _)"} =>
(thm RS @{thm ensure_fref_nresI}) |> Local_Defs.unfold0 ctxt @{thms ensure_fref_nres_unfold}
| _ => raise THM("Expected fref-theorem",~1,[thm])
end
fun ensure_hfref ctxt thm = case rthm_type thm of
RT_HNR => to_hfref ctxt thm
| RT_HFREF => thm
| _ => raise THM("Expected hnr or hfref theorem",~1,[thm])
fun ensure_hnr ctxt thm = case rthm_type thm of
RT_HNR => thm
| RT_HFREF => to_hnr ctxt thm
| _ => raise THM("Expected hnr or hfref theorem",~1,[thm])
fun gen_compose ctxt A B = let
val rtA = rthm_type A
in
if rtA = RT_HOPARAM orelse rtA = RT_FREF then
compose_ff ctxt (ensure_fref ctxt A) (ensure_fref ctxt B)
else
compose_hf ctxt (ensure_hfref ctxt A) ((ensure_fref_nres ctxt B))
end
val parse_fcomp_flags = Refine_Util.parse_paren_lists
(Refine_Util.parse_bool_config "prenorm" cfg_simp_precond)
val fcomp_attrib = parse_fcomp_flags |-- Attrib.thm >> (fn B => Thm.rule_attribute [] (fn context => fn A =>
let
val ctxt = Context.proof_of context
in
gen_compose ctxt A B
end))
end
›
attribute_setup to_fref = ‹
Scan.succeed (Thm.rule_attribute [] (Sepref_Rules.to_fref o Context.proof_of))
› "Convert parametricity theorem to uncurried fref-form"
attribute_setup to_foparam = ‹
Scan.succeed (Thm.rule_attribute [] (Sepref_Rules.to_foparam o Context.proof_of))
› ‹Convert param or fref rule to first order rule›
attribute_setup param_fo = ‹
Scan.succeed (Thm.rule_attribute [] (Sepref_Rules.to_foparam o Context.proof_of))
› ‹Convert param or fref rule to first order rule›
attribute_setup to_hnr = ‹
Scan.succeed (Thm.rule_attribute [] (Sepref_Rules.to_hnr o Context.proof_of))
› "Convert hfref-rule to hnr-rule"
attribute_setup to_hfref = ‹Scan.succeed (
Thm.rule_attribute [] (Context.proof_of #> Sepref_Rules.to_hfref)
)› ‹Convert hnr to hfref theorem›
attribute_setup ensure_fref_nres = ‹Scan.succeed (
Thm.rule_attribute [] (Context.proof_of #> Sepref_Rules.ensure_fref_nres)
)›
attribute_setup sepref_dbg_norm_fcomp_rule = ‹Scan.succeed (
Thm.rule_attribute [] (Context.proof_of #> Sepref_Rules.norm_fcomp_rule)
)›
attribute_setup sepref_simplify_precond = ‹Scan.succeed (
Thm.rule_attribute [] (Context.proof_of #> Sepref_Rules.simplify_precond)
)› ‹Simplify precondition of fref/hfref-theorem›
attribute_setup FCOMP = Sepref_Rules.fcomp_attrib "Composition of refinement rules"
end