Skip to content

Commit

Permalink
[lambda] Böhm tree with basic properties
Browse files Browse the repository at this point in the history
  • Loading branch information
binghe committed Jan 4, 2024
1 parent 7740074 commit 06bbec5
Show file tree
Hide file tree
Showing 7 changed files with 992 additions and 118 deletions.
484 changes: 437 additions & 47 deletions examples/lambda/barendregt/boehmScript.sml

Large diffs are not rendered by default.

16 changes: 16 additions & 0 deletions examples/lambda/barendregt/chap3Script.sml
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,16 @@ val ccbeta_rwt = store_thm(
FULL_SIMP_TAC (srw_ss()) []
]);

Theorem ccbeta_LAMl_rwt :
!vs M N. LAMl vs M -b-> N <=> ?M'. N = LAMl vs M' /\ M -b-> M'
Proof
Induct_on ‘vs’
>> rw [ccbeta_rwt] (* only one goal left *)
>> EQ_TAC >> rw []
>- (Q.EXISTS_TAC ‘M'’ >> art [])
>> Q.EXISTS_TAC ‘LAMl vs M'’ >> rw []
QED

val beta_normal_form_bnf = store_thm(
"beta_normal_form_bnf",
``normal_form beta = bnf``,
Expand Down Expand Up @@ -689,6 +699,7 @@ val ccbeta_lameq = store_thm(
"ccbeta_lameq",
``!M N. M -b-> N ==> M == N``,
SRW_TAC [][lameq_betaconversion, EQC_R]);

val betastar_lameq = store_thm(
"betastar_lameq",
``!M N. M -b->* N ==> M == N``,
Expand Down Expand Up @@ -1578,6 +1589,11 @@ val betastar_eq_cong = store_thm(
``bnf N ==> M -b->* M' ==> (M -b->* N <=> M' -b->* N)``,
METIS_TAC [bnf_triangle, RTC_CASES_RTC_TWICE]);

(* |- !x y z. x -b->* y /\ y -b->* z ==> x -b->* z *)
Theorem betastar_TRANS =
RTC_TRANSITIVE |> Q.ISPEC ‘compat_closure beta’
|> REWRITE_RULE [transitive_def]

val _ = export_theory();
val _ = html_theory "chap3";

Expand Down
113 changes: 75 additions & 38 deletions examples/lambda/barendregt/head_reductionScript.sml
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,9 @@ val hreduce1_FV = store_thm(
``∀M N. M -h-> N ⇒ ∀v. v ∈ FV N ⇒ v ∈ FV M``,
METIS_TAC [SUBSET_DEF, hreduce_ccbeta, cc_beta_FV_SUBSET]);

(* |- !M N. M -h-> N ==> FV N SUBSET FV M *)
Theorem hreduce1_FV_SUBSET = hreduce1_FV |> REWRITE_RULE [GSYM SUBSET_DEF]

Theorem hreduces_FV :
∀M N. M -h->* N ⇒ v ∈ FV N ⇒ v ∈ FV M
Proof
Expand Down Expand Up @@ -985,12 +988,60 @@ Proof
(tail (pcons x r q) = tail (drop i p))’ >- art []
>> simp []
>> DISCH_THEN (fs o wrap)
>> ‘nth_label i p = g i’ by (rw [Abbr ‘p’, nth_label_pgenerate])
>> ‘!i. el i p = f i’ by (rw [Abbr ‘p’, el_pgenerate])
>> ‘nth_label i p = g i’ by rw [Abbr ‘p’, nth_label_pgenerate]
>> ‘!i. el i p = f i’ by rw [Abbr ‘p’, el_pgenerate]
>> ASM_REWRITE_TAC [GSYM ADD1]
>> Q.EXISTS_TAC ‘SUC i’ >> REWRITE_TAC []
QED

(* This theorem guarentees the one-one mapping between the list and the path *)
Theorem finite_head_reduction_path_to_list_11 :
!M p. (p = head_reduction_path M) /\ finite p ==>
?l. l <> [] /\ (HD l = M) /\ hnf (LAST l) /\
(LENGTH l = THE (length p)) /\
(!i. i < LENGTH l ==> (EL i l = el i (head_reduction_path M))) /\
!i. SUC i < LENGTH l ==> EL i l -h-> EL (SUC i) l
Proof
RW_TAC std_ss []
>> qabbrev_tac ‘p = head_reduction_path M’
>> ‘?n. length p = SOME n’ by METIS_TAC [finite_length]
>> Cases_on ‘n’ >- fs [length_never_zero]
>> rename1 ‘length p = SOME (SUC n)’
>> Q.EXISTS_TAC ‘GENLIST (\i. el i p) (SUC n)’ >> simp []
>> STRONG_CONJ_TAC >- rw [NOT_NIL_EQ_LENGTH_NOT_0] >> DISCH_TAC
>> CONJ_TAC >- rw [GSYM EL, Abbr ‘p’, head_reduction_path_def]
>> CONJ_TAC (* hnf *)
>- (qabbrev_tac ‘l = GENLIST (\i. el i p) (SUC n)’ \\
rw [Abbr ‘l’, LAST_EL] \\
Suff ‘el n p = last p’ >- rw [Abbr ‘p’, head_reduction_path_def] \\
rw [finite_last_el])
>> rw [head_reduce1_def]
>> ‘i IN PL p /\ i + 1 IN PL p’ by rw [PL_def]
>> qabbrev_tac ‘q = drop i p’
>> ‘el i p = first q’ by rw [Abbr ‘q’] >> POP_ORW
>> Know ‘el i (tail p) = first (tail q)’
>- (rw [Abbr ‘q’, REWRITE_RULE [ADD1] el_def])
>> Rewr'
>> Know ‘is_head_reduction q’
>- (qunabbrev_tac ‘q’ \\
irule finite_is_head_reduction_drop \\
rw [Abbr ‘p’, head_reduction_path_def])
>> DISCH_TAC
(* perform case analysis *)
>> ‘(?x. q = stopped_at x) \/ ?x r xs. q = pcons x r xs’ by METIS_TAC [path_cases]
>- (Know ‘PL q = IMAGE (\n. n - i) (PL p)’
>- rw [Abbr ‘q’, PL_drop] \\
‘PL p = count (SUC n)’ by rw [PL_def, count_def] >> POP_ORW \\
POP_ORW (* q = stopped_at x *) \\
simp [PL_stopped_at] \\
Suff ‘IMAGE (\n. n - i) (count (SUC n)) <> {0}’ >- METIS_TAC [] \\
rw [Once EXTENSION] \\
Q.EXISTS_TAC ‘n - i’ >> rw [] \\
Q.EXISTS_TAC ‘n’ >> rw [])
>> fs [is_head_reduction_thm]
>> Q.EXISTS_TAC ‘r’ >> art []
QED

Theorem finite_head_reduction_path_to_list :
!M. finite (head_reduction_path M) <=>
?l. l <> [] /\ (HD l = M) /\ hnf (LAST l) /\
Expand All @@ -999,40 +1050,8 @@ Proof
Q.X_GEN_TAC ‘M’
>> qabbrev_tac ‘p = head_reduction_path M’
>> EQ_TAC >> rpt STRIP_TAC
>- (‘?n. length p = SOME n’ by METIS_TAC [finite_length] \\
Cases_on ‘n’ >- fs [length_never_zero] \\
rename1 ‘length p = SOME (SUC n)’ \\
Q.EXISTS_TAC ‘GENLIST (\i. el i p) (SUC n)’ >> simp [] \\
STRONG_CONJ_TAC >- rw [NOT_NIL_EQ_LENGTH_NOT_0] >> DISCH_TAC \\
CONJ_TAC >- rw [GSYM EL, Abbr ‘p’, head_reduction_path_def] \\
CONJ_TAC (* hnf *)
>- (qabbrev_tac ‘l = GENLIST (\i. el i p) (SUC n)’ \\
rw [Abbr ‘l’, LAST_EL] \\
Suff ‘el n p = last p’ >- rw [Abbr ‘p’, head_reduction_path_def] \\
rw [finite_last_el]) \\
rw [head_reduce1_def] \\
‘i IN PL p /\ i + 1 IN PL p’ by rw [PL_def] \\
qabbrev_tac ‘q = drop i p’ \\
‘el i p = first q’ by rw [Abbr ‘q’] >> POP_ORW \\
Know ‘el i (tail p) = first (tail q)’
>- (rw [Abbr ‘q’, REWRITE_RULE [ADD1] el_def]) >> Rewr' \\
Know ‘is_head_reduction q’
>- (qunabbrev_tac ‘q’ \\
irule finite_is_head_reduction_drop \\
rw [Abbr ‘p’, head_reduction_path_def]) >> DISCH_TAC \\
(* perform case analysis *)
‘(?x. q = stopped_at x) \/ ?x r xs. q = pcons x r xs’ by METIS_TAC [path_cases]
>- (Know ‘PL q = IMAGE (\n. n - i) (PL p)’
>- rw [Abbr ‘q’, PL_drop] \\
‘PL p = count (SUC n)’ by rw [PL_def, count_def] >> POP_ORW \\
POP_ORW (* q = stopped_at x *) \\
simp [PL_stopped_at] \\
Suff ‘IMAGE (\n. n - i) (count (SUC n)) <> {0}’ >- METIS_TAC [] \\
rw [Once EXTENSION] \\
Q.EXISTS_TAC ‘n - i’ >> rw [] \\
Q.EXISTS_TAC ‘n’ >> art []) \\
fs [is_head_reduction_thm] \\
Q.EXISTS_TAC ‘r’ >> art [])
>- (MP_TAC (Q.SPECL [‘M’, ‘p’] finite_head_reduction_path_to_list_11) \\
rw [] >> Q.EXISTS_TAC ‘l’ >> rw [])
(* stage work *)
>> qabbrev_tac ‘len = LENGTH l’
>> ‘0 < len /\ len <> 0’ by rw [Abbr ‘len’, GSYM NOT_NIL_EQ_LENGTH_NOT_0]
Expand Down Expand Up @@ -1200,7 +1219,7 @@ QED

Overload hnf_headvar = “\t. THE_VAR (hnf_head t)”

(* hnf_children retrives the ‘args’ part of an abs-free hnf (VAR y @* args) *)
(* hnf_children retrives the ‘args’ part of absfree hnf *)
Definition hnf_children_def :
hnf_children t = if is_comb t then
SNOC (rand t) (hnf_children (rator t))
Expand Down Expand Up @@ -1255,7 +1274,25 @@ Proof
>> rw [hnf_children_appstar, hnf_head_appstar]
QED

Theorem hnf_head_absfree :
Theorem hnf_children_FV_SUBSET :
!M Ms. hnf M /\ ~is_abs M /\ (Ms = hnf_children M) ==>
!i. i < LENGTH Ms ==> FV (EL i Ms) SUBSET FV M
Proof
rpt STRIP_TAC
>> ‘M = hnf_head M @* hnf_children M’ by PROVE_TAC [absfree_hnf_thm]
>> POP_ORW
>> Q.PAT_X_ASSUM ‘Ms = hnf_children M’ (fs o wrap)
>> qabbrev_tac ‘Ms = hnf_children M’
>> simp [FV_appstar]
>> MATCH_MP_TAC SUBSET_TRANS
>> Q.EXISTS_TAC ‘BIGUNION (IMAGE FV (set Ms))’
>> simp []
>> rw [SUBSET_DEF, IN_BIGUNION_IMAGE]
>> Q.EXISTS_TAC ‘EL i Ms’ >> rw [MEM_EL]
>> Q.EXISTS_TAC ‘i’ >> rw []
QED

Theorem absfree_hnf_head :
!y args. hnf_head (VAR y @* args) = VAR y
Proof
rpt GEN_TAC
Expand Down
Loading

0 comments on commit 06bbec5

Please sign in to comment.