(* ========================================================================= *)
(* Results connected with topological dimension.                             *)
(*                                                                           *)
(* At the moment this is just Brouwer's fixpoint theorem. The proof is from  *)
(* Kuhn: "some combinatorial lemmas in topology", IBM J. v4. (1960) p. 518   *)
(* See "http://www.research.ibm.com/journal/rd/045/ibmrd0405K.pdf".          *)
(*                                                                           *)
(* The script below is quite messy, but at least we avoid formalizing any    *)
(* topological machinery; we don't even use barycentric subdivision; this is *)
(* the big advantage of Kuhn's proof over the usual Sperner's lemma one.     *)
(*                                                                           *)
(*              (c) Copyright, John Harrison 1998-2008                       *)
(* ========================================================================= *)

needs "Multivariate/topology.ml";;
needs "Multivariate/paths.ml";;

let BROUWER_COMPACTNESS_LEMMA = 
prove (`!f:real^M->real^N s. compact s /\ f continuous_on s /\ ~(?x. x IN s /\ (f x = vec 0)) ==> ?d. &0 < d /\ !x. x IN s ==> d <= norm(f x)`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`norm o (f:real^M->real^N)`; `s:real^M->bool`] CONTINUOUS_ATTAINS_INF) THEN ASM_CASES_TAC `s:real^M->bool = {}` THEN ASM_REWRITE_TAC[NOT_IN_EMPTY] THENL [MESON_TAC[REAL_LT_01]; ALL_TAC] THEN ASM_SIMP_TAC[CONTINUOUS_ON_COMPOSE; o_ASSOC; CONTINUOUS_ON_LIFT_NORM] THEN REWRITE_TAC[o_THM] THEN ASM_MESON_TAC[NORM_POS_LT]);;
let KUHN_LABELLING_LEMMA = 
prove (`!f:real^N->real^N P Q. (!x. P x ==> P (f x)) ==> (!x. P x ==> (!i. Q i ==> &0 <= x$i /\ x$i <= &1)) ==> ?l. (!x i. l x i <= 1) /\ (!x i. P x /\ Q i /\ (x$i = &0) ==> (l x i = 0)) /\ (!x i. P x /\ Q i /\ (x$i = &1) ==> (l x i = 1)) /\ (!x i. P x /\ Q i /\ (l x i = 0) ==> x$i <= f(x)$i) /\ (!x i. P x /\ Q i /\ (l x i = 1) ==> f(x)$i <= x$i)`,
REPEAT GEN_TAC THEN REWRITE_TAC[AND_FORALL_THM; GSYM SKOLEM_THM] THEN REWRITE_TAC[ARITH_RULE `n <= 1 <=> (n = 0) \/ (n = 1)`; RIGHT_OR_DISTRIB; EXISTS_OR_THM; UNWIND_THM2; ARITH_EQ] THEN MESON_TAC[REAL_ARITH `!x y. &0 <= x /\ x <= &1 /\ &0 <= y /\ y <= &1 ==> ~(x = &1) /\ x <= y \/ ~(x = &0) /\ y <= x`]);;
(* ------------------------------------------------------------------------- *) (* The key "counting" observation, somewhat abstracted. *) (* ------------------------------------------------------------------------- *)
let KUHN_COUNTING_LEMMA = 
prove (`!face:F->S->bool faces simplices comp comp' bnd. FINITE faces /\ FINITE simplices /\ (!f. f IN faces /\ bnd f ==> (CARD {s | s IN simplices /\ face f s} = 1)) /\ (!f. f IN faces /\ ~bnd f ==> (CARD {s | s IN simplices /\ face f s} = 2)) /\ (!s. s IN simplices /\ comp s ==> (CARD {f | f IN faces /\ face f s /\ comp' f} = 1)) /\ (!s. s IN simplices /\ ~comp s ==> (CARD {f | f IN faces /\ face f s /\ comp' f} = 0) \/ (CARD {f | f IN faces /\ face f s /\ comp' f} = 2)) ==> ODD(CARD {f | f IN faces /\ comp' f /\ bnd f}) ==> ODD(CARD {s | s IN simplices /\ comp s})`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `sum simplices (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) = sum simplices (\s. &(CARD {f | f IN {f | f IN faces /\ comp' f /\ bnd f} /\ face f s})) + sum simplices (\s. &(CARD {f | f IN {f | f IN faces /\ comp' f /\ ~(bnd f)} /\ face f s}))` MP_TAC THENL [ASM_SIMP_TAC[GSYM SUM_ADD] THEN MATCH_MP_TAC SUM_EQ THEN ASM_REWRITE_TAC[] THEN REPEAT STRIP_TAC THEN CONV_TAC SYM_CONV THEN REWRITE_TAC[REAL_OF_NUM_ADD; REAL_OF_NUM_EQ] THEN MATCH_MP_TAC CARD_UNION_EQ THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_INTER; IN_UNION; NOT_IN_EMPTY] THEN CONJ_TAC THEN GEN_TAC THEN CONV_TAC TAUT; ALL_TAC] THEN MP_TAC(ISPECL [`\s f. (face:F->S->bool) f s`; `simplices:S->bool`; `{f:F | f IN faces /\ comp' f /\ bnd f}`; `1`] SUM_MULTICOUNT) THEN MP_TAC(ISPECL [`\s f. (face:F->S->bool) f s`; `simplices:S->bool`; `{f:F | f IN faces /\ comp' f /\ ~(bnd f)}`; `2`] SUM_MULTICOUNT) THEN REWRITE_TAC[] THEN REPEAT(ANTS_TAC THENL [ASM_SIMP_TAC[FINITE_RESTRICT] THEN GEN_TAC THEN DISCH_THEN(fun th -> FIRST_ASSUM MATCH_MP_TAC THEN MP_TAC th) THEN SIMP_TAC[IN_ELIM_THM]; DISCH_THEN SUBST1_TAC]) THEN SUBGOAL_THEN `sum simplices (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) = sum {s | s IN simplices /\ comp s} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) + sum {s | s IN simplices /\ ~(comp s)} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f}))` SUBST1_TAC THENL [CONV_TAC SYM_CONV THEN MATCH_MP_TAC SUM_UNION_EQ THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[EXTENSION; NOT_IN_EMPTY] THEN REWRITE_TAC[IN_ELIM_THM; IN_INTER; IN_UNION] THEN CONJ_TAC THEN GEN_TAC THEN CONV_TAC TAUT; ALL_TAC] THEN SUBGOAL_THEN `sum {s | s IN simplices /\ comp s} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) = sum {s | s IN simplices /\ comp s} (\s. &1)` SUBST1_TAC THENL [MATCH_MP_TAC SUM_EQ THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN GEN_TAC THEN REWRITE_TAC[REAL_OF_NUM_EQ] THEN DISCH_THEN(fun th -> FIRST_ASSUM MATCH_MP_TAC THEN MP_TAC th) THEN SIMP_TAC[IN_ELIM_THM]; ALL_TAC] THEN SUBGOAL_THEN `sum {s | s IN simplices /\ ~(comp s)} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) = sum {s | s IN simplices /\ ~(comp s) /\ (CARD {f | f IN faces /\ face f s /\ comp' f} = 0)} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) + sum {s | s IN simplices /\ ~(comp s) /\ (CARD {f | f IN faces /\ face f s /\ comp' f} = 2)} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f}))` SUBST1_TAC THENL [CONV_TAC SYM_CONV THEN MATCH_MP_TAC SUM_UNION_EQ THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN REWRITE_TAC[EXTENSION; NOT_IN_EMPTY; IN_INTER; IN_UNION] THEN CONJ_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN MESON_TAC[ARITH_RULE `~(2 = 0)`]; ALL_TAC] THEN X_GEN_TAC `s:S` THEN UNDISCH_TAC `!s:S. s IN simplices /\ ~comp s ==> (CARD {f:F | f IN faces /\ face f s /\ comp' f} = 0) \/ (CARD {f | f IN faces /\ face f s /\ comp' f} = 2)` THEN DISCH_THEN(MP_TAC o SPEC `s:S`) THEN REWRITE_TAC[IN_ELIM_THM] THEN CONV_TAC TAUT; ALL_TAC] THEN SUBGOAL_THEN `!n. sum {s | s IN simplices /\ ~(comp s) /\ (CARD {f | f IN faces /\ face f s /\ comp' f} = n)} (\s:S. &(CARD {f:F | f IN faces /\ face f s /\ comp' f})) = sum {s | s IN simplices /\ ~(comp s) /\ (CARD {f | f IN faces /\ face f s /\ comp' f} = n)} (\s:S. &n)` (fun th -> REWRITE_TAC[th]) THENL [GEN_TAC THEN MATCH_MP_TAC SUM_EQ THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN SIMP_TAC[IN_ELIM_THM]; ALL_TAC] THEN REWRITE_TAC[SUM_0] THEN ASM_SIMP_TAC[SUM_CONST; FINITE_RESTRICT] THEN REWRITE_TAC[REAL_OF_NUM_MUL; REAL_OF_NUM_ADD; REAL_OF_NUM_EQ] THEN REWRITE_TAC[ADD_CLAUSES; MULT_CLAUSES] THEN FIRST_X_ASSUM(CHOOSE_THEN SUBST1_TAC o GEN_REWRITE_RULE I [ODD_EXISTS]) THEN DISCH_THEN(MP_TAC o AP_TERM `ODD`) THEN REWRITE_TAC[ODD_ADD; ODD_MULT; ARITH_ODD; ODD]);;
(* ------------------------------------------------------------------------- *) (* The odd/even result for faces of complete vertices, generalized. *) (* ------------------------------------------------------------------------- *)
let HAS_SIZE_1_EXISTS = 
prove (`!s. s HAS_SIZE 1 <=> ?!x. x IN s`,
REPEAT GEN_TAC THEN CONV_TAC(LAND_CONV HAS_SIZE_CONV) THEN REWRITE_TAC[EXTENSION; IN_SING] THEN MESON_TAC[]);;
let HAS_SIZE_2_EXISTS = 
prove (`!s. s HAS_SIZE 2 <=> ?x y. ~(x = y) /\ !z. z IN s <=> (z = x) \/ (z = y)`,
REPEAT GEN_TAC THEN CONV_TAC(LAND_CONV HAS_SIZE_CONV) THEN REWRITE_TAC[EXTENSION; IN_INSERT; NOT_IN_EMPTY] THEN MESON_TAC[]);;
let IMAGE_LEMMA_0 = 
prove (`!f:A->B s n. {a | a IN s /\ (IMAGE f (s DELETE a) = t DELETE b)} HAS_SIZE n ==> {s' | ?a. a IN s /\ (s' = s DELETE a) /\ (IMAGE f s' = t DELETE b)} HAS_SIZE n`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `{s' | ?a. a IN s /\ (s' = s DELETE a) /\ (IMAGE f s' = t DELETE b)} = IMAGE (\a. s DELETE a) {a | a IN s /\ (IMAGE (f:A->B) (s DELETE a) = t DELETE b)}` SUBST1_TAC THENL [GEN_REWRITE_TAC I [EXTENSION] THEN REWRITE_TAC[IN_ELIM_THM; IN_IMAGE] THEN MESON_TAC[]; MATCH_MP_TAC HAS_SIZE_IMAGE_INJ THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_DELETE] THEN MESON_TAC[]]);;
let IMAGE_LEMMA_1 = 
prove (`!f:A->B s t b. FINITE s /\ FINITE t /\ (CARD s = CARD t) /\ (IMAGE f s = t) /\ b IN t ==> (CARD {s' | ?a. a IN s /\ (s' = s DELETE a) /\ (IMAGE f s' = t DELETE b)} = 1)`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC HAS_SIZE_CARD THEN MATCH_MP_TAC IMAGE_LEMMA_0 THEN REWRITE_TAC[HAS_SIZE_1_EXISTS] THEN SUBGOAL_THEN `!x y. x IN s /\ y IN s /\ ((f:A->B) x = f y) ==> (x = y)` ASSUME_TAC THENL [ASM_MESON_TAC[IMAGE_IMP_INJECTIVE_GEN]; ALL_TAC] THEN REWRITE_TAC[EXISTS_UNIQUE_THM; IN_ELIM_THM] THEN CONJ_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [EXTENSION]) THEN REWRITE_TAC[IN_IMAGE] THENL [DISCH_THEN(fun th -> MP_TAC(SPEC `b:B` th) THEN MP_TAC th) THEN ASM_REWRITE_TAC[] THEN DISCH_TAC THEN MATCH_MP_TAC MONO_EXISTS THEN REWRITE_TAC[EXTENSION; IN_IMAGE; IN_DELETE] THEN REPEAT STRIP_TAC THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[]; REWRITE_TAC[EXTENSION; IN_IMAGE; IN_DELETE] THEN ASM_MESON_TAC[]]);;
let IMAGE_LEMMA_2 = 
prove (`!f:A->B s t b. FINITE s /\ FINITE t /\ (CARD s = CARD t) /\ (IMAGE f s) SUBSET t /\ ~(IMAGE f s = t) /\ b IN t ==> (CARD {s' | ?a. a IN s /\ (s' = s DELETE a) /\ (IMAGE f s' = t DELETE b)} = 0) \/ (CARD {s' | ?a. a IN s /\ (s' = s DELETE a) /\ (IMAGE f s' = t DELETE b)} = 2)`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `{a | a IN s /\ (IMAGE (f:A->B) (s DELETE a) = t DELETE b)} = {}` THENL [DISJ1_TAC; DISJ2_TAC] THEN MATCH_MP_TAC HAS_SIZE_CARD THEN MATCH_MP_TAC IMAGE_LEMMA_0 THEN ASM_REWRITE_TAC[HAS_SIZE_0; HAS_SIZE_2_EXISTS] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [GSYM MEMBER_NOT_EMPTY]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `a1:A` THEN REWRITE_TAC[IN_ELIM_THM] THEN STRIP_TAC THEN SUBGOAL_THEN `(f:A->B) a1 IN (t DELETE b)` ASSUME_TAC THENL [REWRITE_TAC[IN_DELETE] THEN ASM_MESON_TAC[SUBSET; IN_IMAGE; INSERT_DELETE; IMAGE_CLAUSES]; ALL_TAC] THEN FIRST_ASSUM(MP_TAC o GEN_REWRITE_RULE I [EXTENSION]) THEN DISCH_THEN(MP_TAC o SPEC `(f:A->B) a1`) THEN ASM_REWRITE_TAC[IN_IMAGE] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `a2:A` THEN REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `!x y. x IN (s DELETE a1) /\ y IN (s DELETE a1) /\ ((f:A->B) x = f y) ==> (x = y)` MP_TAC THENL [MATCH_MP_TAC IMAGE_IMP_INJECTIVE_GEN THEN EXISTS_TAC `t DELETE (b:B)` THEN ASM_SIMP_TAC[CARD_DELETE; FINITE_DELETE]; REWRITE_TAC[IN_DELETE] THEN DISCH_TAC] THEN X_GEN_TAC `a:A` THEN ASM_CASES_TAC `a:A = a1` THEN ASM_REWRITE_TAC[] THEN ASM_CASES_TAC `(a:A) IN s` THEN ASM_REWRITE_TAC[] THENL [ALL_TAC; ASM_MESON_TAC[]] THEN MATCH_MP_TAC EQ_TRANS THEN EXISTS_TAC `(f:A->B) a = f a1` THEN CONJ_TAC THENL [ALL_TAC; ASM_MESON_TAC[IN_DELETE]] THEN FIRST_X_ASSUM(fun t -> GEN_REWRITE_TAC (LAND_CONV o RAND_CONV) [SYM t]) THEN REWRITE_TAC[EXTENSION; IN_IMAGE; IN_DELETE] THEN EQ_TAC THENL [DISCH_THEN(MP_TAC o SPEC `(f:A->B) a`); ALL_TAC] THEN ASM_MESON_TAC[]);;
(* ------------------------------------------------------------------------- *) (* Combine this with the basic counting lemma. *) (* ------------------------------------------------------------------------- *)
let KUHN_COMPLETE_LEMMA = 
prove (`!face:(A->bool)->(A->bool)->bool simplices rl bnd n. FINITE simplices /\ (!f s. face f s <=> ?a. a IN s /\ (f = s DELETE a)) /\ (!s. s IN simplices ==> s HAS_SIZE (n + 2) /\ (IMAGE rl s) SUBSET 0..n+1) /\ (!f. f IN {f | ?s. s IN simplices /\ face f s} /\ bnd f ==> (CARD {s | s IN simplices /\ face f s} = 1)) /\ (!f. f IN {f | ?s. s IN simplices /\ face f s} /\ ~bnd f ==> (CARD {s | s IN simplices /\ face f s} = 2)) ==> ODD(CARD {f | f IN {f | ?s. s IN simplices /\ face f s} /\ (IMAGE rl f = 0..n) /\ bnd f}) ==> ODD(CARD {s | s IN simplices /\ (IMAGE rl s = 0..n+1)})`,
REPEAT GEN_TAC THEN STRIP_TAC THEN SUBGOAL_THEN `!P f:A->bool s. s IN simplices ==> (f IN {f | ?s. s IN simplices /\ (?a. a IN s /\ (f = s DELETE a))} /\ (?a. a IN s /\ (f = s DELETE a)) /\ P f <=> (?a. a IN s /\ (f = s DELETE a) /\ P f))` ASSUME_TAC THENL [ASM_REWRITE_TAC[IN_ELIM_THM] THEN MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `0..n = (0..n+1) DELETE (n+1)` SUBST1_TAC THENL [REWRITE_TAC[EXTENSION; IN_NUMSEG; IN_DELETE] THEN ARITH_TAC; ALL_TAC] THEN MATCH_MP_TAC KUHN_COUNTING_LEMMA THEN EXISTS_TAC `face:(A->bool)->(A->bool)->bool` THEN REPEAT CONJ_TAC THEN TRY(FIRST_ASSUM ACCEPT_TAC) THEN ASM_SIMP_TAC[] THENL [SUBGOAL_THEN `{f:A->bool | ?s. s IN simplices /\ (?a. a IN s /\ (f = s DELETE a))} = UNIONS (IMAGE (\s. {f | ?a. a IN s /\ (f = s DELETE a)}) simplices)` SUBST1_TAC THENL [REWRITE_TAC[EXTENSION; UNIONS_IMAGE; IN_ELIM_THM]; ALL_TAC] THEN ASM_SIMP_TAC[FINITE_UNIONS; FINITE_IMAGE] THEN REWRITE_TAC[FORALL_IN_IMAGE] THEN X_GEN_TAC `s:A->bool` THEN DISCH_TAC THEN MATCH_MP_TAC FINITE_SUBSET THEN EXISTS_TAC `{t:A->bool | t SUBSET s}` THEN CONJ_TAC THENL [MATCH_MP_TAC FINITE_POWERSET THEN ASM_MESON_TAC[HAS_SIZE]; SIMP_TAC[SUBSET; IN_ELIM_THM; LEFT_IMP_EXISTS_THM; IN_DELETE]]; REPEAT STRIP_TAC THEN MATCH_MP_TAC IMAGE_LEMMA_1; REPEAT STRIP_TAC THEN MATCH_MP_TAC IMAGE_LEMMA_2] THEN ASM_REWRITE_TAC[FINITE_NUMSEG; IN_NUMSEG; LE_0; LE_REFL] THEN REWRITE_TAC[CARD_NUMSEG; ARITH_RULE `((n + 1) + 1) - 0 = n + 2`] THEN ASM_MESON_TAC[HAS_SIZE]);;
(* ------------------------------------------------------------------------- *) (* We use the following notion of ordering rather than pointwise indexing. *) (* ------------------------------------------------------------------------- *)
let kle = new_definition
  `kle n x y <=> ?k. k SUBSET 1..n /\
                     (!j. y(j) = x(j) + (if j IN k then 1 else 0))`;;
let KLE_REFL = 
prove (`!n x. kle n x x`,
REPEAT GEN_TAC THEN REWRITE_TAC[kle] THEN EXISTS_TAC `{}:num->bool` THEN REWRITE_TAC[ADD_CLAUSES; NOT_IN_EMPTY; EMPTY_SUBSET]);;
let KLE_ANTISYM = 
prove (`!n x y. kle n x y /\ kle n y x <=> (x = y)`,
REPEAT GEN_TAC THEN EQ_TAC THENL [REWRITE_TAC[kle]; MESON_TAC[KLE_REFL]] THEN DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC MP_TAC) THEN ASM_REWRITE_TAC[FUN_EQ_THM] THEN MESON_TAC[ARITH_RULE `(x = (x + a) + b) ==> (x = x + a:num)`]);;
let POINTWISE_MINIMAL,POINTWISE_MAXIMAL = (CONJ_PAIR o prove) (`(!s:(num->num)->bool. FINITE s ==> ~(s = {}) /\ (!x y. x IN s /\ y IN s ==> (!j. x(j) <= y(j)) \/ (!j. y(j) <= x(j))) ==> ?a. a IN s /\ !x. x IN s ==> !j. a(j) <= x(j)) /\ (!s:(num->num)->bool. FINITE s ==> ~(s = {}) /\ (!x y. x IN s /\ y IN s ==> (!j. x(j) <= y(j)) \/ (!j. y(j) <= x(j))) ==> ?a. a IN s /\ !x. x IN s ==> !j. x(j) <= a(j))`, CONJ_TAC THEN (MATCH_MP_TAC FINITE_INDUCT_STRONG THEN REWRITE_TAC[NOT_INSERT_EMPTY] THEN MAP_EVERY X_GEN_TAC [`a:num->num`; `s:(num->num)->bool`] THEN ASM_CASES_TAC `s:(num->num)->bool = {}` THEN ASM_REWRITE_TAC[] THENL [REWRITE_TAC[IN_SING] THEN MESON_TAC[LE_REFL]; ALL_TAC] THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC STRIP_ASSUME_TAC) THEN DISCH_THEN(fun th -> STRIP_TAC THEN MP_TAC th) THEN ANTS_TAC THENL [ASM_MESON_TAC[IN_INSERT]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `b:num->num` STRIP_ASSUME_TAC) THEN FIRST_X_ASSUM(MP_TAC o SPECL [`a:num->num`; `b:num->num`]) THEN ASM_REWRITE_TAC[IN_INSERT] THEN ASM_MESON_TAC[LE_CASES; LE_TRANS]));;
let KLE_IMP_POINTWISE = 
prove (`!n x y. kle n x y ==> !j. x(j) <= y(j)`,
REWRITE_TAC[kle] THEN REPEAT STRIP_TAC THEN ASM_REWRITE_TAC[LE_ADD]);;
let POINTWISE_ANTISYM = 
prove (`!x y:num->num. (!j. x(j) <= y(j)) /\ (!j. y(j) <= x(j)) <=> (x = y)`,
let KLE_TRANS = 
prove (`!x y z n. kle n x y /\ kle n y z /\ (kle n x z \/ kle n z x) ==> kle n x z`,
REPEAT STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `x:num->num = z` (fun th -> REWRITE_TAC[KLE_REFL; th]) THEN REWRITE_TAC[FUN_EQ_THM; GSYM LE_ANTISYM; FORALL_AND_THM] THEN ASM_MESON_TAC[KLE_IMP_POINTWISE; LE_TRANS]);;
let KLE_STRICT = 
prove (`!n x y. kle n x y /\ ~(x = y) ==> (!j. x(j) <= y(j)) /\ (?k. 1 <= k /\ k <= n /\ x(k) < y(k))`,
REPEAT GEN_TAC THEN REWRITE_TAC[kle] THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC ASSUME_TAC) THEN DISCH_THEN(X_CHOOSE_THEN `k:num->bool` MP_TAC) THEN ASM_CASES_TAC `k:num->bool = {}` THENL [ASM_REWRITE_TAC[NOT_IN_EMPTY; ADD_CLAUSES; GSYM FUN_EQ_THM; ETA_AX]; STRIP_TAC THEN ASM_REWRITE_TAC[LE_ADD] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [GSYM MEMBER_NOT_EMPTY]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `i:num` THEN STRIP_TAC THEN ASM_REWRITE_TAC[ARITH_RULE `n < n + 1`] THEN ASM_MESON_TAC[SUBSET; IN_NUMSEG]]);;
let KLE_MINIMAL = 
prove (`!s n. FINITE s /\ ~(s = {}) /\ (!x y. x IN s /\ y IN s ==> kle n x y \/ kle n y x) ==> ?a. a IN s /\ !x. x IN s ==> kle n a x`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `?a:num->num. a IN s /\ !x. x IN s ==> !j. a(j) <= x(j)` MP_TAC THENL [MATCH_MP_TAC(REWRITE_RULE[IMP_IMP] POINTWISE_MINIMAL); ALL_TAC] THEN ASM_MESON_TAC[POINTWISE_ANTISYM; KLE_IMP_POINTWISE]);;
let KLE_MAXIMAL = 
prove (`!s n. FINITE s /\ ~(s = {}) /\ (!x y. x IN s /\ y IN s ==> kle n x y \/ kle n y x) ==> ?a. a IN s /\ !x. x IN s ==> kle n x a`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `?a:num->num. a IN s /\ !x. x IN s ==> !j. x(j) <= a(j)` MP_TAC THENL [MATCH_MP_TAC(REWRITE_RULE[IMP_IMP] POINTWISE_MAXIMAL); ALL_TAC] THEN ASM_MESON_TAC[POINTWISE_ANTISYM; KLE_IMP_POINTWISE]);;
let KLE_STRICT_SET = 
prove (`!n x y. kle n x y /\ ~(x = y) ==> 1 <= CARD {k | k IN 1..n /\ x(k) < y(k)}`,
REPEAT GEN_TAC THEN DISCH_THEN(MP_TAC o MATCH_MP KLE_STRICT) THEN DISCH_THEN(X_CHOOSE_THEN `i:num` STRIP_ASSUME_TAC o CONJUNCT2) THEN MATCH_MP_TAC LE_TRANS THEN EXISTS_TAC `CARD {i:num}` THEN CONJ_TAC THENL [SIMP_TAC[CARD_CLAUSES; FINITE_RULES; ARITH; NOT_IN_EMPTY]; MATCH_MP_TAC CARD_SUBSET THEN SIMP_TAC[FINITE_RESTRICT; FINITE_NUMSEG] THEN SIMP_TAC[IN_ELIM_THM; IN_NUMSEG; SUBSET; IN_SING] THEN ASM_MESON_TAC[]]);;
let KLE_RANGE_COMBINE = 
prove (`!n x y m1 m2. kle n x y /\ kle n y z /\ (kle n x z \/ kle n z x) /\ m1 <= CARD {k | k IN 1..n /\ x(k) < y(k)} /\ m2 <= CARD {k | k IN 1..n /\ y(k) < z(k)} ==> kle n x z /\ m1 + m2 <= CARD {k | k IN 1..n /\ x(k) < z(k)}`,
REPEAT GEN_TAC THEN DISCH_THEN(REPEAT_TCL CONJUNCTS_THEN ASSUME_TAC) THEN MATCH_MP_TAC(TAUT `a /\ (a ==> b) ==> a /\ b`) THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; DISCH_TAC] THEN MATCH_MP_TAC LE_TRANS THEN EXISTS_TAC `CARD {k | k IN 1..n /\ x(k):num < y(k)} + CARD {k | k IN 1..n /\ y(k) < z(k)}` THEN ASM_SIMP_TAC[LE_ADD2] THEN MATCH_MP_TAC EQ_IMP_LE THEN MATCH_MP_TAC CARD_UNION_EQ THEN SIMP_TAC[FINITE_RESTRICT; FINITE_NUMSEG] THEN REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_INTER; IN_UNION; NOT_IN_EMPTY] THEN CONJ_TAC THENL [ALL_TAC; ASM_MESON_TAC[KLE_IMP_POINTWISE; ARITH_RULE `x <= y:num /\ y <= z ==> (x < y \/ y < z <=> x < z)`]] THEN X_GEN_TAC `i:num` THEN UNDISCH_TAC `kle n x z` THEN REWRITE_TAC[kle] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN ASM_CASES_TAC `i IN 1..n` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(ARITH_RULE `d <= 1 ==> ~(a < x /\ x < a + d)`) THEN COND_CASES_TAC THEN REWRITE_TAC[ARITH]);;
let KLE_RANGE_COMBINE_L = 
prove (`!n x y m. kle n x y /\ kle n y z /\ (kle n x z \/ kle n z x) /\ m <= CARD {k | k IN 1..n /\ y(k) < z(k)} ==> kle n x z /\ m <= CARD {k | k IN 1..n /\ x(k) < z(k)}`,
REPEAT GEN_TAC THEN ASM_CASES_TAC `x:num->num = y` THEN ASM_SIMP_TAC[] THEN DISCH_THEN(REPEAT_TCL CONJUNCTS_THEN ASSUME_TAC) THEN SUBGOAL_THEN `kle n x z /\ 1 + m <= CARD {k | k IN 1 .. n /\ x k < z k}` (fun th -> MESON_TAC[th; ARITH_RULE `1 + m <= x ==> m <= x`]) THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `y:num->num` THEN ASM_SIMP_TAC[KLE_STRICT_SET]);;
let KLE_RANGE_COMBINE_R = 
prove (`!n x y m. kle n x y /\ kle n y z /\ (kle n x z \/ kle n z x) /\ m <= CARD {k | k IN 1..n /\ x(k) < y(k)} ==> kle n x z /\ m <= CARD {k | k IN 1..n /\ x(k) < z(k)}`,
REPEAT GEN_TAC THEN ASM_CASES_TAC `y:num->num = z` THEN ASM_SIMP_TAC[] THEN DISCH_THEN(REPEAT_TCL CONJUNCTS_THEN ASSUME_TAC) THEN SUBGOAL_THEN `kle n x z /\ m + 1 <= CARD {k | k IN 1 .. n /\ x k < z k}` (fun th -> MESON_TAC[th; ARITH_RULE `m + 1 <= x ==> m <= x`]) THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `y:num->num` THEN ASM_SIMP_TAC[KLE_STRICT_SET]);;
let KLE_RANGE_INDUCT = 
prove (`!n m s. s HAS_SIZE (SUC m) ==> (!x y. x IN s /\ y IN s ==> kle n x y \/ kle n y x) ==> ?x y. x IN s /\ y IN s /\ kle n x y /\ m <= CARD {k | k IN 1..n /\ x(k) < y(k)}`,
GEN_TAC THEN INDUCT_TAC THENL [GEN_TAC THEN REWRITE_TAC[ARITH; LE_0] THEN CONV_TAC(LAND_CONV HAS_SIZE_CONV) THEN MESON_TAC[IN_SING; KLE_REFL]; ALL_TAC] THEN X_GEN_TAC `s:(num->num)->bool` THEN ONCE_REWRITE_TAC[HAS_SIZE_SUC] THEN REPEAT STRIP_TAC THEN MP_TAC(SPECL [`s:(num->num)->bool`; `n:num`] KLE_MINIMAL) THEN ANTS_TAC THENL [ASM_MESON_TAC[HAS_SIZE_SUC; HAS_SIZE]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `a:num->num` THEN STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `s DELETE (a:num->num)`) THEN REPEAT(ANTS_TAC THENL [ASM_MESON_TAC[IN_DELETE]; ALL_TAC]) THEN DISCH_THEN(X_CHOOSE_THEN `x:num->num` MP_TAC) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `b:num->num` THEN REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[ARITH_RULE `SUC m = 1 + m`] THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `x:num->num` THEN ASM_SIMP_TAC[KLE_STRICT_SET]);;
let KLE_SUC = 
prove (`!n x y. kle n x y ==> kle (n + 1) x y`,
REPEAT GEN_TAC THEN REWRITE_TAC[kle] THEN MATCH_MP_TAC MONO_EXISTS THEN REWRITE_TAC[SUBSET; IN_NUMSEG] THEN MESON_TAC[ARITH_RULE `k <= n ==> k <= n + 1`]);;
let KLE_TRANS_1 = 
prove (`!n x y. kle n x y ==> !j. x j <= y j /\ y j <= x j + 1`,
SIMP_TAC[kle; LEFT_IMP_EXISTS_THM] THEN REPEAT STRIP_TAC THEN COND_CASES_TAC THEN ARITH_TAC);;
let KLE_TRANS_2 = 
prove (`!a b c. kle n a b /\ kle n b c /\ (!j. c j <= a j + 1) ==> kle n a c`,
REPEAT GEN_TAC THEN REWRITE_TAC[IMP_CONJ] THEN REWRITE_TAC[kle] THEN DISCH_THEN(X_CHOOSE_THEN `kk1:num->bool` STRIP_ASSUME_TAC) THEN DISCH_THEN(X_CHOOSE_THEN `kk2:num->bool` STRIP_ASSUME_TAC) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(fun th -> EXISTS_TAC `(kk1:num->bool) UNION kk2` THEN MP_TAC th) THEN ASM_REWRITE_TAC[UNION_SUBSET; IN_UNION] THEN MATCH_MP_TAC MONO_FORALL THEN X_GEN_TAC `i:num` THEN ASM_CASES_TAC `(i:num) IN kk1` THEN ASM_CASES_TAC `(i:num) IN kk2` THEN ASM_REWRITE_TAC[] THEN ARITH_TAC);;
let KLE_BETWEEN_R = 
prove (`!a b c x. kle n a b /\ kle n b c /\ kle n a x /\ kle n c x ==> kle n b x`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC KLE_TRANS_2 THEN EXISTS_TAC `c:num->num` THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[KLE_TRANS_1; ARITH_RULE `x <= c + 1 /\ c <= b ==> x <= b + 1`]);;
let KLE_BETWEEN_L = 
prove (`!a b c x. kle n a b /\ kle n b c /\ kle n x a /\ kle n x c ==> kle n x b`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC KLE_TRANS_2 THEN EXISTS_TAC `a:num->num` THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[KLE_TRANS_1; ARITH_RULE `c <= x + 1 /\ b <= c ==> b <= x + 1`]);;
let KLE_ADJACENT = 
prove (`!a b x k. 1 <= k /\ k <= n /\ (!j. b(j) = if j = k then a(j) + 1 else a(j)) /\ kle n a x /\ kle n x b ==> (x = a) \/ (x = b)`,
REPEAT STRIP_TAC THEN REPEAT(FIRST_X_ASSUM(MP_TAC o MATCH_MP KLE_IMP_POINTWISE)) THEN ASM_REWRITE_TAC[FUN_EQ_THM; IMP_IMP; AND_FORALL_THM] THEN ASM_CASES_TAC `(x:num->num) k = a k` THENL [DISCH_THEN(fun th -> DISJ1_TAC THEN MP_TAC th); DISCH_THEN(fun th -> DISJ2_TAC THEN MP_TAC th)] THEN MATCH_MP_TAC MONO_FORALL THEN GEN_TAC THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[LE_ANTISYM] THEN ASM_MESON_TAC[ARITH_RULE `a <= x /\ x <= a + 1 /\ ~(x = a) ==> (x = a + 1)`]);;
(* ------------------------------------------------------------------------- *) (* Kuhn's notion of a simplex (my reformulation to avoid so much indexing). *) (* ------------------------------------------------------------------------- *)
let ksimplex = new_definition
 `ksimplex p n s <=>
        s HAS_SIZE (n + 1) /\
        (!x j. x IN s ==> x(j) <= p) /\
        (!x j. x IN s /\ ~(1 <= j /\ j <= n) ==> (x j = p)) /\
        (!x y. x IN s /\ y IN s ==> kle n x y \/ kle n y x)`;;
let KSIMPLEX_EXTREMA = 
prove (`!p n s. ksimplex p n s ==> ?a b. a IN s /\ b IN s /\ (!x. x IN s ==> kle n a x /\ kle n x b) /\ (!i. b(i) = if 1 <= i /\ i <= n then a(i) + 1 else a(i))`,
REPEAT GEN_TAC THEN REWRITE_TAC[ksimplex] THEN ASM_CASES_TAC `n = 0` THENL [ASM_REWRITE_TAC[ARITH_RULE `1 <= i /\ i <= 0 <=> F`; GSYM FUN_EQ_THM] THEN REWRITE_TAC[ADD_CLAUSES; ETA_AX] THEN CONV_TAC(LAND_CONV(LAND_CONV HAS_SIZE_CONV)) THEN DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC MP_TAC) THEN ASM_REWRITE_TAC[IN_SING] THEN MESON_TAC[KLE_REFL]; ALL_TAC] THEN REPEAT STRIP_TAC THEN MP_TAC(SPECL [`s:(num->num)->bool`; `n:num`] KLE_MINIMAL) THEN ANTS_TAC THENL [ASM_MESON_TAC[HAS_SIZE; HAS_SIZE_SUC; ADD1]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `a:num->num` THEN STRIP_TAC THEN MP_TAC(SPECL [`s:(num->num)->bool`; `n:num`] KLE_MAXIMAL) THEN ANTS_TAC THENL [ASM_MESON_TAC[HAS_SIZE; HAS_SIZE_SUC; ADD1]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `b:num->num` THEN STRIP_TAC THEN ASM_SIMP_TAC[] THEN MP_TAC(SPECL [`n:num`; `n:num`; `s:(num->num)->bool`] KLE_RANGE_INDUCT) THEN ASM_REWRITE_TAC[ADD1] THEN DISCH_THEN(X_CHOOSE_THEN `c:num->num` (X_CHOOSE_THEN `d:num->num` STRIP_ASSUME_TAC)) THEN SUBGOAL_THEN `{k | k IN 1 .. n /\ a k :num < b k} = 1..n` MP_TAC THENL [MATCH_MP_TAC CARD_SUBSET_LE THEN ASM_REWRITE_TAC[CARD_NUMSEG; ADD_SUB; FINITE_NUMSEG; SUBSET_RESTRICT] THEN SUBGOAL_THEN `kle n a b /\ n <= CARD {k | k IN 1..n /\ a(k) < b(k)}` (fun th -> REWRITE_TAC[th]) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_L THEN EXISTS_TAC `c:num->num` THEN ASM_SIMP_TAC[] THEN SUBGOAL_THEN `kle n c b /\ n <= CARD {k | k IN 1 .. n /\ c k < b k}` (fun th -> REWRITE_TAC[th]) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_R THEN EXISTS_TAC `d:num->num` THEN ASM_SIMP_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `kle n a b` MP_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(STRIP_ASSUME_TAC o GEN_REWRITE_RULE I [kle]) THEN ASM_REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_NUMSEG] THEN MATCH_MP_TAC MONO_FORALL THEN X_GEN_TAC `i:num` THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[ADD_CLAUSES; LT_REFL] THEN ASM_MESON_TAC[SUBSET; IN_NUMSEG]);;
let KSIMPLEX_EXTREMA_STRONG = 
prove (`!p n s. ksimplex p n s /\ ~(n = 0) ==> ?a b. a IN s /\ b IN s /\ ~(a = b) /\ (!x. x IN s ==> kle n a x /\ kle n x b) /\ (!i. b(i) = if 1 <= i /\ i <= n then a(i) + 1 else a(i))`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o MATCH_MP KSIMPLEX_EXTREMA) THEN REPEAT(MATCH_MP_TAC MONO_EXISTS THEN GEN_TAC) THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN DISCH_THEN SUBST_ALL_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `1`) THEN ASM_REWRITE_TAC[LE_REFL; ARITH_RULE `1 <= n <=> ~(n = 0)`] THEN ARITH_TAC);;
let KSIMPLEX_SUCCESSOR = 
prove (`!a p n s. ksimplex p n s /\ a IN s ==> (!x. x IN s ==> kle n x a) \/ (?y. y IN s /\ ?k. 1 <= k /\ k <= n /\ !j. y(j) = if j = k then a(j) + 1 else a(j))`,
REWRITE_TAC[ksimplex] THEN REPEAT STRIP_TAC THEN REWRITE_TAC[TAUT `a \/ b <=> ~a ==> b`] THEN REWRITE_TAC[NOT_FORALL_THM; NOT_IMP] THEN DISCH_TAC THEN MP_TAC(SPECL [`{x | x IN s /\ ~kle n x a}`; `n:num`] KLE_MINIMAL) THEN RULE_ASSUM_TAC(REWRITE_RULE[HAS_SIZE]) THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN ASM_SIMP_TAC[GSYM MEMBER_NOT_EMPTY; IN_ELIM_THM] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `b:num->num` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `1 <= CARD {k | k IN 1..n /\ a(k):num < b(k)}` MP_TAC THENL [MATCH_MP_TAC KLE_STRICT_SET THEN ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(DISJ_CASES_THEN MP_TAC o MATCH_MP (ARITH_RULE `1 <= n ==> (n = 1) \/ 2 <= n`)) THENL [DISCH_TAC THEN MP_TAC(HAS_SIZE_CONV `{k | k IN 1 .. n /\ a k :num < b k} HAS_SIZE 1`) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; FINITE_NUMSEG] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `k:num` THEN REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_SING; IN_NUMSEG] THEN DISCH_THEN(fun th -> CONJ_TAC THENL [MESON_TAC[th]; MP_TAC th]) THEN DISCH_THEN(fun th -> CONJ_TAC THENL [MESON_TAC[th]; MP_TAC th]) THEN MATCH_MP_TAC MONO_FORALL THEN X_GEN_TAC `i:num` THEN SUBGOAL_THEN `kle n a b` MP_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(STRIP_ASSUME_TAC o GEN_REWRITE_RULE I [kle]) THEN ASM_REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_NUMSEG] THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[ADD_CLAUSES; LT_REFL] THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[ADD_CLAUSES; LT_REFL] THEN ASM_MESON_TAC[SUBSET; IN_NUMSEG; ARITH_RULE `~(a + 1 = a)`; ARITH_RULE `a < a + 1`]; ALL_TAC] THEN DISCH_TAC THEN MP_TAC(SPECL [`n:num`; `PRE(CARD {x | x IN s /\ ~(kle n x a)})`; `{x | x IN s /\ ~(kle n x a)}`] KLE_RANGE_INDUCT) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; CARD_EQ_0; GSYM MEMBER_NOT_EMPTY; ARITH_RULE `(n = SUC(PRE n)) <=> ~(n = 0)`] THEN REPEAT(ANTS_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC]) THEN DISCH_THEN(X_CHOOSE_THEN `c:num->num` (X_CHOOSE_THEN `d:num->num` MP_TAC)) THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 (STRIP_ASSUME_TAC o REWRITE_RULE[IN_ELIM_THM]) MP_TAC)) THEN DISCH_TAC THEN MP_TAC(SPECL [`n:num`; `PRE(CARD {x | x IN s /\ kle n x a})`; `{x | x IN s /\ kle n x a}`] KLE_RANGE_INDUCT) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; CARD_EQ_0; GSYM MEMBER_NOT_EMPTY; ARITH_RULE `(n = SUC(PRE n)) <=> ~(n = 0)`] THEN REPEAT(ANTS_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[KLE_REFL]; ALL_TAC]) THEN DISCH_THEN(X_CHOOSE_THEN `e:num->num` (X_CHOOSE_THEN `f:num->num` MP_TAC)) THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 (STRIP_ASSUME_TAC o REWRITE_RULE[IN_ELIM_THM]) MP_TAC)) THEN DISCH_TAC THEN SUBGOAL_THEN `kle n e d /\ n + 1 <= CARD {k | k IN 1..n /\ e(k) < d(k)}` MP_TAC THENL [ALL_TAC; ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_THEN(K ALL_TAC) THEN DISCH_THEN(MP_TAC o CONJUNCT2) THEN REWRITE_TAC[ARITH_RULE `~(n + 1 <= x) <=> x <= n`] THEN MATCH_MP_TAC LE_TRANS THEN EXISTS_TAC `CARD(1..n)` THEN SIMP_TAC[CARD_SUBSET; SUBSET_RESTRICT; FINITE_RESTRICT; FINITE_NUMSEG] THEN REWRITE_TAC[CARD_NUMSEG; ADD_SUB; LE_REFL]] THEN SUBGOAL_THEN `(CARD {x | x IN s /\ kle n x a} - 1) + 2 + (CARD {x | x IN s /\ ~kle n x a} - 1) = n + 1` (SUBST1_TAC o SYM) THENL [MATCH_MP_TAC(ARITH_RULE `~(a = 0) /\ ~(b = 0) /\ (a + b = n + 1) ==> ((a - 1) + 2 + (b - 1) = n + 1)`) THEN ASM_SIMP_TAC[CARD_EQ_0; FINITE_RESTRICT; GSYM MEMBER_NOT_EMPTY] THEN REPEAT (CONJ_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC]) THEN FIRST_ASSUM(SUBST1_TAC o SYM o CONJUNCT2) THEN MATCH_MP_TAC CARD_UNION_EQ THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[EXTENSION; NOT_IN_EMPTY; IN_INTER; IN_UNION; IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC] THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `a:num->num` THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n e a`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_R THEN EXISTS_TAC `f:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`]; ALL_TAC] THEN W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n a d`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `b:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n b d`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_L THEN EXISTS_TAC `c:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`] THEN ASM_MESON_TAC[KLE_TRANS]);;
let KSIMPLEX_PREDECESSOR = 
prove (`!a p n s. ksimplex p n s /\ a IN s ==> (!x. x IN s ==> kle n a x) \/ (?y. y IN s /\ ?k. 1 <= k /\ k <= n /\ !j. a(j) = if j = k then y(j) + 1 else y(j))`,
REWRITE_TAC[ksimplex] THEN REPEAT STRIP_TAC THEN REWRITE_TAC[TAUT `a \/ b <=> ~a ==> b`] THEN REWRITE_TAC[NOT_FORALL_THM; NOT_IMP] THEN DISCH_TAC THEN MP_TAC(SPECL [`{x | x IN s /\ ~kle n a x}`; `n:num`] KLE_MAXIMAL) THEN RULE_ASSUM_TAC(REWRITE_RULE[HAS_SIZE]) THEN ASM_SIMP_TAC[FINITE_RESTRICT] THEN ASM_SIMP_TAC[GSYM MEMBER_NOT_EMPTY; IN_ELIM_THM] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `b:num->num` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `1 <= CARD {k | k IN 1..n /\ b(k):num < a(k)}` MP_TAC THENL [MATCH_MP_TAC KLE_STRICT_SET THEN ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(DISJ_CASES_THEN MP_TAC o MATCH_MP (ARITH_RULE `1 <= n ==> (n = 1) \/ 2 <= n`)) THENL [DISCH_TAC THEN MP_TAC(HAS_SIZE_CONV `{k | k IN 1 .. n /\ b k :num < a k} HAS_SIZE 1`) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; FINITE_NUMSEG] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `k:num` THEN REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_SING; IN_NUMSEG] THEN DISCH_THEN(fun th -> CONJ_TAC THENL [MESON_TAC[th]; MP_TAC th]) THEN DISCH_THEN(fun th -> CONJ_TAC THENL [MESON_TAC[th]; MP_TAC th]) THEN MATCH_MP_TAC MONO_FORALL THEN X_GEN_TAC `i:num` THEN SUBGOAL_THEN `kle n b a` MP_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(STRIP_ASSUME_TAC o GEN_REWRITE_RULE I [kle]) THEN ASM_REWRITE_TAC[EXTENSION; IN_ELIM_THM; IN_NUMSEG] THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[ADD_CLAUSES; LT_REFL] THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[ADD_CLAUSES; LT_REFL] THEN ASM_MESON_TAC[SUBSET; IN_NUMSEG; ARITH_RULE `~(a + 1 = a)`; ARITH_RULE `a < a + 1`]; ALL_TAC] THEN DISCH_TAC THEN MP_TAC(SPECL [`n:num`; `PRE(CARD {x | x IN s /\ ~(kle n a x)})`; `{x | x IN s /\ ~(kle n a x)}`] KLE_RANGE_INDUCT) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; CARD_EQ_0; GSYM MEMBER_NOT_EMPTY; ARITH_RULE `(n = SUC(PRE n)) <=> ~(n = 0)`] THEN REPEAT(ANTS_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC]) THEN DISCH_THEN(X_CHOOSE_THEN `d:num->num` (X_CHOOSE_THEN `c:num->num` MP_TAC)) THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 (STRIP_ASSUME_TAC o REWRITE_RULE[IN_ELIM_THM]) MP_TAC)) THEN DISCH_TAC THEN MP_TAC(SPECL [`n:num`; `PRE(CARD {x | x IN s /\ kle n a x})`; `{x | x IN s /\ kle n a x}`] KLE_RANGE_INDUCT) THEN ASM_SIMP_TAC[HAS_SIZE; FINITE_RESTRICT; CARD_EQ_0; GSYM MEMBER_NOT_EMPTY; ARITH_RULE `(n = SUC(PRE n)) <=> ~(n = 0)`] THEN REPEAT(ANTS_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[KLE_REFL]; ALL_TAC]) THEN DISCH_THEN(X_CHOOSE_THEN `f:num->num` (X_CHOOSE_THEN `e:num->num` MP_TAC)) THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 (STRIP_ASSUME_TAC o REWRITE_RULE[IN_ELIM_THM]) MP_TAC)) THEN DISCH_TAC THEN SUBGOAL_THEN `kle n d e /\ n + 1 <= CARD {k | k IN 1..n /\ d(k) < e(k)}` MP_TAC THENL [ALL_TAC; ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_THEN(K ALL_TAC) THEN DISCH_THEN(MP_TAC o CONJUNCT2) THEN REWRITE_TAC[ARITH_RULE `~(n + 1 <= x) <=> x <= n`] THEN MATCH_MP_TAC LE_TRANS THEN EXISTS_TAC `CARD(1..n)` THEN SIMP_TAC[CARD_SUBSET; SUBSET_RESTRICT; FINITE_RESTRICT; FINITE_NUMSEG] THEN REWRITE_TAC[CARD_NUMSEG; ADD_SUB; LE_REFL]] THEN SUBGOAL_THEN `((CARD {x | x IN s /\ ~kle n a x} - 1) + 2) + (CARD {x | x IN s /\ kle n a x} - 1) = n + 1` (SUBST1_TAC o SYM) THENL [MATCH_MP_TAC(ARITH_RULE `~(a = 0) /\ ~(b = 0) /\ (a + b = n + 1) ==> (((b - 1) + 2) + (a - 1) = n + 1)`) THEN ASM_SIMP_TAC[CARD_EQ_0; FINITE_RESTRICT; GSYM MEMBER_NOT_EMPTY] THEN REPEAT (CONJ_TAC THENL [REWRITE_TAC[IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC]) THEN FIRST_ASSUM(SUBST1_TAC o SYM o CONJUNCT2) THEN MATCH_MP_TAC CARD_UNION_EQ THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[EXTENSION; NOT_IN_EMPTY; IN_INTER; IN_UNION; IN_ELIM_THM] THEN ASM_MESON_TAC[]; ALL_TAC] THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `a:num->num` THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ALL_TAC; W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n a e`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_L THEN EXISTS_TAC `f:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`]] THEN W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n d a`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE THEN EXISTS_TAC `b:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_TRANS]; ALL_TAC] THEN W(fun(asl,w) -> SUBGOAL_THEN(mk_conj(`kle n d b`,w)) (fun th -> REWRITE_TAC[th])) THEN MATCH_MP_TAC KLE_RANGE_COMBINE_R THEN EXISTS_TAC `c:num->num` THEN ASM_REWRITE_TAC[ARITH_RULE `k - 1 = PRE k`] THEN ASM_MESON_TAC[KLE_TRANS]);;
(* ------------------------------------------------------------------------- *) (* The lemmas about simplices that we need. *) (* ------------------------------------------------------------------------- *)
let FINITE_SIMPLICES = 
prove (`!p n. FINITE {s | ksimplex p n s}`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC FINITE_SUBSET THEN EXISTS_TAC `{s | s SUBSET {f | (!i. i IN 1..n ==> f(i) IN 0..p) /\ (!i. ~(i IN 1..n) ==> (f(i) = p))}}` THEN ASM_SIMP_TAC[FINITE_POWERSET; FINITE_FUNSPACE; FINITE_NUMSEG] THEN REWRITE_TAC[SUBSET; IN_ELIM_THM; ksimplex] THEN ASM_SIMP_TAC[IN_NUMSEG; LE_0]);;
let SIMPLEX_TOP_FACE = 
prove (`0 < p /\ (!x. x IN f ==> (x(n + 1) = p)) ==> ((?s a. ksimplex p (n + 1) s /\ a IN s /\ (f = s DELETE a)) <=> ksimplex p n f)`,
REPEAT STRIP_TAC THEN EQ_TAC THENL [REWRITE_TAC[ksimplex; LEFT_IMP_EXISTS_THM] THEN REPEAT GEN_TAC THEN STRIP_TAC THEN ASM_REWRITE_TAC[IN_DELETE] THEN REPEAT CONJ_TAC THENL [UNDISCH_TAC `(s:(num->num)->bool) HAS_SIZE ((n + 1) + 1)` THEN SIMP_TAC[HAS_SIZE; CARD_DELETE; FINITE_DELETE] THEN ASM_REWRITE_TAC[] THEN STRIP_TAC THEN ARITH_TAC; REPEAT STRIP_TAC THEN FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[]; GEN_TAC THEN X_GEN_TAC `j:num` THEN ONCE_REWRITE_TAC[ARITH_RULE `(1 <= j /\ j <= n) <=> (1 <= j /\ j <= n + 1) /\ ~(j = (n + 1))`] THEN ASM_MESON_TAC[IN_DELETE]; REPEAT STRIP_TAC THEN SUBGOAL_THEN `kle (n + 1) x y \/ kle (n + 1) y x` MP_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN MATCH_MP_TAC MONO_OR THEN CONJ_TAC THEN (REWRITE_TAC[kle] THEN MATCH_MP_TAC MONO_EXISTS THEN REWRITE_TAC[GSYM ADD1; NUMSEG_CLAUSES; ARITH_RULE `1 <= SUC n`] THEN X_GEN_TAC `k:num->bool` THEN SIMP_TAC[] THEN REWRITE_TAC[SUBSET; IN_INSERT] THEN ASM_CASES_TAC `(SUC n) IN k` THENL [ALL_TAC; ASM_MESON_TAC[]] THEN DISCH_THEN(MP_TAC o SPEC `n + 1` o CONJUNCT2) THEN ASM_REWRITE_TAC[GSYM ADD1] THEN ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_THEN(K ALL_TAC) THEN MATCH_MP_TAC(ARITH_RULE `(x = p) /\ (y = p) ==> ~(x = SUC y)`) THEN CONJ_TAC THEN ASM_MESON_TAC[ADD1; IN_DELETE])]; ALL_TAC] THEN DISCH_TAC THEN FIRST_ASSUM(MP_TAC o MATCH_MP KSIMPLEX_EXTREMA) THEN DISCH_THEN(X_CHOOSE_THEN `a:num->num` (X_CHOOSE_THEN `b:num->num` STRIP_ASSUME_TAC)) THEN ABBREV_TAC `c = \i. if i = (n + 1) then p - 1 else a(i)` THEN MAP_EVERY EXISTS_TAC [`(c:num->num) INSERT f`; `c:num->num`] THEN REWRITE_TAC[IN_INSERT; DELETE_INSERT] THEN SUBGOAL_THEN `~((c:num->num) IN f)` ASSUME_TAC THENL [DISCH_TAC THEN UNDISCH_TAC `!x:num->num. x IN f ==> (x (n + 1) = p)` THEN DISCH_THEN(MP_TAC o SPEC `c:num->num`) THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "c" THEN REWRITE_TAC[] THEN UNDISCH_TAC `0 < p` THEN ARITH_TAC; ALL_TAC] THEN CONJ_TAC THENL [ALL_TAC; UNDISCH_TAC `~((c:num->num) IN f)` THEN SET_TAC[]] THEN UNDISCH_TAC `ksimplex p n f` THEN REWRITE_TAC[ksimplex] THEN REPEAT(MATCH_MP_TAC MONO_AND THEN CONJ_TAC) THENL [SIMP_TAC[HAS_SIZE; FINITE_RULES; CARD_CLAUSES] THEN ASM_REWRITE_TAC[ADD1]; EXPAND_TAC "c" THEN REWRITE_TAC[IN_INSERT] THEN SIMP_TAC[TAUT `(a \/ b ==> c) <=> (a ==> c) /\ (b ==> c)`] THEN ASM_MESON_TAC[ARITH_RULE `p - 1 <= p`]; EXPAND_TAC "c" THEN REWRITE_TAC[IN_INSERT; TAUT `(a \/ b) /\ c ==> d <=> (a /\ c ==> d) /\ (b /\ c ==> d)`] THEN DISCH_TAC THEN REPEAT GEN_TAC THEN CONJ_TAC THENL [DISCH_THEN(CONJUNCTS_THEN2 SUBST1_TAC MP_TAC); ALL_TAC] THEN ASM_MESON_TAC[LE_REFL; ARITH_RULE `1 <= n + 1`; ARITH_RULE `j <= n ==> j <= n + 1`]; ALL_TAC] THEN DISCH_TAC THEN REWRITE_TAC[IN_INSERT] THEN SUBGOAL_THEN `!x. x IN f ==> kle (n + 1) c x` (fun th -> ASM_MESON_TAC[th; KLE_SUC; KLE_REFL]) THEN X_GEN_TAC `x:num->num` THEN DISCH_TAC THEN SUBGOAL_THEN `kle (n + 1) a x` MP_TAC THENL [ASM_MESON_TAC[KLE_SUC]; ALL_TAC] THEN EXPAND_TAC "c" THEN REWRITE_TAC[kle] THEN DISCH_THEN(X_CHOOSE_THEN `k:num->bool` STRIP_ASSUME_TAC) THEN EXISTS_TAC `(n + 1) INSERT k` THEN ASM_REWRITE_TAC[INSERT_SUBSET; IN_NUMSEG] THEN ASM_REWRITE_TAC[LE_REFL; ARITH_RULE `1 <= n + 1`] THEN X_GEN_TAC `j:num` THEN REWRITE_TAC[IN_INSERT] THEN ASM_CASES_TAC `j = n + 1` THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `~(n + 1 IN k)` (fun th -> ASM_MESON_TAC[th; ARITH_RULE `0 < p ==> (p = (p - 1) + 1)`]) THEN DISCH_TAC THEN UNDISCH_TAC `!x:num->num. x IN f ==> (x (n + 1) = p)` THEN DISCH_THEN(fun th -> MP_TAC(SPEC `x:num->num` th) THEN MP_TAC(SPEC `a:num->num` th)) THEN ASM_REWRITE_TAC[] THEN MESON_TAC[ARITH_RULE `~(p + 1 = p)`]);;
let KSIMPLEX_FIX_PLANE = 
prove (`!p q n j s a a0 a1. ksimplex p n s /\ a IN s /\ 1 <= j /\ j <= n /\ (!x. x IN (s DELETE a) ==> (x j = q)) /\ a0 IN s /\ a1 IN s /\ (!i. a1 i = (if 1 <= i /\ i <= n then a0 i + 1 else a0 i)) ==> (a = a0) \/ (a = a1)`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC(TAUT `(~a /\ ~b ==> F) ==> a \/ b`) THEN STRIP_TAC THEN UNDISCH_TAC `!x:num->num. x IN s DELETE a ==> (x j = q)` THEN DISCH_THEN(fun th -> MP_TAC(SPEC `a0:num->num` th) THEN MP_TAC(SPEC `a1:num->num` th)) THEN ASM_REWRITE_TAC[IN_DELETE] THEN ARITH_TAC);;
let KSIMPLEX_FIX_PLANE_0 = 
prove (`!p n j s a a0 a1. ksimplex p n s /\ a IN s /\ 1 <= j /\ j <= n /\ (!x. x IN (s DELETE a) ==> (x j = 0)) /\ a0 IN s /\ a1 IN s /\ (!i. a1 i = (if 1 <= i /\ i <= n then a0 i + 1 else a0 i)) ==> (a = a1)`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `(a = a0) \/ (a = a1:num->num)` MP_TAC THENL [MATCH_MP_TAC KSIMPLEX_FIX_PLANE THEN MAP_EVERY EXISTS_TAC [`p:num`; `0`; `n:num`; `j:num`; `s:(num->num)->bool`] THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN ASM_CASES_TAC `a0:num->num = a1` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(TAUT `~a ==> (a \/ b ==> b)`) THEN DISCH_THEN SUBST_ALL_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `a1:num->num`) THEN ASM_REWRITE_TAC[IN_DELETE] THEN ARITH_TAC);;
let KSIMPLEX_FIX_PLANE_P = 
prove (`!p n j s a a0 a1. ksimplex p n s /\ a IN s /\ 1 <= j /\ j <= n /\ (!x. x IN (s DELETE a) ==> (x j = p)) /\ a0 IN s /\ a1 IN s /\ (!i. a1 i = (if 1 <= i /\ i <= n then a0 i + 1 else a0 i)) ==> (a = a0)`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `(a = a0) \/ (a = a1:num->num)` MP_TAC THENL [MATCH_MP_TAC KSIMPLEX_FIX_PLANE THEN MAP_EVERY EXISTS_TAC [`p:num`; `p:num`; `n:num`; `j:num`; `s:(num->num)->bool`] THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN ASM_CASES_TAC `a0:num->num = a1` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(TAUT `~b ==> (a \/ b ==> a)`) THEN DISCH_THEN SUBST_ALL_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `a0:num->num`) THEN ASM_REWRITE_TAC[IN_DELETE] THEN DISCH_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [ksimplex]) THEN DISCH_THEN(MP_TAC o SPEC `a1:num->num` o CONJUNCT1 o CONJUNCT2) THEN DISCH_THEN(MP_TAC o SPEC `j:num`) THEN ASM_REWRITE_TAC[] THEN ARITH_TAC);;
let KSIMPLEX_REPLACE_0 = 
prove (`ksimplex p n s /\ a IN s /\ ~(n = 0) /\ (?j. 1 <= j /\ j <= n /\ !x. x IN (s DELETE a) ==> (x j = 0)) ==> (CARD {s' | ksimplex p n s' /\ ?b. b IN s' /\ (s' DELETE b = s DELETE a)} = 1)`,
let lemma = prove
   (`!a a'. (s' DELETE a' = s DELETE a) /\ (a' = a) /\ a' IN s' /\ a IN s
            ==> (s' = s)`,
    SET_TAC[]) in
  REPEAT STRIP_TAC THEN MATCH_MP_TAC HAS_SIZE_CARD THEN
  REWRITE_TAC[HAS_SIZE_1_EXISTS] THEN REWRITE_TAC[IN_ELIM_THM] THEN
  SUBGOAL_THEN
   `!s' a'. ksimplex p n s' /\ a' IN s' /\ (s' DELETE a' = s DELETE a)
            ==> (s' = s)`
   (fun th -> ASM_MESON_TAC[th]) THEN
  REPEAT STRIP_TAC THEN
  MP_TAC(SPECL [`p:num`; `n:num`; `s:(num->num)->bool`]
       KSIMPLEX_EXTREMA_STRONG) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(X_CHOOSE_THEN `a0:num->num` (X_CHOOSE_THEN `a1:num->num`
    STRIP_ASSUME_TAC)) THEN
  MP_TAC(SPECL [`p:num`; `n:num`; `s':(num->num)->bool`]
       KSIMPLEX_EXTREMA_STRONG) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(X_CHOOSE_THEN `b0:num->num` (X_CHOOSE_THEN `b1:num->num`
    STRIP_ASSUME_TAC)) THEN
  SUBGOAL_THEN `a:num->num = a1` SUBST_ALL_TAC THENL
   [MATCH_MP_TAC KSIMPLEX_FIX_PLANE_0 THEN MAP_EVERY EXISTS_TAC
     [`p:num`; `n:num`; `j:num`; `s:(num->num)->bool`; `a0:num->num`] THEN
    ASM_REWRITE_TAC[];
    ALL_TAC] THEN
  SUBGOAL_THEN `a':num->num = b1` SUBST_ALL_TAC THENL
   [MATCH_MP_TAC KSIMPLEX_FIX_PLANE_0 THEN MAP_EVERY EXISTS_TAC
     [`p:num`; `n:num`; `j:num`; `s':(num->num)->bool`; `b0:num->num`] THEN
    ASM_REWRITE_TAC[];
    ALL_TAC] THEN
  MATCH_MP_TAC lemma THEN
  MAP_EVERY EXISTS_TAC [`a1:num->num`; `b1:num->num`] THEN
  ASM_REWRITE_TAC[] THEN
  SUBGOAL_THEN `b0:num->num = a0` MP_TAC THENL
   [ONCE_REWRITE_TAC[GSYM KLE_ANTISYM] THEN ASM_MESON_TAC[IN_DELETE];
    ASM_REWRITE_TAC[FUN_EQ_THM] THEN MESON_TAC[]]);;
let KSIMPLEX_REPLACE_1 = 
prove (`ksimplex p n s /\ a IN s /\ ~(n = 0) /\ (?j. 1 <= j /\ j <= n /\ !x. x IN (s DELETE a) ==> (x j = p)) ==> (CARD {s' | ksimplex p n s' /\ ?b. b IN s' /\ (s' DELETE b = s DELETE a)} = 1)`,
let lemma = prove
   (`!a a'. (s' DELETE a' = s DELETE a) /\ (a' = a) /\ a' IN s' /\ a IN s
            ==> (s' = s)`,
    SET_TAC[]) in
  REPEAT STRIP_TAC THEN MATCH_MP_TAC HAS_SIZE_CARD THEN
  REWRITE_TAC[HAS_SIZE_1_EXISTS] THEN REWRITE_TAC[IN_ELIM_THM] THEN
  SUBGOAL_THEN
   `!s' a'. ksimplex p n s' /\ a' IN s' /\ (s' DELETE a' = s DELETE a)
            ==> (s' = s)`
   (fun th -> ASM_MESON_TAC[th]) THEN
  REPEAT STRIP_TAC THEN
  MP_TAC(SPECL [`p:num`; `n:num`; `s:(num->num)->bool`]
       KSIMPLEX_EXTREMA_STRONG) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(X_CHOOSE_THEN `a0:num->num` (X_CHOOSE_THEN `a1:num->num`
    STRIP_ASSUME_TAC)) THEN
  MP_TAC(SPECL [`p:num`; `n:num`; `s':(num->num)->bool`]
       KSIMPLEX_EXTREMA_STRONG) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(X_CHOOSE_THEN `b0:num->num` (X_CHOOSE_THEN `b1:num->num`
    STRIP_ASSUME_TAC)) THEN
  SUBGOAL_THEN `a:num->num = a0` SUBST_ALL_TAC THENL
   [MATCH_MP_TAC KSIMPLEX_FIX_PLANE_P THEN MAP_EVERY EXISTS_TAC
     [`p:num`; `n:num`; `j:num`; `s:(num->num)->bool`; `a1:num->num`] THEN
    ASM_REWRITE_TAC[];
    ALL_TAC] THEN
  SUBGOAL_THEN `a':num->num = b0` SUBST_ALL_TAC THENL
   [MATCH_MP_TAC KSIMPLEX_FIX_PLANE_P THEN MAP_EVERY EXISTS_TAC
     [`p:num`; `n:num`; `j:num`; `s':(num->num)->bool`; `b1:num->num`] THEN
    ASM_REWRITE_TAC[];
    ALL_TAC] THEN
  MATCH_MP_TAC lemma THEN
  MAP_EVERY EXISTS_TAC [`a0:num->num`; `b0:num->num`] THEN
  ASM_REWRITE_TAC[] THEN
  SUBGOAL_THEN `b1:num->num = a1` MP_TAC THENL
   [ONCE_REWRITE_TAC[GSYM KLE_ANTISYM] THEN ASM_MESON_TAC[IN_DELETE];
    ASM_REWRITE_TAC[FUN_EQ_THM] THEN MATCH_MP_TAC MONO_FORALL THEN
    MESON_TAC[EQ_ADD_RCANCEL]]);;
let KSIMPLEX_REPLACE_2 = 
prove (`ksimplex p n s /\ a IN s /\ ~(n = 0) /\ ~(?j. 1 <= j /\ j <= n /\ !x. x IN (s DELETE a) ==> (x j = 0)) /\ ~(?j. 1 <= j /\ j <= n /\ !x. x IN (s DELETE a) ==> (x j = p)) ==> (CARD {s' | ksimplex p n s' /\ ?b. b IN s' /\ (s' DELETE b = s DELETE a)} = 2)`,
let lemma = prove
   (`!a a'. (s' DELETE a' = s DELETE a) /\ (a' = a) /\ a' IN s' /\ a IN s
            ==> (s' = s)`,
    SET_TAC[])
  and lemma_1 = prove
   (`a IN s /\ ~(b = a) ==> ~(s = b INSERT (s DELETE a))`,
    SET_TAC[]) in
  REPEAT STRIP_TAC THEN
  MP_TAC(SPECL [`p:num`; `n:num`; `s:(num->num)->bool`]
       KSIMPLEX_EXTREMA_STRONG) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(X_CHOOSE_THEN `a0:num->num` (X_CHOOSE_THEN `a1:num->num`
   STRIP_ASSUME_TAC)) THEN
  ASM_CASES_TAC `a:num->num = a0` THENL
   [FIRST_X_ASSUM SUBST_ALL_TAC THEN
    MP_TAC(SPECL [`a0:num->num`; `p:num`; `n:num`; `s:(num->num)->bool`]
                 KSIMPLEX_SUCCESSOR) THEN
    ASM_REWRITE_TAC[] THEN
    MATCH_MP_TAC(TAUT `~a /\ (b ==> c) ==> a \/ b ==> c`) THEN CONJ_TAC THENL
     [DISCH_THEN(MP_TAC o SPEC `a1:num->num`) THEN ASM_REWRITE_TAC[] THEN
      DISCH_THEN(MP_TAC o SPEC `1` o MATCH_MP KLE_IMP_POINTWISE) THEN
      ASM_REWRITE_TAC[ARITH_RULE `1 <= n <=> ~(n = 0)`; ARITH] THEN ARITH_TAC;
      ALL_TAC] THEN
    DISCH_THEN(X_CHOOSE_THEN `a2:num->num`
     (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN
    DISCH_THEN(X_CHOOSE_THEN `k:num` STRIP_ASSUME_TAC) THEN
    ABBREV_TAC `a3 = \j:num. if j = k then a1 j + 1 else a1 j` THEN
    FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [FUN_EQ_THM]) THEN
    REWRITE_TAC[] THEN DISCH_THEN(STRIP_ASSUME_TAC o GSYM) THEN
    MATCH_MP_TAC HAS_SIZE_CARD THEN CONV_TAC HAS_SIZE_CONV THEN
    MAP_EVERY EXISTS_TAC
     [`s:(num->num)->bool`; `a3 INSERT (s DELETE (a0:num->num))`] THEN
    SUBGOAL_THEN `~((a3:num->num) IN s)` ASSUME_TAC THENL
     [DISCH_TAC THEN SUBGOAL_THEN `kle n a3 a1` MP_TAC THENL
       [ASM_MESON_TAC[]; ALL_TAC] THEN
      DISCH_THEN(MP_TAC o SPEC `k:num` o MATCH_MP KLE_IMP_POINTWISE) THEN
      ASM_REWRITE_TAC[LE_REFL] THEN ARITH_TAC;
      ALL_TAC] THEN
    SUBGOAL_THEN `~(a3:num->num = a0) /\ ~(a3 = a1)` STRIP_ASSUME_TAC THENL
     [ASM_MESON_TAC[]; ALL_TAC] THEN
    SUBGOAL_THEN `~(a2:num->num = a0)` ASSUME_TAC THENL
     [ASM_REWRITE_TAC[FUN_EQ_THM] THEN MESON_TAC[ARITH_RULE `~(x + 1 = x)`];
      ALL_TAC] THEN
    CONJ_TAC THENL [MATCH_MP_TAC lemma_1 THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN
    SUBGOAL_THEN `!x. x IN (s DELETE a0) ==> kle n a2 x` ASSUME_TAC THENL
     [GEN_TAC THEN REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN
      SUBGOAL_THEN `kle n a2 x \/ kle n x a2` MP_TAC THENL
       [ASM_MESON_TAC[ksimplex]; ALL_TAC] THEN
      MATCH_MP_TAC(TAUT `(~b ==> ~a) ==> b \/ a ==> b`) THEN
      DISCH_TAC THEN SUBGOAL_THEN `kle n a0 x` MP_TAC THENL
       [ASM_MESON_TAC[]; ALL_TAC] THEN REPEAT STRIP_TAC THEN
      SUBGOAL_THEN `(x:num->num = a0) \/ (x = a2)`
       (fun th -> ASM_MESON_TAC[KLE_REFL; th]) THEN
      MATCH_MP_TAC KLE_ADJACENT THEN
      EXISTS_TAC `k:num` THEN ASM_REWRITE_TAC[];
      ALL_TAC] THEN
    SUBGOAL_THEN `ksimplex p n (a3 INSERT (s DELETE a0))` ASSUME_TAC THENL
     [MP_TAC(ASSUME `ksimplex p n s`) THEN REWRITE_TAC[ksimplex] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [SIMP_TAC[HAS_SIZE; FINITE_INSERT; FINITE_DELETE; CARD_CLAUSES;
                 CARD_DELETE] THEN
        ASM_REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN ARITH_TAC;
        ALL_TAC] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [DISCH_TAC THEN REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
        SUBGOAL_THEN `!j. (a3:num->num) j <= p`
         (fun th -> ASM_MESON_TAC[th]) THEN
        X_GEN_TAC `j:num` THEN ONCE_ASM_REWRITE_TAC[] THEN COND_CASES_TAC THENL
         [ALL_TAC; ASM_MESON_TAC[]] THEN
        FIRST_X_ASSUM SUBST_ALL_TAC THEN
        UNDISCH_TAC
         `~(?j. 1 <= j /\ j <= n /\
                (!x. x IN s DELETE a0 ==> (x j = (p:num))))` THEN
        REWRITE_TAC[NOT_EXISTS_THM] THEN DISCH_THEN(MP_TAC o SPEC `k:num`) THEN
        REWRITE_TAC[ASSUME `1 <= k`; ASSUME `k:num <= n`; NOT_FORALL_THM] THEN
        DISCH_THEN(X_CHOOSE_THEN `a4:num->num` MP_TAC) THEN
        REWRITE_TAC[IN_DELETE; NOT_IMP] THEN STRIP_TAC THEN
        UNDISCH_TAC `!x. x IN s DELETE a0 ==> kle n a2 x` THEN
        DISCH_THEN(MP_TAC o SPEC `a4:num->num`) THEN
        ASM_REWRITE_TAC[IN_DELETE] THEN
        DISCH_THEN(MP_TAC o MATCH_MP KLE_IMP_POINTWISE) THEN
        ASM_REWRITE_TAC[] THEN DISCH_THEN(MP_TAC o SPEC `k:num`) THEN
        ASM_REWRITE_TAC[] THEN
        UNDISCH_TAC `~((a4:num->num) k = p)` THEN
        SUBGOAL_THEN `(a4:num->num) k <= p` MP_TAC THENL
         [ASM_MESON_TAC[ksimplex]; ARITH_TAC];
        ALL_TAC] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [REWRITE_TAC[IN_INSERT; IN_DELETE] THEN REPEAT STRIP_TAC THENL
         [ALL_TAC; ASM_MESON_TAC[]] THEN
        FIRST_X_ASSUM SUBST_ALL_TAC THEN
        ONCE_ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[];
        ALL_TAC] THEN
      DISCH_TAC THEN REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
      SUBGOAL_THEN `!x. x IN s /\ ~(x = a0) ==> kle n x a3`
       (fun th -> ASM_MESON_TAC[th; KLE_REFL]) THEN
      X_GEN_TAC `x:num->num` THEN STRIP_TAC THEN
      SUBGOAL_THEN `kle n a2 x /\ kle n x a1` MP_TAC THENL
       [ASM_MESON_TAC[IN_DELETE]; ALL_TAC] THEN
      REWRITE_TAC[IMP_CONJ] THEN
      DISCH_THEN(MP_TAC o SPEC `k:num` o MATCH_MP KLE_IMP_POINTWISE) THEN
      DISCH_TAC THEN REWRITE_TAC[kle] THEN
      DISCH_THEN(X_CHOOSE_THEN `kk:num->bool` STRIP_ASSUME_TAC) THEN
      EXISTS_TAC `(k:num) INSERT kk` THEN
      REWRITE_TAC[INSERT_SUBSET; IN_NUMSEG] THEN
      CONJ_TAC THENL [ASM_REWRITE_TAC[]; ALL_TAC] THEN
      X_GEN_TAC `j:num` THEN
      FIRST_ASSUM(fun th -> GEN_REWRITE_TAC LAND_CONV [th]) THEN
      REWRITE_TAC[IN_INSERT] THEN ASM_CASES_TAC `j:num = k` THENL
       [ALL_TAC; ASM_MESON_TAC[]] THEN
      FIRST_X_ASSUM SUBST_ALL_TAC THEN REWRITE_TAC[] THEN
      FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (ARITH_RULE
       `a2 <= x ==> !a0. x <= a1 /\ (a1 = a0 + 1) /\ (a2 = a0 + 1)
             ==> (a1 + 1 = x + 1)`)) THEN
      EXISTS_TAC `(a0:num->num) k` THEN
      ASM_MESON_TAC[KLE_IMP_POINTWISE];
      ALL_TAC] THEN
    GEN_REWRITE_TAC I [EXTENSION] THEN
    REWRITE_TAC[IN_ELIM_THM; IN_INSERT; NOT_IN_EMPTY] THEN
    X_GEN_TAC `s':(num->num)->bool` THEN EQ_TAC THENL
     [ALL_TAC;
      DISCH_THEN(DISJ_CASES_THEN SUBST_ALL_TAC) THENL
       [ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[]; ALL_TAC] THEN
      ASM_REWRITE_TAC[] THEN EXISTS_TAC `a3:num->num` THEN
      REWRITE_TAC[IN_INSERT; DELETE_INSERT] THEN
      UNDISCH_TAC `~((a3:num->num) IN s)` THEN SET_TAC[]] THEN
    DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN
    DISCH_THEN(X_CHOOSE_THEN `a':num->num` STRIP_ASSUME_TAC) THEN
    MP_TAC(SPECL [`p:num`; `n:num`; `s':(num->num)->bool`]
                 KSIMPLEX_EXTREMA_STRONG) THEN ASM_REWRITE_TAC[] THEN
    DISCH_THEN(X_CHOOSE_THEN `a_min:num->num` (X_CHOOSE_THEN `a_max:num->num`
      STRIP_ASSUME_TAC)) THEN
    SUBGOAL_THEN `(a':num->num = a_min) \/ (a' = a_max)` MP_TAC THENL
     [MATCH_MP_TAC KSIMPLEX_FIX_PLANE THEN MAP_EVERY EXISTS_TAC
       [`p:num`; `(a2:num->num) k`; `n:num`;
        `k:num`; `s':(num->num)->bool`] THEN
      REPEAT CONJ_TAC THEN TRY(FIRST_ASSUM MATCH_ACCEPT_TAC) THEN
      X_GEN_TAC `x:num->num` THEN ASM_REWRITE_TAC[] THEN DISCH_TAC THEN
      SUBGOAL_THEN `kle n a2 x /\ kle n x a1` MP_TAC THENL
       [ASM_MESON_TAC[IN_DELETE]; ALL_TAC] THEN
      DISCH_THEN(CONJUNCTS_THEN (MP_TAC o SPEC `k:num` o MATCH_MP
        KLE_IMP_POINTWISE)) THEN
      ASM_REWRITE_TAC[] THEN ARITH_TAC;
      ALL_TAC] THEN
    DISCH_THEN(DISJ_CASES_THEN SUBST_ALL_TAC) THENL
     [DISJ1_TAC THEN MATCH_MP_TAC lemma THEN
      MAP_EVERY EXISTS_TAC [`a0:num->num`; `a_min:num->num`] THEN
      ASM_REWRITE_TAC[] THEN
      SUBGOAL_THEN `a_max:num->num = a1` MP_TAC THENL
       [SUBGOAL_THEN `a1:num->num IN (s' DELETE a_min) /\
                      a_max:num->num IN (s DELETE a0)`
        MP_TAC THENL
         [ASM_MESON_TAC[IN_DELETE]; ASM_MESON_TAC[KLE_ANTISYM; IN_DELETE]];
        ALL_TAC] THEN
      ASM_REWRITE_TAC[FUN_EQ_THM] THEN MATCH_MP_TAC MONO_FORALL THEN
      MESON_TAC[EQ_ADD_RCANCEL];
      DISJ2_TAC THEN MATCH_MP_TAC lemma THEN
      MAP_EVERY EXISTS_TAC [`a3:num->num`; `a_max:num->num`] THEN
      ASM_REWRITE_TAC[IN_INSERT] THEN CONJ_TAC THENL
       [UNDISCH_TAC `~(a3:num->num IN s)` THEN SET_TAC[]; ALL_TAC] THEN
      SUBGOAL_THEN `a_min:num->num = a2` MP_TAC THENL
       [SUBGOAL_THEN `a2:num->num IN (s' DELETE a_max) /\
                      a_min:num->num IN (s DELETE a0)`
        MP_TAC THENL
         [ASM_MESON_TAC[IN_DELETE]; ASM_MESON_TAC[KLE_ANTISYM; IN_DELETE]];
        ALL_TAC] THEN
      ASM_REWRITE_TAC[FUN_EQ_THM] THEN MATCH_MP_TAC MONO_FORALL THEN
      MESON_TAC[EQ_ADD_RCANCEL]];
    ALL_TAC] THEN
  ASM_CASES_TAC `a:num->num = a1` THENL
   [FIRST_X_ASSUM SUBST_ALL_TAC THEN
    MP_TAC(SPECL [`a1:num->num`; `p:num`; `n:num`; `s:(num->num)->bool`]
                 KSIMPLEX_PREDECESSOR) THEN
    ANTS_TAC THENL [ASM_REWRITE_TAC[]; ALL_TAC] THEN
    MATCH_MP_TAC(TAUT `~a /\ (b ==> c) ==> a \/ b ==> c`) THEN CONJ_TAC THENL
     [DISCH_THEN(MP_TAC o SPEC `a0:num->num`) THEN ASM_REWRITE_TAC[] THEN
      DISCH_THEN(MP_TAC o SPEC `1` o MATCH_MP KLE_IMP_POINTWISE) THEN
      ASM_REWRITE_TAC[ARITH_RULE `1 <= n <=> ~(n = 0)`; ARITH] THEN ARITH_TAC;
      ALL_TAC] THEN
    DISCH_THEN(X_CHOOSE_THEN `a2:num->num`
     (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN
    DISCH_THEN(X_CHOOSE_THEN `k:num` STRIP_ASSUME_TAC) THEN
    SUBGOAL_THEN `!x. x IN (s DELETE a1) ==> kle n x a2` ASSUME_TAC THENL
     [GEN_TAC THEN REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN
      SUBGOAL_THEN `kle n a2 x \/ kle n x a2` MP_TAC THENL
       [ASM_MESON_TAC[ksimplex]; ALL_TAC] THEN
      MATCH_MP_TAC(TAUT `(~b ==> ~a) ==> a \/ b ==> b`) THEN
      DISCH_TAC THEN SUBGOAL_THEN `kle n x a1` MP_TAC THENL
       [ASM_MESON_TAC[]; ALL_TAC] THEN REPEAT STRIP_TAC THEN
      SUBGOAL_THEN `(x:num->num = a2) \/ (x = a1)`
       (fun th -> ASM_MESON_TAC[KLE_REFL; th]) THEN
      MATCH_MP_TAC KLE_ADJACENT THEN EXISTS_TAC `k:num` THEN
      REPEAT CONJ_TAC THEN FIRST_X_ASSUM MATCH_ACCEPT_TAC;
      ALL_TAC] THEN
    SUBGOAL_THEN `~(a2:num->num = a1)` ASSUME_TAC THENL
     [REWRITE_TAC[FUN_EQ_THM] THEN ASM_MESON_TAC[ARITH_RULE `~(x + 1 = x)`];
      ALL_TAC] THEN
    ABBREV_TAC `a3 = \j:num. if j = k then a0 j - 1 else a0 j` THEN
    SUBGOAL_THEN `!j:num. a0(j) = if j = k then a3(j) + 1 else a3 j`
    ASSUME_TAC THENL
     [X_GEN_TAC `j:num` THEN EXPAND_TAC "a3" THEN REWRITE_TAC[] THEN
      COND_CASES_TAC THEN
      REWRITE_TAC[ARITH_RULE `(a = a - 1 + 1) <=> ~(a = 0)`] THEN
      FIRST_X_ASSUM SUBST_ALL_TAC THEN DISCH_TAC THEN
      UNDISCH_TAC `!j:num. a1 j = (if j = k then a2 j + 1 else a2 j)` THEN
      DISCH_THEN(MP_TAC o SPEC `k:num`) THEN ASM_REWRITE_TAC[] THEN
      REWRITE_TAC[ARITH_RULE `(0 + 1 = x + 1) <=> (x = 0)`] THEN DISCH_TAC THEN
      UNDISCH_TAC
       `~(?j. 1 <= j /\ j <= n /\ (!x. x IN s DELETE a1 ==> (x j = 0)))` THEN
      REWRITE_TAC[NOT_EXISTS_THM] THEN EXISTS_TAC `k:num` THEN
      ASM_MESON_TAC[KLE_IMP_POINTWISE; LE];
      ALL_TAC] THEN
    SUBGOAL_THEN `~(kle n a0 a3)` ASSUME_TAC THENL
     [ASM_MESON_TAC[KLE_IMP_POINTWISE; ARITH_RULE `~(a + 1 <= a)`];
      ALL_TAC] THEN
    SUBGOAL_THEN `~(a3:num->num IN s)` ASSUME_TAC THENL
     [ASM_MESON_TAC[]; ALL_TAC] THEN
    SUBGOAL_THEN `kle n a3 a2` ASSUME_TAC THENL
     [SUBGOAL_THEN `kle n a0 a1` MP_TAC THENL
       [ASM_MESON_TAC[]; ALL_TAC] THEN
      REWRITE_TAC[kle] THEN MATCH_MP_TAC MONO_EXISTS THEN
      GEN_TAC THEN MATCH_MP_TAC MONO_AND THEN REWRITE_TAC[] THEN
      MATCH_MP_TAC MONO_FORALL THEN GEN_TAC THEN
      ONCE_REWRITE_TAC[
        ASSUME `!j:num. a0 j = (if j = k then a3 j + 1 else a3 j)`;
        ASSUME `!j:num. a1 j = (if j = k then a2 j + 1 else a2 j)`] THEN
      REPEAT(COND_CASES_TAC THEN REWRITE_TAC[]) THEN ARITH_TAC;
      ALL_TAC] THEN
    SUBGOAL_THEN `kle n a3 a0` ASSUME_TAC THENL
     [REWRITE_TAC[kle] THEN EXISTS_TAC `{k:num}` THEN
      ASM_REWRITE_TAC[SUBSET; IN_SING; IN_NUMSEG] THEN
      ASM_MESON_TAC[ADD_CLAUSES];
      ALL_TAC] THEN
    MATCH_MP_TAC HAS_SIZE_CARD THEN CONV_TAC HAS_SIZE_CONV THEN
    MAP_EVERY EXISTS_TAC
     [`s:(num->num)->bool`; `a3 INSERT (s DELETE (a1:num->num))`] THEN
    SUBGOAL_THEN `~(a3:num->num = a1) /\ ~(a3 = a0)` STRIP_ASSUME_TAC THENL
     [ASM_MESON_TAC[]; ALL_TAC] THEN
    CONJ_TAC THENL [MATCH_MP_TAC lemma_1 THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN
    SUBGOAL_THEN `ksimplex p n (a3 INSERT (s DELETE a1))` ASSUME_TAC THENL
     [MP_TAC(ASSUME `ksimplex p n s`) THEN REWRITE_TAC[ksimplex] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [SIMP_TAC[HAS_SIZE; FINITE_INSERT; FINITE_DELETE; CARD_CLAUSES;
                 CARD_DELETE] THEN
        ASM_REWRITE_TAC[IN_DELETE] THEN STRIP_TAC THEN ARITH_TAC;
        ALL_TAC] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [DISCH_TAC THEN REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
        SUBGOAL_THEN `!j. (a3:num->num) j <= p`
         (fun th -> ASM_MESON_TAC[th]) THEN
        X_GEN_TAC `j:num` THEN
        FIRST_X_ASSUM(MP_TAC o SPECL [`a0:num->num`; `j:num`]) THEN
        ASM_REWRITE_TAC[] THEN COND_CASES_TAC THEN ARITH_TAC;
        ALL_TAC] THEN
      MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
       [REWRITE_TAC[IN_INSERT; IN_DELETE] THEN REPEAT STRIP_TAC THENL
         [ALL_TAC; ASM_MESON_TAC[]] THEN
        FIRST_X_ASSUM SUBST_ALL_TAC THEN
        EXPAND_TAC "a3" THEN REWRITE_TAC[] THEN ASM_MESON_TAC[];
        ALL_TAC] THEN
      DISCH_TAC THEN REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
      SUBGOAL_THEN `!x. x IN s /\ ~(x = a1) ==> kle n a3 x`
       (fun th -> ASM_MESON_TAC[th; KLE_REFL]) THEN
      X_GEN_TAC `x:num->num` THEN STRIP_TAC THEN
      MATCH_MP_TAC KLE_BETWEEN_L THEN
      MAP_EVERY EXISTS_TAC [`a0:num->num`; `a2:num->num`] THEN
      ASM_MESON_TAC[IN_DELETE];
      ALL_TAC] THEN
    GEN_REWRITE_TAC I [EXTENSION] THEN
    REWRITE_TAC[IN_ELIM_THM; IN_INSERT; NOT_IN_EMPTY] THEN
    X_GEN_TAC `s':(num->num)->bool` THEN EQ_TAC THENL
     [ALL_TAC;
      DISCH_THEN(DISJ_CASES_THEN SUBST_ALL_TAC) THENL
       [ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[]; ALL_TAC] THEN
      ASM_REWRITE_TAC[] THEN EXISTS_TAC `a3:num->num` THEN
      REWRITE_TAC[IN_INSERT; DELETE_INSERT] THEN
      UNDISCH_TAC `~((a3:num->num) IN s)` THEN SET_TAC[]] THEN
    DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN
    DISCH_THEN(X_CHOOSE_THEN `a':num->num` STRIP_ASSUME_TAC) THEN
    MP_TAC(SPECL [`p:num`; `n:num`; `s':(num->num)->bool`]
                 KSIMPLEX_EXTREMA_STRONG) THEN ASM_REWRITE_TAC[] THEN
    DISCH_THEN(X_CHOOSE_THEN `a_min:num->num` (X_CHOOSE_THEN `a_max:num->num`
      STRIP_ASSUME_TAC)) THEN
    SUBGOAL_THEN `(a':num->num = a_min) \/ (a' = a_max)` MP_TAC THENL
     [MATCH_MP_TAC KSIMPLEX_FIX_PLANE THEN MAP_EVERY EXISTS_TAC
       [`p:num`; `(a2:num->num) k`; `n:num`;
        `k:num`; `s':(num->num)->bool`] THEN
      REPEAT CONJ_TAC THEN TRY(FIRST_ASSUM MATCH_ACCEPT_TAC) THEN
      X_GEN_TAC `x:num->num` THEN ASM_REWRITE_TAC[] THEN DISCH_TAC THEN
      SUBGOAL_THEN `kle n a0 x /\ kle n x a2` MP_TAC THENL
       [ASM_MESON_TAC[IN_DELETE]; ALL_TAC] THEN
      DISCH_THEN(CONJUNCTS_THEN (MP_TAC o SPEC `k:num` o MATCH_MP
        KLE_IMP_POINTWISE)) THEN
      SUBGOAL_THEN `(a2:num->num) k <= a0 k`
        (fun th -> MP_TAC th THEN ARITH_TAC) THEN
      UNDISCH_TAC `!j:num. a1 j = (if j = k then a2 j + 1 else a2 j)` THEN
      DISCH_THEN(MP_TAC o SPEC `k:num`) THEN ASM_REWRITE_TAC[] THEN ARITH_TAC;
      ALL_TAC] THEN
    DISCH_THEN(DISJ_CASES_THEN SUBST_ALL_TAC) THENL
     [DISJ2_TAC THEN MATCH_MP_TAC lemma THEN
      MAP_EVERY EXISTS_TAC [`a3:num->num`; `a_min:num->num`] THEN
      ASM_REWRITE_TAC[IN_INSERT] THEN CONJ_TAC THENL
       [UNDISCH_TAC `~(a3:num->num IN s)` THEN SET_TAC[]; ALL_TAC] THEN
      SUBGOAL_THEN `a_max:num->num = a2` MP_TAC THENL
       [SUBGOAL_THEN `a2:num->num IN (s' DELETE a_min) /\
                      a_max:num->num IN (s DELETE a1)`
        MP_TAC THENL
         [ASM_MESON_TAC[IN_DELETE]; ASM_MESON_TAC[KLE_ANTISYM; IN_DELETE]];
        ALL_TAC] THEN
      SUBGOAL_THEN
       `!j. a2 j = if 1 <= j /\ j <= n then a3 j + 1 else a3 j`
       (fun th -> ASM_REWRITE_TAC[th; FUN_EQ_THM])
      THENL
       [ALL_TAC;
        MATCH_MP_TAC MONO_FORALL THEN MESON_TAC[EQ_ADD_RCANCEL]] THEN
      UNDISCH_TAC `!j:num. a1 j = (if j = k then a2 j + 1 else a2 j)` THEN
      ASM_REWRITE_TAC[] THEN MATCH_MP_TAC MONO_FORALL THEN
      MESON_TAC[EQ_ADD_RCANCEL];
      DISJ1_TAC THEN MATCH_MP_TAC lemma THEN
      MAP_EVERY EXISTS_TAC [`a1:num->num`; `a_max:num->num`] THEN
      REPEAT CONJ_TAC THEN TRY(FIRST_ASSUM ACCEPT_TAC) THEN
      SUBGOAL_THEN `a_min:num->num = a0` MP_TAC THENL
       [SUBGOAL_THEN `a0:num->num IN (s' DELETE a_max) /\
                      a_min:num->num IN (s DELETE a1)`
        MP_TAC THENL
         [ASM_MESON_TAC[IN_DELETE]; ASM_MESON_TAC[KLE_ANTISYM; IN_DELETE]];
        ALL_TAC] THEN
      UNDISCH_THEN `!j:num. a1 j = (if j = k then a2 j + 1 else a2 j)`
       (K ALL_TAC) THEN
      ASM_REWRITE_TAC[FUN_EQ_THM] THEN MATCH_MP_TAC MONO_FORALL THEN
      MESON_TAC[EQ_ADD_RCANCEL]];
    ALL_TAC] THEN
  MP_TAC(SPECL [`a:num->num`; `p:num`; `n:num`; `s:(num->num)->bool`]
        KSIMPLEX_PREDECESSOR) THEN ASM_REWRITE_TAC[] THEN
  MATCH_MP_TAC(TAUT `~a /\ (b ==> c) ==> a \/ b ==> c`) THEN CONJ_TAC THENL
   [ASM_MESON_TAC[KLE_ANTISYM]; ALL_TAC] THEN
  DISCH_THEN(X_CHOOSE_THEN `u:num->num`
   (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN
  DISCH_THEN(X_CHOOSE_THEN `k:num` STRIP_ASSUME_TAC) THEN
  MP_TAC(SPECL [`a:num->num`; `p:num`; `n:num`; `s:(num->num)->bool`]
        KSIMPLEX_SUCCESSOR) THEN
  REWRITE_TAC[ASSUME `ksimplex p n s`; ASSUME `a:num->num IN s`] THEN
  MATCH_MP_TAC(TAUT `~a /\ (b ==> c) ==> a \/ b ==> c`) THEN CONJ_TAC THENL
   [ASM_MESON_TAC[KLE_ANTISYM]; ALL_TAC] THEN
  DISCH_THEN(X_CHOOSE_THEN `v:num->num`
   (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN
  DISCH_THEN(X_CHOOSE_THEN `l:num` STRIP_ASSUME_TAC) THEN
  ABBREV_TAC `a' = \j:num. if j = l then u(j) + 1 else u(j)` THEN
  FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [FUN_EQ_THM]) THEN
  REWRITE_TAC[] THEN DISCH_THEN(ASSUME_TAC o GSYM) THEN
  SUBGOAL_THEN `~(k:num = l)` ASSUME_TAC THENL
   [DISCH_TAC THEN
    UNDISCH_TAC `!j:num. v j = (if j = l then a j + 1 else a j)` THEN
    ASM_REWRITE_TAC[] THEN DISCH_THEN(MP_TAC o SPEC `l:num`) THEN
    REWRITE_TAC[] THEN DISCH_TAC THEN
    FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [ksimplex]) THEN
    DISCH_THEN(MP_TAC o last o CONJUNCTS) THEN
    DISCH_THEN(MP_TAC o SPECL [`u:num->num`; `v:num->num`]) THEN
    ASM_REWRITE_TAC[] THEN
    ASM_REWRITE_TAC[kle] THEN
    DISCH_THEN(DISJ_CASES_THEN (CHOOSE_THEN (MP_TAC o SPEC `l:num` o
      CONJUNCT2))) THEN
    ASM_REWRITE_TAC[] THEN COND_CASES_TAC THEN ARITH_TAC;
    ALL_TAC] THEN
  SUBGOAL_THEN `~(a':num->num = a)` ASSUME_TAC THENL
   [ASM_REWRITE_TAC[FUN_EQ_THM] THEN DISCH_THEN(MP_TAC o SPEC `k:num`) THEN
    ASM_REWRITE_TAC[] THEN ARITH_TAC;
    ALL_TAC] THEN
  SUBGOAL_THEN `~((a':num->num) IN s)` ASSUME_TAC THENL
   [DISCH_TAC THEN FIRST_ASSUM(MP_TAC o GEN_REWRITE_RULE I [ksimplex]) THEN
    DISCH_THEN(MP_TAC o SPECL [`a:num->num`; `a':num->num`] o
      last o CONJUNCTS) THEN ASM_REWRITE_TAC[] THEN
    DISCH_THEN(DISJ_CASES_THEN (MP_TAC o MATCH_MP KLE_IMP_POINTWISE)) THENL
     [DISCH_THEN(MP_TAC o SPEC `k:num`) THEN ASM_REWRITE_TAC[] THEN ARITH_TAC;
      DISCH_THEN(MP_TAC o SPEC `l:num`) THEN ASM_REWRITE_TAC[] THEN
      ARITH_TAC];
    ALL_TAC] THEN
  SUBGOAL_THEN
   `kle n u a /\ kle n u a' /\ kle n a v /\ kle n a' v`
  STRIP_ASSUME_TAC THENL
   [REWRITE_TAC[kle] THEN REPEAT CONJ_TAC THENL
     [EXISTS_TAC `{k:num}`;
      EXISTS_TAC `{l:num}`;
      EXISTS_TAC `{l:num}`;
      EXISTS_TAC `{k:num}`] THEN
    ASM_REWRITE_TAC[IN_SING; SUBSET; IN_NUMSEG] THEN
    ASM_MESON_TAC[ADD_CLAUSES];
    ALL_TAC] THEN
  SUBGOAL_THEN `!x. kle n u x /\ kle n x v
                    ==> ((x = u) \/ (x = a) \/ (x = a') \/ (x = v))`
  ASSUME_TAC THENL
   [X_GEN_TAC `x:num->num` THEN
    DISCH_THEN(CONJUNCTS_THEN (MP_TAC o MATCH_MP KLE_IMP_POINTWISE)) THEN
    ASM_REWRITE_TAC[FUN_EQ_THM; IMP_IMP; AND_FORALL_THM] THEN
    ONCE_REWRITE_TAC[COND_RAND] THEN
    ASM_CASES_TAC `(x:num->num) k = u k` THEN
    ASM_CASES_TAC `(x:num->num) l = u l` THENL
     [DISCH_THEN(fun th -> DISJ1_TAC THEN MP_TAC th);
      DISCH_THEN(fun th -> DISJ2_TAC THEN DISJ2_TAC THEN DISJ1_TAC THEN
                           MP_TAC th);
      DISCH_THEN(fun th -> DISJ2_TAC THEN DISJ1_TAC THEN MP_TAC th);
      DISCH_THEN(fun th -> REPEAT DISJ2_TAC THEN MP_TAC th)] THEN
    MATCH_MP_TAC MONO_FORALL THEN X_GEN_TAC `j:num` THEN
    REPEAT(COND_CASES_TAC THEN
           ASM_REWRITE_TAC[LE_ANTISYM;
       ARITH_RULE `x <= u + 1 /\ u <= x <=> (x = u) \/ (x = u + 1)`]);
    ALL_TAC] THEN
  SUBGOAL_THEN `kle n u v` ASSUME_TAC THENL
   [ASM_MESON_TAC[KLE_TRANS; ksimplex]; ALL_TAC] THEN
  SUBGOAL_THEN `ksimplex p n (a' INSERT (s DELETE a))` ASSUME_TAC THENL
   [MP_TAC(ASSUME `ksimplex p n s`) THEN REWRITE_TAC[ksimplex] THEN
    MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
     [SIMP_TAC[HAS_SIZE; FINITE_INSERT; FINITE_DELETE; CARD_CLAUSES;
               CARD_DELETE; IN_DELETE] THEN
      ASM_REWRITE_TAC[] THEN STRIP_TAC THEN ARITH_TAC;
      ALL_TAC] THEN
    MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
     [REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
      SIMP_TAC[TAUT `a \/ b ==> c <=> (a ==> c) /\ (b ==> c)`] THEN
      REWRITE_TAC[RIGHT_FORALL_IMP_THM; LEFT_FORALL_IMP_THM; EXISTS_REFL] THEN
      ASM_REWRITE_TAC[] THEN
      DISCH_THEN(fun th -> X_GEN_TAC `j:num` THEN MP_TAC th) THEN
      COND_CASES_TAC THEN ASM_SIMP_TAC[] THEN
      DISCH_THEN(MP_TAC o SPEC `v:num->num`) THEN ASM_REWRITE_TAC[] THEN
      DISCH_THEN(MP_TAC o SPEC `l:num`) THEN ASM_REWRITE_TAC[];
      ALL_TAC] THEN
    MATCH_MP_TAC MONO_AND THEN CONJ_TAC THENL
     [REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
      REWRITE_TAC[TAUT `(a \/ b) /\ c <=> a /\ c \/ b /\ c`] THEN
      SIMP_TAC[TAUT `a \/ b ==> c <=> (a ==> c) /\ (b ==> c)`] THEN
      REWRITE_TAC[IMP_CONJ; RIGHT_FORALL_IMP_THM] THEN
      REWRITE_TAC[EXISTS_REFL; LEFT_FORALL_IMP_THM] THEN
      ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[];
      ALL_TAC] THEN
    REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
    SUBGOAL_THEN
     `!x. x IN s /\ kle n v x ==> kle n a' x`
    ASSUME_TAC THENL
     [X_GEN_TAC `x:num->num` THEN STRIP_TAC THEN
      MATCH_MP_TAC KLE_BETWEEN_R THEN
      MAP_EVERY EXISTS_TAC [`u:num->num`; `v:num->num`] THEN
      ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[ksimplex; KLE_TRANS];
      ALL_TAC] THEN
    SUBGOAL_THEN
     `!x. x IN s /\ kle n x u ==> kle n x a'`
    ASSUME_TAC THENL
     [X_GEN_TAC `x:num->num` THEN STRIP_TAC THEN
      MATCH_MP_TAC KLE_BETWEEN_L THEN
      MAP_EVERY EXISTS_TAC [`u:num->num`; `v:num->num`] THEN
      ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[ksimplex; KLE_TRANS];
      ALL_TAC] THEN
    SUBGOAL_THEN
     `!x. x IN s /\ ~(x = a) ==> kle n a' x \/ kle n x a'`
     (fun th -> MESON_TAC[th; KLE_REFL; ASSUME `(a:num->num) IN s`]) THEN
    X_GEN_TAC `x:num->num` THEN STRIP_TAC THEN
    ASM_CASES_TAC `kle n v x` THENL [ASM_MESON_TAC[]; ALL_TAC] THEN
    ASM_CASES_TAC `kle n x u` THENL [ASM_MESON_TAC[]; ALL_TAC] THEN
    SUBGOAL_THEN `(x:num->num = u) \/ (x = a) \/ (x = a') \/ (x = v)`
     (fun th -> ASM_MESON_TAC[th; KLE_REFL]) THEN
    FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_MESON_TAC[ksimplex];
    ALL_TAC] THEN
  MATCH_MP_TAC HAS_SIZE_CARD THEN CONV_TAC HAS_SIZE_CONV THEN
  MAP_EVERY EXISTS_TAC
   [`s:(num->num)->bool`; `a' INSERT (s DELETE (a:num->num))`] THEN
  CONJ_TAC THENL
   [REWRITE_TAC[EXTENSION; IN_DELETE; IN_INSERT] THEN ASM_MESON_TAC[];
    ALL_TAC] THEN
  GEN_REWRITE_TAC I [EXTENSION] THEN REWRITE_TAC[IN_ELIM_THM] THEN
  X_GEN_TAC `s':(num->num)->bool` THEN
  REWRITE_TAC[IN_INSERT; NOT_IN_EMPTY] THEN EQ_TAC THENL
   [ALL_TAC;
    DISCH_THEN(DISJ_CASES_THEN SUBST1_TAC) THENL
     [ASM_MESON_TAC[]; ALL_TAC] THEN
    ASM_REWRITE_TAC[] THEN EXISTS_TAC `a':num->num` THEN
    REWRITE_TAC[EXTENSION; IN_INSERT; IN_DELETE] THEN ASM_MESON_TAC[]] THEN
  DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN
  DISCH_THEN(X_CHOOSE_THEN `a'':num->num` STRIP_ASSUME_TAC) THEN
  SUBGOAL_THEN `(a:num->num) IN s' \/ a' IN s'` MP_TAC THENL
   [ALL_TAC;
    MATCH_MP_TAC MONO_OR THEN CONJ_TAC THEN DISCH_TAC THEN
    MP_TAC(ASSUME `s' DELETE a'' = s DELETE (a:num->num)`) THEN
    REWRITE_TAC[EXTENSION] THEN
    DISCH_THEN(fun th -> MP_TAC th THEN MP_TAC th) THENL
     [DISCH_THEN(MP_TAC o SPEC `a:num->num`);
      DISCH_THEN(MP_TAC o SPEC `a':num->num`)] THEN
    REWRITE_TAC[IN_DELETE] THEN ASM_REWRITE_TAC[IN_INSERT; IN_DELETE] THEN
    DISCH_THEN(SUBST_ALL_TAC o SYM) THEN ASM_MESON_TAC[]] THEN
  SUBGOAL_THEN `~(u:num->num = v)` ASSUME_TAC THENL
   [ASM_REWRITE_TAC[FUN_EQ_THM] THEN DISCH_THEN(MP_TAC o SPEC `l:num`) THEN
    ASM_REWRITE_TAC[] THEN ARITH_TAC;
    ALL_TAC] THEN
  SUBGOAL_THEN `~(kle n v u)` ASSUME_TAC THENL
   [ASM_MESON_TAC[KLE_ANTISYM]; ALL_TAC] THEN
  SUBGOAL_THEN `~(u:num->num = a)` ASSUME_TAC THENL
   [ASM_REWRITE_TAC[FUN_EQ_THM] THEN DISCH_THEN(MP_TAC o SPEC `k:num`) THEN
    ASM_REWRITE_TAC[] THEN ARITH_TAC;
    ALL_TAC] THEN
  SUBGOAL_THEN `~(v:num->num = a)` ASSUME_TAC THENL
   [ASM_REWRITE_TAC[FUN_EQ_THM] THEN DISCH_THEN(MP_TAC o SPEC `l:num`) THEN
    ASM_REWRITE_TAC[] THEN ARITH_TAC;
    ALL_TAC] THEN
  SUBGOAL_THEN `u:num->num IN s' /\ v IN s'` STRIP_ASSUME_TAC THENL
   [ASM_MESON_TAC[EXTENSION; IN_DELETE]; ALL_TAC] THEN
  ASM_CASES_TAC
   `!x. x IN s' ==> kle n x u \/ kle n v x`
  THENL
   [ALL_TAC;
    FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [NOT_FORALL_THM]) THEN
    DISCH_THEN(X_CHOOSE_THEN `w:num->num` MP_TAC) THEN
    REWRITE_TAC[NOT_IMP; DE_MORGAN_THM] THEN STRIP_TAC THEN
    SUBGOAL_THEN `(w:num->num = u) \/ (w = a) \/ (w = a') \/ (w = v)`
     (fun th -> ASM_MESON_TAC[KLE_REFL; th]) THEN
    FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_MESON_TAC[ksimplex]] THEN
  MP_TAC(SPECL [`u:num->num`; `p:num`; `n:num`; `s':(num->num)->bool`]
               KSIMPLEX_SUCCESSOR) THEN
  ANTS_TAC THENL [ASM_MESON_TAC[EXTENSION; IN_DELETE]; ALL_TAC] THEN
  DISCH_THEN(DISJ_CASES_THEN2 (MP_TAC o SPEC `v:num->num`) MP_TAC) THENL
   [ASM_MESON_TAC[EXTENSION; IN_DELETE]; ALL_TAC] THEN
  ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_THEN(K ALL_TAC) THEN
  UNDISCH_TAC `!x. x IN s' ==> kle n x u \/ kle n v x` THEN
  REWRITE_TAC[NOT_EXISTS_THM] THEN MATCH_MP_TAC MONO_FORALL THEN
  X_GEN_TAC `w:num->num` THEN
  DISCH_THEN(fun th -> STRIP_TAC THEN MP_TAC th) THEN
  ASM_REWRITE_TAC[] THEN
  DISCH_THEN(DISJ_CASES_THEN(MP_TAC o MATCH_MP KLE_IMP_POINTWISE)) THEN
  ASM_REWRITE_TAC[] THENL
   [MESON_TAC[ARITH_RULE `~(i + 1 <= i)`]; ALL_TAC] THEN
  DISCH_THEN(fun th -> MP_TAC(SPEC `k:num` th) THEN
                       MP_TAC(SPEC `l:num` th)) THEN
  ASM_REWRITE_TAC[] THEN
  REPEAT(COND_CASES_TAC THEN ASM_REWRITE_TAC[]) THEN TRY ARITH_TAC THEN
  UNDISCH_TAC `~(k:num = l)` THEN ASM_REWRITE_TAC[]);;
(* ------------------------------------------------------------------------- *) (* Hence another step towards concreteness. *) (* ------------------------------------------------------------------------- *)
let KUHN_SIMPLEX_LEMMA = 
prove (`!p n. (!s. ksimplex p (n + 1) s ==> (IMAGE rl s SUBSET 0..n+1)) /\ ODD(CARD{f | (?s a. ksimplex p (n + 1) s /\ a IN s /\ (f = s DELETE a)) /\ (IMAGE rl f = 0 .. n) /\ ((?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = 0)) \/ (?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = p)))}) ==> ODD(CARD {s | s IN {s | ksimplex p (n + 1) s} /\ (IMAGE rl s = 0..n+1)})`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `ODD(CARD {f | f IN {f | ?s. s IN {s | ksimplex p (n + 1) s} /\ (?a. a IN s /\ (f = s DELETE a))} /\ (IMAGE rl f = 0..n) /\ ((?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = 0)) \/ (?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = p)))})` MP_TAC THENL [ASM_REWRITE_TAC[IN_ELIM_THM; RIGHT_AND_EXISTS_THM]; ALL_TAC] THEN MATCH_MP_TAC KUHN_COMPLETE_LEMMA THEN REWRITE_TAC[FINITE_SIMPLICES] THEN ASM_REWRITE_TAC[IN_ELIM_THM] THEN REPEAT STRIP_TAC THEN CONV_TAC(LAND_CONV(ONCE_DEPTH_CONV SYM_CONV)) THEN ASM_REWRITE_TAC[] THENL [ASM_MESON_TAC[ksimplex; ARITH_RULE `(n + 1) + 1 = n + 2`]; ASM_SIMP_TAC[]; MATCH_MP_TAC KSIMPLEX_REPLACE_0; MATCH_MP_TAC KSIMPLEX_REPLACE_1; MATCH_MP_TAC KSIMPLEX_REPLACE_2] THEN ASM_MESON_TAC[ARITH_RULE `~(n + 1 = 0)`]);;
(* ------------------------------------------------------------------------- *) (* Reduced labelling. *) (* ------------------------------------------------------------------------- *)
let reduced = new_definition
 `reduced label n (x:num->num) =
     @k. k <= n /\
         (!i. 1 <= i /\ i < k + 1 ==> (label x i = 0)) /\
         ((k = n) \/ ~(label x (k + 1) = 0))`;;
let REDUCED_LABELLING = 
prove (`!label x n. reduced label n x <= n /\ (!i. 1 <= i /\ i < reduced label n x + 1 ==> (label x i = 0)) /\ ((reduced label n x = n) \/ ~(label x (reduced label n x + 1) = 0))`,
REPEAT GEN_TAC THEN REWRITE_TAC[reduced] THEN CONV_TAC SELECT_CONV THEN MP_TAC(SPEC `\j. j <= n /\ ~(label (x:num->num) (j + 1) = 0) \/ (n = j)` num_WOP) THEN REWRITE_TAC[] THEN MATCH_MP_TAC(TAUT `a /\ (b ==> c) ==> (a <=> b) ==> c`) THEN CONJ_TAC THENL [MESON_TAC[]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `k:num` THEN ASM_CASES_TAC `k = n:num` THEN ASM_REWRITE_TAC[] THEN STRIP_TAC THEN ASM_REWRITE_TAC[LE_REFL] THEN X_GEN_TAC `i:num` THEN STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `i - 1`) THEN SIMP_TAC[LT_IMP_LE] THEN ASM_SIMP_TAC[ARITH_RULE `1 <= i /\ i < n + 1 ==> i - 1 < n`] THEN ASM_SIMP_TAC[ARITH_RULE `1 <= i /\ i < n + 1 ==> ~(n = i - 1)`] THEN ASM_SIMP_TAC[SUB_ADD] THEN POP_ASSUM_LIST(MP_TAC o end_itlist CONJ) THEN ARITH_TAC);;
let REDUCED_LABELLING_UNIQUE = 
prove (`!label x n. r <= n /\ (!i. 1 <= i /\ i < r + 1 ==> (label x i = 0)) /\ ((r = n) \/ ~(label x (r + 1) = 0)) ==> (reduced label n x = r)`,
REPEAT GEN_TAC THEN DISCH_THEN(REPEAT_TCL CONJUNCTS_THEN ASSUME_TAC) THEN REPEAT_TCL CONJUNCTS_THEN ASSUME_TAC(SPECL [`label:(num->num)->(num->num)`; `x:num->num`; `n:num`] REDUCED_LABELLING) THEN MATCH_MP_TAC(ARITH_RULE `~(a < b) /\ ~(b < a:num) ==> (a = b)`) THEN ASM_MESON_TAC[ARITH_RULE `s < r:num /\ r <= n ==> ~(s = n)`; ARITH_RULE `s < r ==> 1 <= s + 1 /\ s + 1 < r + 1`]);;
let REDUCED_LABELLING_0 = 
prove (`!label n x j. 1 <= j /\ j <= n /\ (label x j = 0) ==> ~(reduced label n x = j - 1)`,
REPEAT STRIP_TAC THEN MP_TAC(SPECL [`label:(num->num)->num->num`; `x:num->num`; `n:num`] REDUCED_LABELLING) THEN ASM_SIMP_TAC[SUB_ADD; ARITH_RULE `1 <= j /\ j <= n ==> ~(j - 1 = n)`]);;
let REDUCED_LABELLING_1 = 
prove (`!label n x j. 1 <= j /\ j <= n /\ ~(label x j = 0) ==> reduced label n x < j`,
REWRITE_TAC[GSYM NOT_LE] THEN REPEAT STRIP_TAC THEN MP_TAC(SPECL [`label:(num->num)->num->num`; `x:num->num`; `n:num`] REDUCED_LABELLING) THEN DISCH_THEN(MP_TAC o SPEC `j:num` o CONJUNCT1 o CONJUNCT2) THEN ASM_REWRITE_TAC[ARITH_RULE `y < x + 1 <=> (y <= x)`]);;
let REDUCED_LABELLING_SUC = 
prove (`!lab n x. ~(reduced lab (n + 1) x = n + 1) ==> (reduced lab (n + 1) x = reduced lab n x)`,
REPEAT STRIP_TAC THEN CONV_TAC SYM_CONV THEN MATCH_MP_TAC REDUCED_LABELLING_UNIQUE THEN ASM_MESON_TAC[REDUCED_LABELLING; ARITH_RULE `x <= n + 1 /\ ~(x = n + 1) ==> x <= n`]);;
let COMPLETE_FACE_TOP = 
prove (`!lab f n. (!x j. x IN f /\ 1 <= j /\ j <= n + 1 /\ (x j = 0) ==> (lab x j = 0)) /\ (!x j. x IN f /\ 1 <= j /\ j <= n + 1 /\ (x j = p) ==> (lab x j = 1)) ==> ((IMAGE (reduced lab (n + 1)) f = 0..n) /\ ((?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = 0)) \/ (?j. 1 <= j /\ j <= n + 1 /\ !x. x IN f ==> (x j = p))) <=> (IMAGE (reduced lab (n + 1)) f = 0..n) /\ (!x. x IN f ==> (x (n + 1) = p)))`,
REPEAT STRIP_TAC THEN EQ_TAC THENL [ALL_TAC; MESON_TAC[ARITH_RULE `1 <= n + 1`; LE_REFL]] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [EXTENSION]) THENL [DISCH_THEN(MP_TAC o SPEC `j - 1`) THEN ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_THEN(K ALL_TAC) THEN ASM_SIMP_TAC[IN_IMAGE; IN_NUMSEG; LE_0; NOT_EXISTS_THM; ARITH_RULE `j <= n + 1 ==> j - 1 <= n`] THEN ASM_MESON_TAC[REDUCED_LABELLING_0]; DISCH_THEN(MP_TAC o SPEC `j:num`) THEN REWRITE_TAC[IN_IMAGE; IN_NUMSEG; LE_0; NOT_LE] THEN ASM_SIMP_TAC[ARITH_RULE `j <= n + 1 ==> ((j <= n) <=> ~(j = n + 1))`] THEN ASM_MESON_TAC[REDUCED_LABELLING_1; ARITH_RULE `~(1 = 0)`; LT_REFL]]);;
(* ------------------------------------------------------------------------- *) (* Hence we get just about the nice induction. *) (* ------------------------------------------------------------------------- *)
let KUHN_INDUCTION = 
prove (`!p n. 0 < p /\ (!x j. (!j. x(j) <= p) /\ 1 <= j /\ j <= n + 1 /\ (x j = 0) ==> (lab x j = 0)) /\ (!x j. (!j. x(j) <= p) /\ 1 <= j /\ j <= n + 1 /\ (x j = p) ==> (lab x j = 1)) ==> ODD(CARD {f | ksimplex p n f /\ (IMAGE (reduced lab n) f = 0..n)}) ==> ODD(CARD {s | ksimplex p (n + 1) s /\ (IMAGE (reduced lab (n + 1)) s = 0..n+1)})`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC(REWRITE_RULE[IN_ELIM_THM] KUHN_SIMPLEX_LEMMA) THEN CONJ_TAC THENL [REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; IN_NUMSEG; LE_0] THEN MESON_TAC[ARITH_RULE `x <= n ==> x <= n + 1`; REDUCED_LABELLING]; ALL_TAC] THEN FIRST_ASSUM(fun th -> MP_TAC th THEN MATCH_MP_TAC EQ_IMP THEN AP_TERM_TAC) THEN AP_TERM_TAC THEN GEN_REWRITE_TAC I [EXTENSION] THEN X_GEN_TAC `f:(num->num)->bool` THEN REWRITE_TAC[IN_ELIM_THM] THEN ASM_CASES_TAC `(!x j. x IN f /\ 1 <= j /\ j <= n + 1 /\ (x j = 0) ==> (lab x j = 0)) /\ (!x j. x IN f /\ 1 <= j /\ j <= n + 1 /\ (x j = p) ==> (lab x j = 1))` THENL [ALL_TAC; MATCH_MP_TAC(TAUT `~a /\ ~b ==> (a /\ c <=> b /\ d)`) THEN CONJ_TAC THEN FIRST_X_ASSUM(MP_TAC o check (is_neg o concl)) THEN REWRITE_TAC[CONTRAPOS_THM] THEN REWRITE_TAC[ksimplex] THEN ASM_MESON_TAC[IN_DELETE]] THEN ASM_SIMP_TAC[COMPLETE_FACE_TOP] THEN ASM_CASES_TAC `!x. x IN f ==> (x(n + 1):num = p)` THENL [ALL_TAC; ASM_REWRITE_TAC[] THEN DISCH_THEN(MP_TAC o CONJUNCT1) THEN REWRITE_TAC[ksimplex] THEN ASM_MESON_TAC[ARITH_RULE `~(n + 1 <= n)`]] THEN ASM_SIMP_TAC[SIMPLEX_TOP_FACE] THEN ASM_CASES_TAC `ksimplex p n f` THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[EXTENSION; IN_IMAGE] THEN AP_TERM_TAC THEN GEN_REWRITE_TAC I [FUN_EQ_THM] THEN X_GEN_TAC `k:num` THEN REWRITE_TAC[] THEN AP_THM_TAC THEN AP_TERM_TAC THEN AP_TERM_TAC THEN GEN_REWRITE_TAC I [FUN_EQ_THM] THEN X_GEN_TAC `x:num->num` THEN REWRITE_TAC[] THEN ASM_CASES_TAC `(x:num->num) IN f` THEN ASM_REWRITE_TAC[] THEN AP_TERM_TAC THEN CONV_TAC SYM_CONV THEN MATCH_MP_TAC REDUCED_LABELLING_SUC THEN MATCH_MP_TAC(ARITH_RULE `a:num < b ==> ~(a = b)`) THEN MATCH_MP_TAC REDUCED_LABELLING_1 THEN REWRITE_TAC[LE_REFL; ARITH_RULE `1 <= n + 1`] THEN MATCH_MP_TAC(ARITH_RULE `(n = 1) ==> ~(n = 0)`) THEN FIRST_X_ASSUM MATCH_MP_TAC THEN REWRITE_TAC[LE_REFL; ARITH_RULE `1 <= n + 1`] THEN ASM_MESON_TAC[ksimplex]);;
(* ------------------------------------------------------------------------- *) (* And so we get the final combinatorial result. *) (* ------------------------------------------------------------------------- *)
let KSIMPLEX_0 = 
prove (`ksimplex p 0 s <=> (s = {(\x. p)})`,
REWRITE_TAC[ksimplex; ADD_CLAUSES] THEN CONV_TAC(LAND_CONV(LAND_CONV HAS_SIZE_CONV)) THEN REWRITE_TAC[ARITH_RULE `1 <= j /\ j <= 0 <=> F`] THEN ONCE_REWRITE_TAC[TAUT `a /\ b <=> ~(a ==> ~b)`] THEN SIMP_TAC[LEFT_IMP_EXISTS_THM] THEN REWRITE_TAC[IN_SING] THEN SIMP_TAC[RIGHT_FORALL_IMP_THM] THEN REWRITE_TAC[KLE_REFL] THEN REWRITE_TAC[LEFT_FORALL_IMP_THM; EXISTS_REFL] THEN REWRITE_TAC[AND_FORALL_THM; ARITH_RULE `x <= y:num /\ (x = y) <=> (x = y)`] THEN REWRITE_TAC[NOT_FORALL_THM; NOT_IMP] THEN REWRITE_TAC[GSYM FUN_EQ_THM] THEN ONCE_REWRITE_TAC[CONJ_SYM] THEN REWRITE_TAC[UNWIND_THM2]);;
let REDUCE_LABELLING_0 = 
prove (`!lab x. reduced lab 0 x = 0`,
REPEAT GEN_TAC THEN MATCH_MP_TAC REDUCED_LABELLING_UNIQUE THEN REWRITE_TAC[LE_REFL] THEN ARITH_TAC);;
let KUHN_COMBINATORIAL = 
prove (`!lab p n. 0 < p /\ (!x j. (!j. x(j) <= p) /\ 1 <= j /\ j <= n /\ (x j = 0) ==> (lab x j = 0)) /\ (!x j. (!j. x(j) <= p) /\ 1 <= j /\ j <= n /\ (x j = p) ==> (lab x j = 1)) ==> ODD(CARD {s | ksimplex p n s /\ (IMAGE (reduced lab n) s = 0..n)})`,
GEN_TAC THEN GEN_TAC THEN ONCE_REWRITE_TAC[IMP_CONJ] THEN REWRITE_TAC[RIGHT_FORALL_IMP_THM] THEN DISCH_TAC THEN INDUCT_TAC THENL [DISCH_THEN(K ALL_TAC) THEN SUBGOAL_THEN `{s | ksimplex p 0 s /\ (IMAGE (reduced lab 0) s = 0 .. 0)} = {{(\x. p)}}` (fun th -> SIMP_TAC[CARD_CLAUSES; NOT_IN_EMPTY; FINITE_RULES; th; ARITH]) THEN GEN_REWRITE_TAC I [EXTENSION] THEN REWRITE_TAC[IN_ELIM_THM; KSIMPLEX_0; IN_SING] THEN GEN_TAC THEN MATCH_MP_TAC(TAUT `(a ==> b) ==> (a /\ b <=> a)`) THEN DISCH_THEN SUBST_ALL_TAC THEN REWRITE_TAC[NUMSEG_SING; EXTENSION; IN_SING; IN_IMAGE] THEN REWRITE_TAC[REDUCE_LABELLING_0] THEN MESON_TAC[]; STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o check (is_imp o concl)) THEN ANTS_TAC THENL [ASM_MESON_TAC[ARITH_RULE `j <= n ==> j <= SUC n`]; ALL_TAC] THEN REWRITE_TAC[ADD1] THEN MATCH_MP_TAC KUHN_INDUCTION THEN ASM_REWRITE_TAC[GSYM ADD1]]);;
let KUHN_LEMMA = 
prove (`!n p label. 0 < p /\ 0 < n /\ (!x. (!i. 1 <= i /\ i <= n ==> x(i) <= p) ==> !i. 1 <= i /\ i <= n ==> (label x i = 0) \/ (label x i = 1)) /\ (!x. (!i. 1 <= i /\ i <= n ==> x(i) <= p) ==> !i. 1 <= i /\ i <= n /\ (x i = 0) ==> (label x i = 0)) /\ (!x. (!i. 1 <= i /\ i <= n ==> x(i) <= p) ==> !i. 1 <= i /\ i <= n /\ (x i = p) ==> (label x i = 1)) ==> ?q. (!i. 1 <= i /\ i <= n ==> q(i) < p) /\ (!i. 1 <= i /\ i <= n ==> ?r s. (!j. 1 <= j /\ j <= n ==> q(j) <= r(j) /\ r(j) <= q(j) + 1) /\ (!j. 1 <= j /\ j <= n ==> q(j) <= s(j) /\ s(j) <= q(j) + 1) /\ ~(label r i = label s i))`,
REPEAT STRIP_TAC THEN MP_TAC(SPECL [`label:(num->num)->num->num`; `p:num`; `n:num`] KUHN_COMBINATORIAL) THEN ANTS_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN ASM_CASES_TAC `{s | ksimplex p n s /\ (IMAGE (reduced label n) s = 0 .. n)} = {}` THENL [ASM_REWRITE_TAC[CARD_CLAUSES; ARITH]; ALL_TAC] THEN DISCH_THEN(K ALL_TAC) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [GSYM MEMBER_NOT_EMPTY]) THEN REWRITE_TAC[IN_ELIM_THM] THEN DISCH_THEN(X_CHOOSE_THEN `s:(num->num)->bool` STRIP_ASSUME_TAC) THEN MP_TAC(SPECL [`p:num`; `n:num`; `s:(num->num)->bool`] KSIMPLEX_EXTREMA_STRONG) THEN ASM_REWRITE_TAC[GSYM LT_NZ] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `a:num->num` THEN DISCH_THEN(X_CHOOSE_THEN `b:num->num` STRIP_ASSUME_TAC) THEN CONJ_TAC THEN X_GEN_TAC `i:num` THEN STRIP_TAC THENL [MATCH_MP_TAC(ARITH_RULE `x + 1 <= y ==> x < y`) THEN MATCH_MP_TAC LE_TRANS THEN EXISTS_TAC `(b:num->num) i` THEN CONJ_TAC THENL [ASM_REWRITE_TAC[LE_REFL]; ALL_TAC] THEN ASM_MESON_TAC[ksimplex]; ALL_TAC] THEN UNDISCH_TAC `IMAGE (reduced label n) s = 0 .. n` THEN REWRITE_TAC[EXTENSION; IN_IMAGE] THEN DISCH_THEN(fun th -> MP_TAC(SPEC `i - 1` th) THEN MP_TAC(SPEC `i:num` th)) THEN ASM_REWRITE_TAC[IN_NUMSEG; LE_0] THEN DISCH_THEN(X_CHOOSE_THEN `u:num->num` (STRIP_ASSUME_TAC o GSYM)) THEN ASM_SIMP_TAC[ARITH_RULE `1 <= i /\ i <= n ==> i - 1 <= n`] THEN DISCH_THEN(X_CHOOSE_THEN `v:num->num` (STRIP_ASSUME_TAC o GSYM)) THEN MAP_EVERY EXISTS_TAC [`u:num->num`; `v:num->num`] THEN REWRITE_TAC[CONJ_ASSOC] THEN CONJ_TAC THENL [ASM_MESON_TAC[KLE_IMP_POINTWISE]; ALL_TAC] THEN MP_TAC(SPECL [`label:(num->num)->num->num`; `u:num->num`; `n:num`] REDUCED_LABELLING) THEN MP_TAC(SPECL [`label:(num->num)->num->num`; `v:num->num`; `n:num`] REDUCED_LABELLING) THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[ARITH_RULE `1 <= i /\ i <= n ==> ~(i - 1 = n)`] THEN ASM_SIMP_TAC[SUB_ADD] THEN ASM_MESON_TAC[ARITH_RULE `i < i + 1`]);;
(* ------------------------------------------------------------------------- *) (* The main result for the unit cube. *) (* ------------------------------------------------------------------------- *)
let BROUWER_CUBE = 
prove (`!f:real^N->real^N. f continuous_on (interval [vec 0,vec 1]) /\ IMAGE f (interval [vec 0,vec 1]) SUBSET (interval [vec 0,vec 1]) ==> ?x. x IN interval[vec 0,vec 1] /\ (f x = x)`,
REPEAT STRIP_TAC THEN ABBREV_TAC `n = dimindex(:N)` THEN SUBGOAL_THEN `1 <= n /\ 0 < n /\ ~(n = 0)` STRIP_ASSUME_TAC THENL [EXPAND_TAC "n" THEN REWRITE_TAC[DIMINDEX_NONZERO; DIMINDEX_GE_1] THEN ASM_MESON_TAC[LT_NZ; DIMINDEX_NONZERO]; ALL_TAC] THEN GEN_REWRITE_TAC I [TAUT `p <=> ~ ~ p`] THEN PURE_REWRITE_TAC[NOT_EXISTS_THM; TAUT `~(a /\ b) <=> a ==> ~b`] THEN DISCH_TAC THEN SUBGOAL_THEN `?d. &0 < d /\ !x:real^N. x IN interval[vec 0,vec 1] ==> d <= norm(f x - x)` STRIP_ASSUME_TAC THENL [MATCH_MP_TAC BROUWER_COMPACTNESS_LEMMA THEN ASM_SIMP_TAC[COMPACT_INTERVAL; CONTINUOUS_ON_SUB; CONTINUOUS_ON_CONST; CONTINUOUS_ON_ID] THEN ASM_MESON_TAC[VECTOR_SUB_EQ]; ALL_TAC] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [SUBSET]) THEN REWRITE_TAC[FORALL_IN_IMAGE] THEN FREEZE_THEN(fun th -> DISCH_THEN(MP_TAC o MATCH_MP th)) (SPEC `f:real^N->real^N` KUHN_LABELLING_LEMMA) THEN DISCH_THEN(MP_TAC o SPEC `\i. 1 <= i /\ i <= n`) THEN ANTS_TAC THENL [ASM_SIMP_TAC[IN_INTERVAL; VEC_COMPONENT]; ALL_TAC] THEN REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `label:real^N->num->num` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `!x y i. x IN interval[vec 0,vec 1] /\ y IN interval[vec 0,vec 1] /\ 1 <= i /\ i <= n /\ ~(label (x:real^N) i :num = label y i) ==> abs((f(x) - x)$i) <= norm(f(y) - f(x)) + norm(y - x)` ASSUME_TAC THENL [REPEAT STRIP_TAC THEN MATCH_MP_TAC REAL_LE_TRANS THEN EXISTS_TAC `abs(((f:real^N->real^N)(y) - f(x))$i) + abs((y - x)$i)` THEN ASM_SIMP_TAC[REAL_LE_ADD2; COMPONENT_LE_NORM] THEN ASM_SIMP_TAC[VECTOR_SUB_COMPONENT] THEN MATCH_MP_TAC(REAL_ARITH `!x y fx fy d. (x <= fx /\ fy <= y \/ fx <= x /\ y <= fy) ==> abs(fx - x) <= abs(fy - fx) + abs(y - x)`) THEN FIRST_X_ASSUM(MP_TAC o MATCH_MP (ARITH_RULE `~(a = b) ==> a <= 1 /\ b <= 1 ==> (a = 0) /\ (b = 1) \/ (a = 1) /\ (b = 0)`)) THEN ASM_SIMP_TAC[] THEN STRIP_TAC THEN ASM_SIMP_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `?e. &0 < e /\ !x y z i. x IN interval[vec 0,vec 1] /\ y IN interval[vec 0,vec 1] /\ z IN interval[vec 0,vec 1] /\ 1 <= i /\ i <= n /\ norm(x - z) < e /\ norm(y - z) < e /\ ~(label (x:real^N) i :num = label y i) ==> abs((f(z) - z)$i) < d / &n` MP_TAC THENL [SUBGOAL_THEN `(f:real^N->real^N) uniformly_continuous_on interval[vec 0,vec 1]` MP_TAC THENL [ASM_SIMP_TAC[COMPACT_UNIFORMLY_CONTINUOUS; COMPACT_INTERVAL]; ALL_TAC] THEN REWRITE_TAC[uniformly_continuous_on] THEN DISCH_THEN(MP_TAC o SPEC `d / &n / &8`) THEN SUBGOAL_THEN `&0 < d / &n / &8` ASSUME_TAC THENL [ASM_SIMP_TAC[REAL_LT_DIV; REAL_OF_NUM_LT; LT_MULT; ARITH]; ALL_TAC] THEN ASM_REWRITE_TAC[dist] THEN DISCH_THEN(X_CHOOSE_THEN `e:real` STRIP_ASSUME_TAC) THEN EXISTS_TAC `min (e / &2) (d / &n / &8)` THEN ASM_REWRITE_TAC[REAL_MUL_LZERO; REAL_LT_MIN; REAL_HALF] THEN MAP_EVERY X_GEN_TAC [`x:real^N`; `y:real^N`; `z:real^N`; `i:num`] THEN STRIP_TAC THEN ASM_SIMP_TAC[VECTOR_SUB_COMPONENT] THEN MATCH_MP_TAC(REAL_ARITH `!x fx n1 n2 n3 n4 d4. abs(fx - x) <= n1 + n2 /\ abs(fx - fz) <= n3 /\ abs(x - z) <= n4 /\ n1 < d4 /\ n2 < &2 * d4 /\ n3 < d4 /\ n4 < d4 /\ (&8 * d4 = d) ==> abs(fz - z) < d`) THEN MAP_EVERY EXISTS_TAC [`(x:real^N)$i`; `(f:real^N->real^N)(x)$i`; `norm((f:real^N->real^N) y - f x)`; `norm(y - x:real^N)`; `norm((f:real^N->real^N) x - f z)`; `norm(x - z:real^N)`; `d / &n / &8`] THEN ASM_SIMP_TAC[GSYM VECTOR_SUB_COMPONENT; COMPONENT_LE_NORM] THEN SIMP_TAC[REAL_DIV_LMUL; REAL_OF_NUM_EQ; ARITH] THEN REPEAT CONJ_TAC THENL [FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC REAL_LET_TRANS THEN EXISTS_TAC `norm(x - z:real^N) + norm(y - z)` THEN ASM_SIMP_TAC[REAL_ARITH `a < e / &2 /\ b < e / &2 /\ (&2 * (e / &2) = e) ==> a + b < e`; REAL_DIV_LMUL; REAL_OF_NUM_EQ; ARITH_EQ] THEN REWRITE_TAC[GSYM dist] THEN MESON_TAC[DIST_TRIANGLE; DIST_SYM]; MATCH_MP_TAC REAL_LET_TRANS THEN EXISTS_TAC `norm(x - z:real^N) + norm(y - z)` THEN ASM_SIMP_TAC[REAL_ARITH `a < e /\ b < e ==> a + b < &2 * e`] THEN REWRITE_TAC[GSYM dist] THEN MESON_TAC[DIST_TRIANGLE; DIST_SYM]; FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(REAL_ARITH `a < e / &2 /\ &0 < e /\ (&2 * (e / &2) = e) ==> a < e`) THEN ASM_REWRITE_TAC[] THEN SIMP_TAC[REAL_DIV_LMUL; REAL_OF_NUM_EQ; ARITH_EQ]]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `e:real` (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN X_CHOOSE_THEN `p:num` MP_TAC (SPEC `&1 + &n / e` REAL_ARCH_SIMPLE) THEN DISJ_CASES_TAC(ARITH_RULE `(p = 0) \/ 0 < p`) THENL [DISCH_THEN(fun th -> DISCH_THEN(K ALL_TAC) THEN MP_TAC th) THEN ASM_REWRITE_TAC[LT_REFL; REAL_NOT_LE] THEN ASM_SIMP_TAC[REAL_LT_DIV; REAL_OF_NUM_LT; REAL_ARITH `&0 < x ==> &0 < &1 + x`]; ALL_TAC] THEN DISCH_TAC THEN ASM_REWRITE_TAC[NOT_FORALL_THM] THEN MP_TAC(SPECL [`n:num`; `p:num`; `\v:(num->num). label((lambda i. &(v i) / &p):real^N):num->num`] KUHN_LEMMA) THEN ASM_REWRITE_TAC[ARITH_RULE `(x = 0) \/ (x = 1) <=> x <= 1`] THEN ANTS_TAC THENL [REPEAT STRIP_TAC THEN FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_SIMP_TAC[LAMBDA_BETA; IN_INTERVAL; VEC_COMPONENT] THEN ASM_SIMP_TAC[REAL_LE_RDIV_EQ; REAL_LE_LDIV_EQ; REAL_OF_NUM_LT] THEN ASM_SIMP_TAC[REAL_DIV_REFL; REAL_MUL_LZERO; REAL_MUL_LID; REAL_LT_IMP_NZ; REAL_OF_NUM_LT] THEN ASM_REWRITE_TAC[LE_0; REAL_OF_NUM_LE] THEN REWRITE_TAC[real_div; REAL_MUL_LZERO]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `q:num->num` STRIP_ASSUME_TAC) THEN GEN_REWRITE_TAC BINDER_CONV [SWAP_EXISTS_THM] THEN GEN_REWRITE_TAC I [SWAP_EXISTS_THM] THEN ABBREV_TAC `z:real^N = lambda i. &(q i) / &p` THEN EXISTS_TAC `z:real^N` THEN REWRITE_TAC[TAUT `~(a ==> b) <=> ~b /\ a`] THEN GEN_REWRITE_TAC BINDER_CONV [SWAP_EXISTS_THM] THEN ONCE_REWRITE_TAC[RIGHT_EXISTS_AND_THM] THEN GEN_REWRITE_TAC I [SWAP_EXISTS_THM] THEN ONCE_REWRITE_TAC[RIGHT_EXISTS_AND_THM] THEN SUBGOAL_THEN `z:real^N IN interval[vec 0,vec 1]` ASSUME_TAC THENL [EXPAND_TAC "z" THEN SIMP_TAC[IN_INTERVAL; LAMBDA_BETA; VEC_COMPONENT] THEN ASM_SIMP_TAC[REAL_LE_RDIV_EQ; REAL_LE_LDIV_EQ; REAL_OF_NUM_LT] THEN REWRITE_TAC[REAL_MUL_LZERO; REAL_MUL_LID; REAL_OF_NUM_LE] THEN ASM_SIMP_TAC[LE_0; LT_IMP_LE]; ALL_TAC] THEN SUBGOAL_THEN `?i. 1 <= i /\ i <= n /\ d / &n <= abs((f z - z:real^N)$i)` MP_TAC THENL [SUBGOAL_THEN `d <= norm(f z - z:real^N)` MP_TAC THENL [ASM_SIMP_TAC[]; ALL_TAC] THEN ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN REWRITE_TAC[NOT_EXISTS_THM; TAUT `~(a /\ b /\ c) <=> a /\ b ==> ~c`] THEN REWRITE_TAC[REAL_NOT_LE] THEN DISCH_TAC THEN MATCH_MP_TAC REAL_LET_TRANS THEN EXISTS_TAC `sum(1..dimindex(:N)) (\i. abs((f z - z:real^N)$i))` THEN REWRITE_TAC[NORM_LE_L1] THEN MATCH_MP_TAC SUM_BOUND_LT_GEN THEN REWRITE_TAC[FINITE_NUMSEG; IN_NUMSEG; NUMSEG_EMPTY; CARD_NUMSEG] THEN ASM_REWRITE_TAC[NOT_LT; ADD_SUB]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `i:num` THEN STRIP_TAC THEN ASM_REWRITE_TAC[REAL_NOT_LT] THEN FIRST_X_ASSUM(MP_TAC o SPEC `i:num`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `r:num->num` (X_CHOOSE_THEN `s:num->num` STRIP_ASSUME_TAC)) THEN EXISTS_TAC `(lambda i. &(r i) / &p) :real^N` THEN EXISTS_TAC `(lambda i. &(s i) / &p) :real^N` THEN ASM_REWRITE_TAC[] THEN REPEAT CONJ_TAC THENL [SIMP_TAC[IN_INTERVAL; LAMBDA_BETA; VEC_COMPONENT] THEN ASM_SIMP_TAC[REAL_LE_RDIV_EQ; REAL_LE_LDIV_EQ; REAL_OF_NUM_LT] THEN REWRITE_TAC[REAL_MUL_LZERO; REAL_MUL_LID; REAL_OF_NUM_LE] THEN ASM_MESON_TAC[LE_0; ARITH_RULE `r <= q + 1 /\ q < p ==> r <= p`]; SIMP_TAC[IN_INTERVAL; LAMBDA_BETA; VEC_COMPONENT] THEN ASM_SIMP_TAC[REAL_LE_RDIV_EQ; REAL_LE_LDIV_EQ; REAL_OF_NUM_LT] THEN REWRITE_TAC[REAL_MUL_LZERO; REAL_MUL_LID; REAL_OF_NUM_LE] THEN ASM_MESON_TAC[LE_0; ARITH_RULE `r <= q + 1 /\ q < p ==> r <= p`]; ALL_TAC; ALL_TAC] THEN MATCH_MP_TAC(MATCH_MP (REAL_ARITH `a <= b ==> b < e ==> a < e`) (SPEC_ALL NORM_LE_L1)) THEN MATCH_MP_TAC SUM_BOUND_LT_GEN THEN REWRITE_TAC[FINITE_NUMSEG; IN_NUMSEG; NUMSEG_EMPTY; CARD_NUMSEG] THEN ASM_REWRITE_TAC[NOT_LT; ADD_SUB] THEN EXPAND_TAC "z" THEN EXPAND_TAC "n" THEN SIMP_TAC[VECTOR_SUB_COMPONENT; LAMBDA_BETA] THEN ASM_REWRITE_TAC[real_div; GSYM REAL_SUB_RDISTRIB] THEN REWRITE_TAC[GSYM real_div; REAL_ABS_DIV; REAL_ABS_NUM] THEN ASM_SIMP_TAC[REAL_LT_LDIV_EQ; REAL_OF_NUM_LT] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC REAL_LET_TRANS THEN EXISTS_TAC `&1` THEN ASM_SIMP_TAC[REAL_ARITH `q <= r /\ r <= q + &1 ==> abs(r - q) <= &1`; REAL_OF_NUM_LE; REAL_OF_NUM_ADD] THEN GEN_REWRITE_TAC BINOP_CONV [GSYM REAL_INV_INV] THEN MATCH_MP_TAC REAL_LT_INV2 THEN REWRITE_TAC[REAL_INV_DIV; REAL_INV_MUL] THEN ASM_SIMP_TAC[GSYM real_div; REAL_LT_LDIV_EQ; REAL_LT_RDIV_EQ; REAL_OF_NUM_LT; ARITH] THEN ASM_REWRITE_TAC[REAL_MUL_LZERO; REAL_OF_NUM_LT] THEN REWRITE_TAC[REAL_INV_1; REAL_MUL_LID] THEN ASM_SIMP_TAC[GSYM REAL_LT_LDIV_EQ; REAL_ARITH `&1 + x <= y ==> x < y`]);;
(* ------------------------------------------------------------------------- *) (* Retractions. *) (* ------------------------------------------------------------------------- *) parse_as_infix("retract_of",(12,"right"));;
let retraction = new_definition
  `retraction (s,t) (r:real^N->real^N) <=>
        t SUBSET s /\ r continuous_on s /\ (IMAGE r s SUBSET t) /\
        (!x. x IN t ==> (r x = x))`;;
let retract_of = new_definition
  `t retract_of s <=> ?r. retraction (s,t) r`;;
let RETRACTION = 
prove (`!s t r. retraction (s,t) r <=> t SUBSET s /\ r continuous_on s /\ IMAGE r s = t /\ (!x. x IN t ==> r x = x)`,
REWRITE_TAC[retraction] THEN SET_TAC[]);;
let RETRACT_OF_IMP_EXTENSIBLE = 
prove (`!f:real^M->real^N u s t. s retract_of t /\ f continuous_on s /\ IMAGE f s SUBSET u ==> ?g. g continuous_on t /\ IMAGE g t SUBSET u /\ (!x. x IN s ==> g x = f x)`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN REWRITE_TAC[RETRACTION; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `r:real^M->real^M` THEN STRIP_TAC THEN EXISTS_TAC `(f:real^M->real^N) o (r:real^M->real^M)` THEN REWRITE_TAC[IMAGE_o; o_THM] THEN CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE; ASM SET_TAC[]] THEN ASM_MESON_TAC[]);;
let RETRACTION_IDEMPOTENT = 
prove (`!r s t. retraction (s,t) r ==> !x. x IN s ==> (r(r(x)) = r(x))`,
REWRITE_TAC[retraction; SUBSET; FORALL_IN_IMAGE] THEN MESON_TAC[]);;
let IDEMPOTENT_IMP_RETRACTION = 
prove (`!f:real^N->real^N s. f continuous_on s /\ IMAGE f s SUBSET s /\ (!x. x IN s ==> f(f x) = f x) ==> retraction (s,IMAGE f s) f`,
REWRITE_TAC[retraction] THEN SET_TAC[]);;
let RETRACTION_SUBSET = 
prove (`!r s s' t. retraction (s,t) r /\ t SUBSET s' /\ s' SUBSET s ==> retraction (s',t) r`,
SIMP_TAC[retraction] THEN MESON_TAC[IMAGE_SUBSET; SUBSET_TRANS; CONTINUOUS_ON_SUBSET]);;
let RETRACT_OF_SUBSET = 
prove (`!s s' t. t retract_of s /\ t SUBSET s' /\ s' SUBSET s ==> t retract_of s'`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of; LEFT_AND_EXISTS_THM] THEN MATCH_MP_TAC MONO_EXISTS THEN MESON_TAC[RETRACTION_SUBSET]);;
let RETRACT_OF_TRANSLATION = 
prove (`!a t s:real^N->bool. t retract_of s ==> (IMAGE (\x. a + x) t) retract_of (IMAGE (\x. a + x) s)`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of; retraction] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^N->real^N` STRIP_ASSUME_TAC) THEN EXISTS_TAC `(\x:real^N. a + x) o r o (\x:real^N. --a + x)` THEN ASM_SIMP_TAC[IMAGE_SUBSET; FORALL_IN_IMAGE] THEN REPEAT CONJ_TAC THENL [REPEAT(MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN SIMP_TAC[CONTINUOUS_ON_ADD; CONTINUOUS_ON_CONST; CONTINUOUS_ON_ID]) THEN ASM_REWRITE_TAC[GSYM IMAGE_o; o_DEF; VECTOR_ARITH `--a + a + x:real^N = x`; IMAGE_ID]; REWRITE_TAC[IMAGE_o] THEN MATCH_MP_TAC IMAGE_SUBSET THEN GEN_REWRITE_TAC (LAND_CONV o RAND_CONV o ONCE_DEPTH_CONV) [GSYM IMAGE_o] THEN ASM_REWRITE_TAC[o_DEF; VECTOR_ARITH `--a + a + x:real^N = x`; IMAGE_ID]; ASM_SIMP_TAC[o_DEF; VECTOR_ARITH `--a + a + x:real^N = x`]]);;
let RETRACT_OF_TRANSLATION_EQ = 
prove (`!a t s:real^N->bool. (IMAGE (\x. a + x) t) retract_of (IMAGE (\x. a + x) s) <=> t retract_of s`,
REPEAT GEN_TAC THEN EQ_TAC THEN REWRITE_TAC[RETRACT_OF_TRANSLATION] THEN DISCH_THEN(MP_TAC o SPEC `--a:real^N` o MATCH_MP RETRACT_OF_TRANSLATION) THEN REWRITE_TAC[GSYM IMAGE_o; o_DEF; IMAGE_ID; VECTOR_ARITH `--a + a + x:real^N = x`]);;
add_translation_invariants [RETRACT_OF_TRANSLATION_EQ];;
let RETRACT_OF_INJECTIVE_LINEAR_IMAGE = 
prove (`!f:real^M->real^N s t. linear f /\ (!x y. f x = f y ==> x = y) /\ t retract_of s ==> (IMAGE f t) retract_of (IMAGE f s)`,
REPEAT GEN_TAC THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN REWRITE_TAC[retract_of; retraction] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^M->real^M` STRIP_ASSUME_TAC) THEN MP_TAC(ISPEC `f:real^M->real^N` LINEAR_INJECTIVE_LEFT_INVERSE) THEN ASM_REWRITE_TAC[FUN_EQ_THM; o_THM; I_THM] THEN DISCH_THEN(X_CHOOSE_THEN `g:real^N->real^M` STRIP_ASSUME_TAC) THEN EXISTS_TAC `(f:real^M->real^N) o r o (g:real^N->real^M)` THEN UNDISCH_THEN `!x y. (f:real^M->real^N) x = f y ==> x = y` (K ALL_TAC) THEN ASM_SIMP_TAC[IMAGE_SUBSET; FORALL_IN_IMAGE] THEN REPEAT CONJ_TAC THENL [REPEAT(MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN ASM_SIMP_TAC[LINEAR_CONTINUOUS_ON]) THEN ASM_REWRITE_TAC[GSYM IMAGE_o; o_DEF; IMAGE_ID]; REWRITE_TAC[IMAGE_o] THEN MATCH_MP_TAC IMAGE_SUBSET THEN GEN_REWRITE_TAC (LAND_CONV o RAND_CONV o ONCE_DEPTH_CONV) [GSYM IMAGE_o] THEN ASM_REWRITE_TAC[o_DEF; IMAGE_ID]; ASM_SIMP_TAC[o_DEF]]);;
let RETRACT_OF_LINEAR_IMAGE_EQ = 
prove (`!f:real^M->real^N s t. linear f /\ (!x y. f x = f y ==> x = y) /\ (!y. ?x. f x = y) ==> ((IMAGE f t) retract_of (IMAGE f s) <=> t retract_of s)`,
REPEAT GEN_TAC THEN DISCH_TAC THEN EQ_TAC THENL [DISCH_TAC; ASM_MESON_TAC[RETRACT_OF_INJECTIVE_LINEAR_IMAGE]] THEN FIRST_ASSUM(X_CHOOSE_THEN `h:real^N->real^M` STRIP_ASSUME_TAC o MATCH_MP LINEAR_BIJECTIVE_LEFT_RIGHT_INVERSE) THEN SUBGOAL_THEN `!s. s = IMAGE (h:real^N->real^M) (IMAGE (f:real^M->real^N) s)` (fun th -> ONCE_REWRITE_TAC[th]) THENL [ASM SET_TAC[]; ALL_TAC] THEN MATCH_MP_TAC RETRACT_OF_INJECTIVE_LINEAR_IMAGE THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[]);;
add_linear_invariants [RETRACT_OF_LINEAR_IMAGE_EQ];;
let RETRACTION_REFL = 
prove (`!s. retraction (s,s) (\x. x)`,
REWRITE_TAC[retraction; IMAGE_ID; SUBSET_REFL; CONTINUOUS_ON_ID]);;
let RETRACT_OF_REFL = 
prove (`!s. s retract_of s`,
REWRITE_TAC[retract_of] THEN MESON_TAC[RETRACTION_REFL]);;
let RETRACT_OF_IMP_SUBSET = 
prove (`!s t. s retract_of t ==> s SUBSET t`,
SIMP_TAC[retract_of; retraction] THEN MESON_TAC[]);;
let RETRACT_OF_EMPTY = 
prove (`(!s:real^N->bool. {} retract_of s <=> s = {}) /\ (!s:real^N->bool. s retract_of {} <=> s = {})`,
REWRITE_TAC[retract_of; retraction; SUBSET_EMPTY; IMAGE_CLAUSES] THEN CONJ_TAC THEN X_GEN_TAC `s:real^N->bool` THEN ASM_CASES_TAC `s:real^N->bool = {}` THEN ASM_REWRITE_TAC[NOT_IN_EMPTY; IMAGE_EQ_EMPTY; CONTINUOUS_ON_EMPTY; SUBSET_REFL]);;
let RETRACT_OF_SING = 
prove (`!s x:real^N. {x} retract_of s <=> x IN s`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of; RETRACTION] THEN EQ_TAC THENL [SET_TAC[]; ALL_TAC] THEN DISCH_TAC THEN EXISTS_TAC `(\y. x):real^N->real^N` THEN REWRITE_TAC[CONTINUOUS_ON_CONST] THEN ASM SET_TAC[]);;
let RETRACTION_o = 
prove (`!f g s t u:real^N->bool. retraction (s,t) f /\ retraction (t,u) g ==> retraction (s,u) (g o f)`,
REPEAT GEN_TAC THEN REWRITE_TAC[retraction] THEN REPEAT STRIP_TAC THENL [ASM SET_TAC[]; MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[CONTINUOUS_ON_SUBSET]; REWRITE_TAC[IMAGE_o] THEN ASM SET_TAC[]; REWRITE_TAC[o_THM] THEN ASM SET_TAC[]]);;
let RETRACT_OF_TRANS = 
prove (`!s t u:real^N->bool. s retract_of t /\ t retract_of u ==> s retract_of u`,
REWRITE_TAC[retract_of] THEN MESON_TAC[RETRACTION_o]);;
let CLOSED_IN_RETRACT = 
prove (`!s t:real^N->bool. s retract_of t ==> closed_in (subtopology euclidean t) s`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of; retraction] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^N->real^N` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `s = {x:real^N | x IN t /\ lift(norm(r x - x)) = vec 0}` SUBST1_TAC THENL [REWRITE_TAC[GSYM DROP_EQ; DROP_VEC; LIFT_DROP; NORM_EQ_0] THEN REWRITE_TAC[VECTOR_SUB_EQ] THEN ASM SET_TAC[]; MATCH_MP_TAC CONTINUOUS_CLOSED_IN_PREIMAGE_CONSTANT THEN MATCH_MP_TAC CONTINUOUS_ON_LIFT_NORM_COMPOSE THEN MATCH_MP_TAC CONTINUOUS_ON_SUB THEN ASM_SIMP_TAC[CONTINUOUS_ON_ID]]);;
let RETRACT_OF_CONTRACTIBLE = 
prove (`!s t:real^N->bool. contractible t /\ s retract_of t ==> contractible s`,
REPEAT GEN_TAC THEN REWRITE_TAC[contractible; retract_of] THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC (X_CHOOSE_TAC `r:real^N->real^N`)) THEN SIMP_TAC[HOMOTOPIC_WITH; PCROSS; LEFT_IMP_EXISTS_THM] THEN FIRST_X_ASSUM(STRIP_ASSUME_TAC o GEN_REWRITE_RULE I [retraction]) THEN MAP_EVERY X_GEN_TAC [`a:real^N`; `h:real^(1,N)finite_sum->real^N`] THEN RULE_ASSUM_TAC(REWRITE_RULE[SUBSET; FORALL_IN_IMAGE]) THEN STRIP_TAC THEN MAP_EVERY EXISTS_TAC [`(r:real^N->real^N) a`; `(r:real^N->real^N) o (h:real^(1,N)finite_sum->real^N)`] THEN ASM_SIMP_TAC[o_THM; IMAGE_o; SUBSET] THEN CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP(REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]; ASM SET_TAC[]]);;
let RETRACT_OF_COMPACT = 
prove (`!s t:real^N->bool. compact t /\ s retract_of t ==> compact s`,
REWRITE_TAC[retract_of; RETRACTION] THEN MESON_TAC[COMPACT_CONTINUOUS_IMAGE]);;
let RETRACT_OF_CLOSED = 
prove (`!s t. closed t /\ s retract_of t ==> closed t`,
let RETRACT_OF_CONNECTED = 
prove (`!s t:real^N->bool. connected t /\ s retract_of t ==> connected s`,
REWRITE_TAC[retract_of; RETRACTION] THEN MESON_TAC[CONNECTED_CONTINUOUS_IMAGE]);;
let RETRACT_OF_PATH_CONNECTED = 
prove (`!s t:real^N->bool. path_connected t /\ s retract_of t ==> path_connected s`,
REWRITE_TAC[retract_of; RETRACTION] THEN MESON_TAC[PATH_CONNECTED_CONTINUOUS_IMAGE]);;
let RETRACT_OF_SIMPLY_CONNECTED = 
prove (`!s t:real^N->bool. simply_connected t /\ s retract_of t ==> simply_connected s`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC ASSUME_TAC) THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ] (REWRITE_RULE[CONJ_ASSOC] SIMPLY_CONNECTED_RETRACTION_GEN)) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `r:real^N->real^N` THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[IMAGE_ID; CONTINUOUS_ON_ID]);;
let RETRACT_OF_HOMOTOPICALLY_TRIVIAL = 
prove (`!s t:real^N->bool u:real^M->bool. t retract_of s /\ (!f g. f continuous_on u /\ IMAGE f u SUBSET s /\ g continuous_on u /\ IMAGE g u SUBSET s ==> homotopic_with (\x. T) (u,s) f g) ==> (!f g. f continuous_on u /\ IMAGE f u SUBSET t /\ g continuous_on u /\ IMAGE g u SUBSET t ==> homotopic_with (\x. T) (u,t) f g)`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ONCE_REWRITE_TAC[TAUT `p /\ q /\ r /\ s <=> p /\ q /\ T /\ r /\ s /\ T`] THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ] HOMOTOPICALLY_TRIVIAL_RETRACTION_GEN) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `r:real^N->real^N` THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[CONTINUOUS_ON_ID; IMAGE_ID]);;
let RETRACT_OF_HOMOTOPICALLY_TRIVIAL_NULL = 
prove (`!s t:real^N->bool u:real^M->bool. t retract_of s /\ (!f. f continuous_on u /\ IMAGE f u SUBSET s ==> ?c. homotopic_with (\x. T) (u,s) f (\x. c)) ==> (!f. f continuous_on u /\ IMAGE f u SUBSET t ==> ?c. homotopic_with (\x. T) (u,t) f (\x. c))`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ONCE_REWRITE_TAC[TAUT `p /\ q <=> p /\ q /\ T`] THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ] HOMOTOPICALLY_TRIVIAL_RETRACTION_NULL_GEN) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `r:real^N->real^N` THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[CONTINUOUS_ON_ID; IMAGE_ID]);;
let RETRACT_OF_COHOMOTOPICALLY_TRIVIAL = 
prove (`!s t:real^N->bool u:real^M->bool. t retract_of s /\ (!f g. f continuous_on s /\ IMAGE f s SUBSET u /\ g continuous_on s /\ IMAGE g s SUBSET u ==> homotopic_with (\x. T) (s,u) f g) ==> (!f g. f continuous_on t /\ IMAGE f t SUBSET u /\ g continuous_on t /\ IMAGE g t SUBSET u ==> homotopic_with (\x. T) (t,u) f g)`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ONCE_REWRITE_TAC[TAUT `p /\ q /\ r /\ s <=> p /\ q /\ T /\ r /\ s /\ T`] THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ] COHOMOTOPICALLY_TRIVIAL_RETRACTION_GEN) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `r:real^N->real^N` THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[CONTINUOUS_ON_ID; IMAGE_ID]);;
let RETRACT_OF_COHOMOTOPICALLY_TRIVIAL_NULL = 
prove (`!s t:real^N->bool u:real^M->bool. t retract_of s /\ (!f. f continuous_on s /\ IMAGE f s SUBSET u ==> ?c. homotopic_with (\x. T) (s,u) f (\x. c)) ==> (!f. f continuous_on t /\ IMAGE f t SUBSET u ==> ?c. homotopic_with (\x. T) (t,u) f (\x. c))`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ONCE_REWRITE_TAC[TAUT `p /\ q <=> p /\ q /\ T`] THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ] COHOMOTOPICALLY_TRIVIAL_RETRACTION_NULL_GEN) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `r:real^N->real^N` THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[CONTINUOUS_ON_ID; IMAGE_ID]);;
let RETRACTION_IMP_QUOTIENT_MAP = 
prove (`!r s t:real^N->bool. retraction (s,t) r ==> !u. u SUBSET t ==> (open_in (subtopology euclidean s) {x | x IN s /\ r x IN u} <=> open_in (subtopology euclidean t) u)`,
REPEAT GEN_TAC THEN REWRITE_TAC[RETRACTION] THEN STRIP_TAC THEN MATCH_MP_TAC CONTINUOUS_RIGHT_INVERSE_IMP_QUOTIENT_MAP THEN EXISTS_TAC `\x:real^N. x` THEN ASM_REWRITE_TAC[CONTINUOUS_ON_ID; SUBSET_REFL; IMAGE_ID]);;
let RETRACT_OF_LOCALLY_CONNECTED = 
prove (`!s t:real^N->bool. s retract_of t /\ locally connected t ==> locally connected s`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of] THEN DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC MP_TAC) THEN FIRST_ASSUM(SUBST1_TAC o SYM o el 2 o CONJUNCTS o GEN_REWRITE_RULE I [RETRACTION]) THEN MATCH_MP_TAC(REWRITE_RULE[IMP_CONJ] LOCALLY_CONNECTED_QUOTIENT_IMAGE) THEN MATCH_MP_TAC RETRACTION_IMP_QUOTIENT_MAP THEN ASM_MESON_TAC[RETRACTION]);;
let RETRACT_OF_LOCALLY_PATH_CONNECTED = 
prove (`!s t:real^N->bool. s retract_of t /\ locally path_connected t ==> locally path_connected s`,
REPEAT GEN_TAC THEN REWRITE_TAC[retract_of] THEN DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC MP_TAC) THEN FIRST_ASSUM(SUBST1_TAC o SYM o el 2 o CONJUNCTS o GEN_REWRITE_RULE I [RETRACTION]) THEN MATCH_MP_TAC(REWRITE_RULE[IMP_CONJ] LOCALLY_PATH_CONNECTED_QUOTIENT_IMAGE) THEN MATCH_MP_TAC RETRACTION_IMP_QUOTIENT_MAP THEN ASM_MESON_TAC[RETRACTION]);;
let RETRACT_OF_LOCALLY_COMPACT = 
prove (`!s t:real^N->bool. locally compact s /\ t retract_of s ==> locally compact t`,
let ABSOLUTE_RETRACTION_CONVEX_CLOSED_RELATIVE = 
prove (`!s:real^N->bool t. convex s /\ closed s /\ ~(s = {}) /\ s SUBSET t ==> ?r. retraction (t,s) r /\ !x. x IN (affine hull s) DIFF (relative_interior s) ==> r(x) IN relative_frontier s`,
REPEAT STRIP_TAC THEN REWRITE_TAC[retraction] THEN EXISTS_TAC `closest_point(s:real^N->bool)` THEN ASM_SIMP_TAC[CONTINUOUS_ON_CLOSEST_POINT; CLOSEST_POINT_SELF] THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; CLOSEST_POINT_IN_SET] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC CLOSEST_POINT_IN_RELATIVE_FRONTIER THEN ASM_MESON_TAC[SUBSET; RELATIVE_INTERIOR_SUBSET]);;
let ABSOLUTE_RETRACTION_CONVEX_CLOSED = 
prove (`!s:real^N->bool t. convex s /\ closed s /\ ~(s = {}) /\ s SUBSET t ==> ?r. retraction (t,s) r /\ (!x. ~(x IN s) ==> r(x) IN frontier s)`,
REPEAT STRIP_TAC THEN REWRITE_TAC[retraction] THEN EXISTS_TAC `closest_point(s:real^N->bool)` THEN ASM_SIMP_TAC[CONTINUOUS_ON_CLOSEST_POINT; CLOSEST_POINT_SELF] THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; CLOSEST_POINT_IN_SET] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC CLOSEST_POINT_IN_FRONTIER THEN ASM_MESON_TAC[SUBSET; INTERIOR_SUBSET]);;
let ABSOLUTE_RETRACT_CONVEX_CLOSED = 
prove (`!s:real^N->bool t. convex s /\ closed s /\ ~(s = {}) /\ s SUBSET t ==> s retract_of t`,
REWRITE_TAC[retract_of] THEN MESON_TAC[ABSOLUTE_RETRACTION_CONVEX_CLOSED]);;
(* ------------------------------------------------------------------------- *) (* A neighbourhood retract is an ANR for Euclidean space. *) (* ------------------------------------------------------------------------- *)
let NEIGHBOURHOOD_RETRACT_IMP_ANR = 
prove (`!s:real^M->bool s':real^N->bool t u. s retract_of t /\ open t /\ s homeomorphic s' /\ s' SUBSET u ==> ?t'. open_in (subtopology euclidean u) t' /\ s' retract_of t'`,
MAP_EVERY X_GEN_TAC [`X:real^M->bool`; `Y:real^N->bool`; `U:real^M->bool`; `K:real^N->bool`] THEN STRIP_TAC THEN SUBGOAL_THEN `locally compact (Y:real^N->bool)` ASSUME_TAC THENL [ASM_MESON_TAC[RETRACT_OF_LOCALLY_COMPACT; OPEN_IMP_LOCALLY_COMPACT; HOMEOMORPHIC_LOCAL_COMPACTNESS]; ALL_TAC] THEN SUBGOAL_THEN `?W:real^N->bool. open_in (subtopology euclidean K) W /\ closed_in (subtopology euclidean W) Y` STRIP_ASSUME_TAC THENL [FIRST_ASSUM(X_CHOOSE_THEN `W:real^N->bool` STRIP_ASSUME_TAC o MATCH_MP LOCALLY_COMPACT_CLOSED_IN_OPEN) THEN EXISTS_TAC `K INTER W:real^N->bool` THEN ASM_SIMP_TAC[OPEN_IN_OPEN_INTER; CLOSED_IN_CLOSED] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [CLOSED_IN_CLOSED]) THEN ASM SET_TAC[]; ALL_TAC] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [homeomorphic]) THEN REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`f:real^M->real^N`; `g:real^N->real^M`] THEN REWRITE_TAC[homeomorphism] THEN STRIP_TAC THEN MP_TAC(ISPECL [`g:real^N->real^M`; `W:real^N->bool`; `Y:real^N->bool`] TIETZE_UNBOUNDED) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `h:real^N->real^M` STRIP_ASSUME_TAC) THEN EXISTS_TAC `{x | x IN W /\ (h:real^N->real^M) x IN U}` THEN CONJ_TAC THENL [MATCH_MP_TAC OPEN_IN_TRANS THEN EXISTS_TAC `W:real^N->bool` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC CONTINUOUS_OPEN_IN_PREIMAGE_GEN THEN EXISTS_TAC `(:real^M)` THEN ASM_REWRITE_TAC[SUBTOPOLOGY_UNIV; GSYM OPEN_IN; SUBSET_UNIV]; ALL_TAC] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN REWRITE_TAC[retraction; retract_of; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `r:real^M->real^M` THEN STRIP_TAC THEN EXISTS_TAC `(f:real^M->real^N) o r o (h:real^N->real^M)` THEN SUBGOAL_THEN `(W:real^N->bool) SUBSET K /\ Y SUBSET W` STRIP_ASSUME_TAC THENL [ASM_MESON_TAC[OPEN_IN_IMP_SUBSET; CLOSED_IN_IMP_SUBSET]; ALL_TAC] THEN REWRITE_TAC[IMAGE_o; o_THM] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN REPEAT(MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC) THEN REWRITE_TAC[IMAGE_o] THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]);;
let NEIGHBOURHOOD_RETRACT_IMP_ANR_UNIV = 
prove (`!s:real^M->bool s':real^N->bool t. s retract_of t /\ open t /\ s homeomorphic s' ==> ?t'. open t' /\ s' retract_of t'`,
REPEAT STRIP_TAC THEN ONCE_REWRITE_TAC[OPEN_IN] THEN ONCE_REWRITE_TAC[GSYM SUBTOPOLOGY_UNIV] THEN MATCH_MP_TAC NEIGHBOURHOOD_RETRACT_IMP_ANR THEN ASM_MESON_TAC[SUBSET_UNIV]);;
let HOMEOMORPHIC_ANRNESS = 
prove (`!s:real^M->bool t:real^N->bool. s homeomorphic t ==> ((?u. open u /\ s retract_of u) <=> (?u. open u /\ t retract_of u))`,
REPEAT STRIP_TAC THEN EQ_TAC THEN DISCH_TAC THENL [MP_TAC(ISPECL [`s:real^M->bool`; `t:real^N->bool`] NEIGHBOURHOOD_RETRACT_IMP_ANR_UNIV); MP_TAC(ISPECL [`t:real^N->bool`; `s:real^M->bool`] NEIGHBOURHOOD_RETRACT_IMP_ANR_UNIV)] THEN ASM_REWRITE_TAC[] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_MESON_TAC[HOMEOMORPHIC_SYM]);;
(* ------------------------------------------------------------------------- *) (* Likewise for ARs, at least given closedness. *) (* ------------------------------------------------------------------------- *)
let ABSOLUTE_RETRACT_IMP_AR_GEN = 
prove (`!s:real^M->bool s':real^N->bool t u. s retract_of t /\ convex t /\ closed t /\ ~(t = {}) /\ s homeomorphic s' /\ closed_in (subtopology euclidean u) s' ==> s' retract_of u`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [homeomorphic]) THEN REWRITE_TAC[homeomorphism; LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`f:real^M->real^N`; `g:real^N->real^M`] THEN STRIP_TAC THEN MP_TAC(ISPECL [`g:real^N->real^M`; `u:real^N->bool`; `s':real^N->bool`] TIETZE_UNBOUNDED) THEN ASM_REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `h:real^N->real^M` THEN STRIP_TAC THEN SUBGOAL_THEN `s retract_of (:real^M)` MP_TAC THENL [MATCH_MP_TAC RETRACT_OF_TRANS THEN EXISTS_TAC `t:real^M->bool` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC ABSOLUTE_RETRACT_CONVEX_CLOSED THEN ASM_REWRITE_TAC[SUBSET_UNIV]; REWRITE_TAC[retract_of; retraction; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `r:real^M->real^M` THEN STRIP_TAC THEN EXISTS_TAC `(f:real^M->real^N) o r o (h:real^N->real^M)` THEN FIRST_ASSUM(ASSUME_TAC o MATCH_MP CLOSED_IN_IMP_SUBSET) THEN ASM_REWRITE_TAC[IMAGE_o; o_THM] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN REPEAT(MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC) THEN REWRITE_TAC[IMAGE_o] THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]]);;
let ABSOLUTE_RETRACT_IMP_AR = 
prove (`!s s'. s retract_of (:real^M) /\ s homeomorphic s' /\ closed s' ==> s' retract_of (:real^N)`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC ABSOLUTE_RETRACT_IMP_AR_GEN THEN MAP_EVERY EXISTS_TAC [`s:real^M->bool`; `(:real^M)`] THEN ASM_REWRITE_TAC[SUBTOPOLOGY_UNIV; GSYM CLOSED_IN] THEN REWRITE_TAC[CONVEX_UNIV; CLOSED_UNIV; UNIV_NOT_EMPTY]);;
let HOMEOMORPHIC_COMPACT_ARNESS = 
prove (`!s s'. s homeomorphic s' ==> (compact s /\ s retract_of (:real^M) <=> compact s' /\ s' retract_of (:real^N))`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `compact(s:real^M->bool) /\ compact(s':real^N->bool)` THENL [ALL_TAC; ASM_MESON_TAC[HOMEOMORPHIC_COMPACTNESS]] THEN ASM_REWRITE_TAC[] THEN EQ_TAC THEN MATCH_MP_TAC(ONCE_REWRITE_RULE[IMP_CONJ_ALT] ABSOLUTE_RETRACT_IMP_AR) THEN ASM_MESON_TAC[HOMEOMORPHIC_SYM; COMPACT_IMP_CLOSED]);;
(* ------------------------------------------------------------------------- *) (* More retract properties including connection of complements. *) (* ------------------------------------------------------------------------- *)
let ABSOLUTE_RETRACT_HOMEOMORPHIC_CONVEX_COMPACT = 
prove (`!s:real^N->bool t u:real^M->bool. s homeomorphic u /\ ~(s = {}) /\ s SUBSET t /\ convex u /\ compact u ==> s retract_of t`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC RETRACT_OF_SUBSET THEN EXISTS_TAC `(:real^N)` THEN ASM_REWRITE_TAC[SUBSET_UNIV] THEN FIRST_ASSUM(MP_TAC o MATCH_MP HOMEOMORPHIC_COMPACT_ARNESS) THEN FIRST_ASSUM(MP_TAC o MATCH_MP HOMEOMORPHIC_COMPACTNESS) THEN ASM_SIMP_TAC[] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC ABSOLUTE_RETRACT_CONVEX_CLOSED THEN ASM_MESON_TAC[COMPACT_IMP_CLOSED; HOMEOMORPHIC_EMPTY; SUBSET_UNIV]);;
let ABSOLUTE_RETRACT_PATH_IMAGE_ARC = 
prove (`!g s:real^N->bool. arc g /\ path_image g SUBSET s ==> (path_image g) retract_of s`,
REPEAT STRIP_TAC THEN MP_TAC (ISPECL [`path_image g:real^N->bool`; `s:real^N->bool`; `interval[vec 0:real^1,vec 1:real^1]`] ABSOLUTE_RETRACT_HOMEOMORPHIC_CONVEX_COMPACT) THEN DISCH_THEN MATCH_MP_TAC THEN ASM_REWRITE_TAC[PATH_IMAGE_NONEMPTY] THEN REWRITE_TAC[COMPACT_INTERVAL; CONVEX_INTERVAL] THEN ONCE_REWRITE_TAC[HOMEOMORPHIC_SYM] THEN MATCH_MP_TAC HOMEOMORPHIC_COMPACT THEN EXISTS_TAC `g:real^1->real^N` THEN RULE_ASSUM_TAC(REWRITE_RULE[arc; path; path_image]) THEN ASM_REWRITE_TAC[COMPACT_INTERVAL; path_image]);;
let RELATIVE_BOUNDARY_RETRACT_OF_PUNCTURED_AFFINE_HULL = 
prove (`!s a:real^N. convex s /\ compact s /\ a IN relative_interior s ==> (s DIFF relative_interior s) retract_of ((affine hull s) DELETE a)`,
REPEAT GEN_TAC THEN GEOM_ORIGIN_TAC `a:real^N` THEN REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`s:real^N->bool`; `vec 0:real^N`] RAY_TO_RELATIVE_FRONTIER) THEN REWRITE_TAC[relative_frontier] THEN ASM_SIMP_TAC[VECTOR_ADD_LID; RIGHT_IMP_EXISTS_THM; COMPACT_IMP_BOUNDED; CLOSURE_CLOSED; COMPACT_IMP_CLOSED] THEN REWRITE_TAC[SKOLEM_THM; LEFT_IMP_EXISTS_THM] THEN REWRITE_TAC[TAUT `p ==> q /\ r <=> (p ==> q) /\ (p ==> r)`] THEN REWRITE_TAC[FORALL_AND_THM] THEN X_GEN_TAC `dd:real^N->real` THEN STRIP_TAC THEN REWRITE_TAC[retract_of; retraction] THEN EXISTS_TAC `\x:real^N. dd x % x` THEN REPEAT CONJ_TAC THENL [MP_TAC(ISPECL [`affine:(real^N->bool)->bool`; `s:real^N->bool`] HULL_SUBSET) THEN ASM SET_TAC[]; MATCH_MP_TAC CONTINUOUS_ON_COMPACT_SURFACE_PROJECTION THEN EXISTS_TAC `s DIFF relative_interior s:real^N->bool` THEN ASM_SIMP_TAC[COMPACT_RELATIVE_BOUNDARY] THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC(SET_RULE `s SUBSET t /\ z IN s' ==> s DIFF s' SUBSET t DELETE z`) THEN ASM_REWRITE_TAC[HULL_SUBSET]; MATCH_MP_TAC(MESON[AFFINE_EQ_SUBSPACE; SUBSPACE_IMP_CONIC] `affine s /\ vec 0 IN s ==> conic s`) THEN REWRITE_TAC[AFFINE_AFFINE_HULL] THEN ASM_MESON_TAC[HULL_INC; SUBSET; RELATIVE_INTERIOR_SUBSET]; MAP_EVERY X_GEN_TAC [`x:real^N`; `k:real`] THEN REWRITE_TAC[IN_DELETE; IN_DIFF] THEN STRIP_TAC THEN EQ_TAC THENL [STRIP_TAC; ASM_MESON_TAC[IN_DIFF]] THEN MATCH_MP_TAC(REAL_ARITH `~(a < b) /\ ~(b < a) ==> a = b`) THEN CONJ_TAC THEN DISCH_TAC THENL [ALL_TAC; ASM_MESON_TAC[REAL_LT_IMP_LE]] THEN MP_TAC(ISPECL [`s:real^N->bool`; `vec 0:real^N`; `k % x:real^N`] IN_RELATIVE_INTERIOR_CLOSURE_CONVEX_SEGMENT) THEN ASM_SIMP_TAC[COMPACT_IMP_CLOSED; CLOSURE_CLOSED] THEN REWRITE_TAC[SUBSET; IN_SEGMENT; NOT_FORALL_THM] THEN EXISTS_TAC `dd(x) % x:real^N` THEN RULE_ASSUM_TAC(REWRITE_RULE[IN_DIFF]) THEN ASM_SIMP_TAC[] THEN ONCE_REWRITE_TAC[EQ_SYM_EQ] THEN ASM_SIMP_TAC[VECTOR_MUL_EQ_0; REAL_LT_IMP_NZ] THEN EXISTS_TAC `(dd:real^N->real) x / k` THEN ASM_SIMP_TAC[REAL_LT_DIV; VECTOR_MUL_RZERO; VECTOR_ADD_LID; REAL_LT_LDIV_EQ; VECTOR_MUL_ASSOC; REAL_MUL_LID; REAL_DIV_RMUL; REAL_LT_IMP_NZ]]; ASM_REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; IN_DELETE; IN_UNIV]; X_GEN_TAC `x:real^N` THEN DISCH_TAC THEN REWRITE_TAC[] THEN SUBGOAL_THEN `x IN affine hull s /\ ~(x:real^N = vec 0)` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`affine:(real^N->bool)->bool`; `s:real^N->bool`] HULL_SUBSET) THEN ASM SET_TAC[]; REPEAT(FIRST_X_ASSUM(MP_TAC o SPEC `x:real^N`)) THEN ASM_REWRITE_TAC[IN_DIFF] THEN REPEAT STRIP_TAC] THEN SUBGOAL_THEN `(dd:real^N->real) x = &1` (fun th -> REWRITE_TAC[th; VECTOR_MUL_LID]) THEN MATCH_MP_TAC(REAL_ARITH `~(d < &1) /\ ~(&1 < d) ==> d = &1`) THEN CONJ_TAC THEN DISCH_TAC THEN MP_TAC(ISPECL [`s:real^N->bool`; `vec 0:real^N`] IN_RELATIVE_INTERIOR_CLOSURE_CONVEX_SEGMENT) THENL [DISCH_THEN(MP_TAC o SPEC `x:real^N`); DISCH_THEN(MP_TAC o SPEC `(dd x) % x:real^N`)] THEN RULE_ASSUM_TAC(REWRITE_RULE[IN_DIFF]) THEN ASM_SIMP_TAC[CLOSURE_CLOSED; COMPACT_IMP_CLOSED] THEN REWRITE_TAC[SUBSET; IN_SEGMENT; NOT_FORALL_THM; NOT_IMP] THEN REWRITE_TAC[VECTOR_MUL_RZERO; VECTOR_ADD_LID] THENL [EXISTS_TAC `dd x % x:real^N` THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[]; EXISTS_TAC `x:real^N` THEN ASM_SIMP_TAC[VECTOR_ARITH `vec 0 = a % x <=> a % x:real^N = vec 0`] THEN ASM_SIMP_TAC[VECTOR_MUL_EQ_0; REAL_LT_IMP_NZ] THEN EXISTS_TAC `inv((dd:real^N->real) x)` THEN ASM_SIMP_TAC[VECTOR_MUL_ASSOC; REAL_MUL_LINV; REAL_LT_IMP_NZ] THEN ASM_SIMP_TAC[VECTOR_MUL_LID; REAL_LT_INV_EQ; REAL_INV_LT_1]]]);;
let RELATIVE_FRONTIER_RETRACT_OF_PUNCTURED_AFFINE_HULL = 
prove (`!s a:real^N. convex s /\ bounded s /\ a IN relative_interior s ==> relative_frontier s retract_of (affine hull s) DELETE a`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`closure s:real^N->bool`; `a:real^N`] RELATIVE_BOUNDARY_RETRACT_OF_PUNCTURED_AFFINE_HULL) THEN ASM_SIMP_TAC[COMPACT_CLOSURE; CONVEX_CLOSURE; relative_frontier; CONVEX_RELATIVE_INTERIOR_CLOSURE; AFFINE_HULL_CLOSURE]);;
let ANR_RELATIVE_FRONTIER_CONVEX = 
prove (`!s:real^N->bool. bounded s /\ convex s ==> ?t. open t /\ (relative_frontier s) retract_of t`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `s:real^N->bool = {}` THEN ASM_REWRITE_TAC[RELATIVE_FRONTIER_EMPTY] THENL [ASM_MESON_TAC[RETRACT_OF_REFL; OPEN_EMPTY]; ALL_TAC] THEN SUBGOAL_THEN `~(relative_interior s:real^N->bool = {})` MP_TAC THENL [ASM_SIMP_TAC[RELATIVE_INTERIOR_EQ_EMPTY]; ALL_TAC] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `a:real^N` THEN DISCH_TAC THEN EXISTS_TAC `{x | x IN (:real^N) /\ closest_point (affine hull s) x IN ((:real^N) DELETE a)}` THEN CONJ_TAC THENL [REWRITE_TAC[OPEN_IN] THEN ONCE_REWRITE_TAC[GSYM SUBTOPOLOGY_UNIV] THEN MATCH_MP_TAC CONTINUOUS_OPEN_IN_PREIMAGE_GEN THEN EXISTS_TAC `(:real^N)` THEN SIMP_TAC[OPEN_IN_DELETE; OPEN_IN_REFL; SUBSET_UNIV; ETA_AX]; MATCH_MP_TAC RETRACT_OF_TRANS THEN EXISTS_TAC `(affine hull s) DELETE (a:real^N)` THEN CONJ_TAC THENL [MATCH_MP_TAC RELATIVE_FRONTIER_RETRACT_OF_PUNCTURED_AFFINE_HULL THEN ASM_REWRITE_TAC[]; REWRITE_TAC[retract_of; retraction] THEN EXISTS_TAC `closest_point (affine hull s:real^N->bool)` THEN REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; IN_DELETE] THEN ASM_SIMP_TAC[IN_ELIM_THM; IN_UNIV; CLOSEST_POINT_SELF; CLOSEST_POINT_IN_SET; AFFINE_HULL_EQ_EMPTY; CLOSED_AFFINE_HULL]]] THEN MATCH_MP_TAC CONTINUOUS_ON_CLOSEST_POINT THEN ASM_SIMP_TAC[AFFINE_IMP_CONVEX; AFFINE_AFFINE_HULL; CLOSED_AFFINE_HULL; AFFINE_HULL_EQ_EMPTY]);;
let FRONTIER_RETRACT_OF_PUNCTURED_UNIVERSE = 
prove (`!s a. convex s /\ bounded s /\ a IN interior s ==> (frontier s) retract_of ((:real^N) DELETE a)`,
REPEAT STRIP_TAC THEN FIRST_ASSUM(ASSUME_TAC o MATCH_MP (SET_RULE `a IN s ==> ~(s = {})`)) THEN MP_TAC(ISPECL [`s:real^N->bool`; `a:real^N`] RELATIVE_FRONTIER_RETRACT_OF_PUNCTURED_AFFINE_HULL) THEN ASM_SIMP_TAC[RELATIVE_FRONTIER_NONEMPTY_INTERIOR; RELATIVE_INTERIOR_NONEMPTY_INTERIOR; AFFINE_HULL_NONEMPTY_INTERIOR]);;
let SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE_GEN = 
prove (`!a r b:real^N. b IN ball(a,r) ==> sphere(a,r) retract_of ((:real^N) DELETE b)`,
REPEAT STRIP_TAC THEN REWRITE_TAC[GSYM FRONTIER_CBALL] THEN MATCH_MP_TAC FRONTIER_RETRACT_OF_PUNCTURED_UNIVERSE THEN ASM_REWRITE_TAC[CONVEX_CBALL; BOUNDED_CBALL; INTERIOR_CBALL]);;
let SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE = 
prove (`!a r. &0 < r ==> sphere(a,r) retract_of ((:real^N) DELETE a)`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE_GEN THEN ASM_REWRITE_TAC[CENTRE_IN_BALL]);;
let RETRACT_OF_PCROSS = 
prove (`!s:real^M->bool s' t:real^N->bool t'. s retract_of s' /\ t retract_of t' ==> (s PCROSS t) retract_of (s' PCROSS t')`,
REPEAT GEN_TAC THEN REWRITE_TAC[PCROSS] THEN REWRITE_TAC[retract_of; retraction; SUBSET; FORALL_IN_IMAGE] THEN DISCH_THEN(CONJUNCTS_THEN2 (X_CHOOSE_THEN `f:real^M->real^M` STRIP_ASSUME_TAC) (X_CHOOSE_THEN `g:real^N->real^N` STRIP_ASSUME_TAC)) THEN EXISTS_TAC `\z. pastecart ((f:real^M->real^M) (fstcart z)) ((g:real^N->real^N) (sndcart z))` THEN REWRITE_TAC[FORALL_PASTECART; IN_ELIM_PASTECART_THM] THEN ASM_SIMP_TAC[FSTCART_PASTECART; SNDCART_PASTECART] THEN MATCH_MP_TAC CONTINUOUS_ON_PASTECART THEN CONJ_TAC THEN GEN_REWRITE_TAC LAND_CONV [GSYM o_DEF] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN SIMP_TAC[LINEAR_CONTINUOUS_ON; LINEAR_FSTCART; LINEAR_SNDCART] THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; FORALL_IN_GSPEC] THEN SIMP_TAC[FSTCART_PASTECART; SNDCART_PASTECART]);;
let LOCALLY_PATH_CONNECTED_SPHERE_GEN = 
prove (`!s:real^N->bool. bounded s /\ convex s ==> locally path_connected (relative_frontier s)`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `relative_interior(s:real^N->bool) = {}` THENL [UNDISCH_TAC `relative_interior(s:real^N->bool) = {}` THEN ASM_SIMP_TAC[RELATIVE_INTERIOR_EQ_EMPTY] THEN REWRITE_TAC[LOCALLY_EMPTY; RELATIVE_FRONTIER_EMPTY]; FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [GSYM MEMBER_NOT_EMPTY]) THEN DISCH_THEN(X_CHOOSE_TAC `a:real^N`) THEN MATCH_MP_TAC RETRACT_OF_LOCALLY_PATH_CONNECTED THEN EXISTS_TAC `(affine hull s) DELETE (a:real^N)` THEN ASM_SIMP_TAC[RELATIVE_FRONTIER_RETRACT_OF_PUNCTURED_AFFINE_HULL] THEN MATCH_MP_TAC LOCALLY_OPEN_SUBSET THEN EXISTS_TAC `affine hull s:real^N->bool` THEN SIMP_TAC[OPEN_IN_DELETE; OPEN_IN_REFL] THEN SIMP_TAC[CONVEX_IMP_LOCALLY_PATH_CONNECTED; AFFINE_IMP_CONVEX; AFFINE_AFFINE_HULL]]);;
let LOCALLY_CONNECTED_SPHERE_GEN = 
prove (`!s:real^N->bool. bounded s /\ convex s ==> locally connected (relative_frontier s)`,
let LOCALLY_PATH_CONNECTED_SPHERE = 
prove (`!a:real^N r. locally path_connected (sphere(a,r))`,
REPEAT GEN_TAC THEN MP_TAC(ISPEC `cball(a:real^N,r)` LOCALLY_PATH_CONNECTED_SPHERE_GEN) THEN MP_TAC(ISPECL [`a:real^N`; `r:real`] RELATIVE_FRONTIER_CBALL) THEN COND_CASES_TAC THEN ASM_SIMP_TAC[SPHERE_SING; LOCALLY_SING; PATH_CONNECTED_SING; BOUNDED_CBALL; CONVEX_CBALL]);;
let LOCALLY_CONNECTED_SPHERE = 
prove (`!a:real^N r. locally connected(sphere(a,r))`,
(* ------------------------------------------------------------------------- *) (* Extending a function into an ANR. *) (* ------------------------------------------------------------------------- *)
let NEIGHBOURHOOD_EXTENSION_INTO_ANR_LOCAL = 
prove (`!f:real^M->real^N c s t u. f continuous_on c /\ IMAGE f c SUBSET t /\ open u /\ t retract_of u /\ closed_in (subtopology euclidean s) c ==> ?v g. c SUBSET v /\ open_in (subtopology euclidean s) v /\ g continuous_on v /\ IMAGE g v SUBSET t /\ !x. x IN c ==> g x = f x`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`f:real^M->real^N`; `s:real^M->bool`; `c:real^M->bool`] TIETZE_UNBOUNDED) THEN ASM_REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `g:real^M->real^N` THEN STRIP_TAC THEN EXISTS_TAC `{x | x IN s /\ (g:real^M->real^N) x IN u}` THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN REWRITE_TAC[retraction] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^N->real^N` STRIP_ASSUME_TAC) THEN EXISTS_TAC `(r:real^N->real^N) o (g:real^M->real^N)` THEN FIRST_ASSUM(ASSUME_TAC o MATCH_MP CLOSED_IN_IMP_SUBSET) THEN REPEAT CONJ_TAC THENL [ASM SET_TAC[]; MATCH_MP_TAC CONTINUOUS_OPEN_IN_PREIMAGE THEN ASM_REWRITE_TAC[]; MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]; REWRITE_TAC[IMAGE_o] THEN ASM SET_TAC[]; REWRITE_TAC[o_THM] THEN ASM SET_TAC[]]);;
let NEIGHBOURHOOD_EXTENSION_INTO_ANR = 
prove (`!f:real^M->real^N s t u. f continuous_on s /\ IMAGE f s SUBSET t /\ open u /\ t retract_of u /\ closed s ==> ?v g. s SUBSET v /\ open v /\ g continuous_on v /\ IMAGE g v SUBSET t /\ !x. x IN s ==> g x = f x`,
REPEAT GEN_TAC THEN MP_TAC(ISPECL [`f:real^M->real^N`; `s:real^M->bool`; `(:real^M)`; `t:real^N->bool`; `u:real^N->bool`] NEIGHBOURHOOD_EXTENSION_INTO_ANR_LOCAL) THEN REWRITE_TAC[GSYM OPEN_IN; GSYM CLOSED_IN; SUBTOPOLOGY_UNIV]);;
(* ------------------------------------------------------------------------- *) (* Borsuk homotopy extension thorem. It's only this late so we can use the *) (* concept of retraction, essentially that the range is an ANR. *) (* ------------------------------------------------------------------------- *)
let BORSUK_HOMOTOPY_EXTENSION = 
prove (`!f:real^M->real^N g s t u v. closed s /\ closed t /\ u retract_of v /\ open v /\ f continuous_on t /\ IMAGE f t SUBSET u /\ g continuous_on s /\ IMAGE g s SUBSET u /\ homotopic_with (\x. T) (s,u) f g ==> ?g'. g' continuous_on t /\ IMAGE g' t SUBSET u /\ !x. x IN s ==> g'(x) = g(x)`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [homotopic_with]) THEN REWRITE_TAC[PCROSS] THEN DISCH_THEN(X_CHOOSE_THEN `h:real^(1,M)finite_sum->real^N` STRIP_ASSUME_TAC) THEN ABBREV_TAC `B = {pastecart (vec 0:real^1) (x:real^M) | x IN t} UNION {pastecart u x | u IN interval[vec 0,vec 1] /\ x IN s}` THEN SUBGOAL_THEN `closed(B:real^(1,M)finite_sum->bool)` ASSUME_TAC THENL [EXPAND_TAC "B" THEN MATCH_MP_TAC CLOSED_UNION THEN REWRITE_TAC[SET_RULE `{pastecart (vec 0) (x:real^M) | x IN t} = {pastecart u x | u IN {vec 0} /\ x IN t}`] THEN ASM_SIMP_TAC[REWRITE_RULE[PCROSS] CLOSED_PCROSS; CLOSED_SING; CLOSED_INTERVAL]; ALL_TAC] THEN SUBGOAL_THEN `?k:real^(1,M)finite_sum->real^N. k continuous_on (:real^(1,M)finite_sum) /\ (!x. x IN t ==> k(pastecart (vec 0) x) = f(x)) /\ (!u x. u IN interval[vec 0,vec 1] /\ x IN s ==> k(pastecart u x) = h(pastecart u x))` STRIP_ASSUME_TAC THENL [SUBGOAL_THEN `?k:real^(1,M)finite_sum->real^N. k continuous_on B /\ (!x. x IN t ==> k(pastecart (vec 0) x) = f(x)) /\ (!u x. u IN interval[vec 0,vec 1] /\ x IN s ==> k(pastecart u x) = h(pastecart u x))` STRIP_ASSUME_TAC THENL [EXISTS_TAC `\z. if fstcart z = vec 0 then f(sndcart z) else (h:real^(1,M)finite_sum->real^N) z` THEN REWRITE_TAC[FSTCART_PASTECART; SNDCART_PASTECART] THEN CONJ_TAC THENL [ALL_TAC; ASM_MESON_TAC[]] THEN EXPAND_TAC "B" THEN MATCH_MP_TAC CONTINUOUS_ON_CASES THEN REWRITE_TAC[SET_RULE `{pastecart (vec 0) (x:real^M) | x IN t} = {pastecart u x | u IN {vec 0} /\ x IN t}`] THEN ASM_SIMP_TAC[REWRITE_RULE[PCROSS] CLOSED_PCROSS; CLOSED_SING; CLOSED_INTERVAL] THEN REWRITE_TAC[TAUT `(a \/ b /\ c ==> d) <=> (a ==> d) /\ (b ==> c ==> d)`; IMP_CONJ; FORALL_AND_THM; FORALL_IN_GSPEC; IN_ELIM_PASTECART_THM] THEN ASM_SIMP_TAC[FSTCART_PASTECART; SNDCART_PASTECART; IN_SING] THEN GEN_REWRITE_TAC LAND_CONV [GSYM o_DEF] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN SIMP_TAC[LINEAR_CONTINUOUS_ON; LINEAR_SNDCART] THEN MATCH_MP_TAC CONTINUOUS_ON_SUBSET THEN EXISTS_TAC `t:real^M->bool` THEN ASM_REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; FORALL_IN_GSPEC] THEN SIMP_TAC[SNDCART_PASTECART]; MP_TAC(ISPECL [`k:real^(1,M)finite_sum->real^N`; `(:real^(1,M)finite_sum)`; `B:real^(1,M)finite_sum->bool`] TIETZE_UNBOUNDED) THEN ASM_REWRITE_TAC[SUBTOPOLOGY_UNIV; GSYM CLOSED_IN] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `g:real^(1,M)finite_sum->real^N` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]]; ALL_TAC] THEN FIRST_ASSUM(ASSUME_TAC o MATCH_MP RETRACT_OF_IMP_SUBSET) THEN ABBREV_TAC `V = {x | x IN UNIV /\ (k:real^(1,M)finite_sum->real^N) x IN v}` THEN SUBGOAL_THEN `open(V:real^(1,M)finite_sum->bool)` ASSUME_TAC THENL [EXPAND_TAC "V" THEN MATCH_MP_TAC CONTINUOUS_OPEN_PREIMAGE THEN ASM_REWRITE_TAC[OPEN_UNIV]; ALL_TAC] THEN SUBGOAL_THEN `(B:real^(1,M)finite_sum->bool) SUBSET V` ASSUME_TAC THENL [MAP_EVERY EXPAND_TAC ["B";
"V"] THEN REWRITE_TAC[UNION_SUBSET] THEN REWRITE_TAC[SUBSET; FORALL_IN_GSPEC] THEN ASM_SIMP_TAC[IN_ELIM_THM; IN_UNIV] THEN ASM SET_TAC[]; ALL_TAC] THEN ABBREV_TAC `s' = {x | ?u. u IN interval[vec 0,vec 1] /\ ~((pastecart (u:real^1) (x:real^M)) IN V)}` THEN SUBGOAL_THEN `closed(s':real^M->bool)` ASSUME_TAC THENL [EXPAND_TAC "s'" THEN REWRITE_TAC[SET_RULE `~(x IN s) <=> x IN (UNIV DIFF s)`] THEN MATCH_MP_TAC CLOSED_COMPACT_PROJECTION THEN ASM_REWRITE_TAC[GSYM OPEN_CLOSED; COMPACT_INTERVAL]; ALL_TAC] THEN MP_TAC(ISPECL [`s:real^M->bool`; `s':real^M->bool`; `vec 1:real^1`; `vec 0:real^1`] URYSOHN) THEN ASM_REWRITE_TAC[] THEN ANTS_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN ONCE_REWRITE_TAC[SEGMENT_SYM] THEN REWRITE_TAC[SEGMENT_1; DROP_VEC; REAL_POS] THEN DISCH_THEN(X_CHOOSE_THEN `a:real^M->real^1` STRIP_ASSUME_TAC) THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN REWRITE_TAC[retraction; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `r:real^N->real^N` THEN STRIP_TAC THEN EXISTS_TAC `(r:real^N->real^N) o (\x. (k:real^(1,M)finite_sum->real^N) (pastecart (a x) x))` THEN REWRITE_TAC[o_THM; IMAGE_o] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THENL [GEN_REWRITE_TAC LAND_CONV [GSYM o_DEF] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_PASTECART THEN REWRITE_TAC[CONTINUOUS_ON_ID] THEN ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]; ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]]; FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]]);;
let NULLHOMOTOPIC_INTO_ANR_EXTENSION = 
prove (`!f:real^M->real^N s t u. closed s /\ f continuous_on s /\ ~(s = {}) /\ IMAGE f s SUBSET t /\ open u /\ t retract_of u ==> ((?c. homotopic_with (\x. T) (s,t) f (\x. c)) <=> (?g. g continuous_on (:real^M) /\ IMAGE g (:real^M) SUBSET t /\ !x. x IN s ==> g x = f x))`,
REPEAT STRIP_TAC THEN EQ_TAC THEN STRIP_TAC THENL [MATCH_MP_TAC BORSUK_HOMOTOPY_EXTENSION THEN ONCE_REWRITE_TAC[HOMOTOPIC_WITH_SYM] THEN MAP_EVERY EXISTS_TAC [`(\x. c):real^M->real^N`; `u:real^N->bool`] THEN ASM_REWRITE_TAC[CLOSED_UNIV; CONTINUOUS_ON_CONST] THEN FIRST_X_ASSUM(MP_TAC o MATCH_MP HOMOTOPIC_WITH_IMP_SUBSET) THEN ASM SET_TAC[]; MP_TAC(ISPECL [`g:real^M->real^N`; `(:real^M)`; `t:real^N->bool`] NULLHOMOTOPIC_FROM_CONTRACTIBLE) THEN ASM_REWRITE_TAC[CONTRACTIBLE_UNIV] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `c:real^N` THEN DISCH_TAC THEN MATCH_MP_TAC HOMOTOPIC_WITH_EQ THEN MAP_EVERY EXISTS_TAC [`g:real^M->real^N`; `(\x. c):real^M->real^N`] THEN ASM_SIMP_TAC[] THEN MATCH_MP_TAC HOMOTOPIC_WITH_SUBSET_LEFT THEN EXISTS_TAC `(:real^M)` THEN ASM_REWRITE_TAC[SUBSET_UNIV]]);;
let NULLHOMOTOPIC_INTO_RELATIVE_FRONTIER_EXTENSION = 
prove (`!f:real^M->real^N s t. closed s /\ f continuous_on s /\ ~(s = {}) /\ IMAGE f s SUBSET relative_frontier t /\ convex t /\ bounded t ==> ((?c. homotopic_with (\x. T) (s,relative_frontier t) f (\x. c)) <=> (?g. g continuous_on (:real^M) /\ IMAGE g (:real^M) SUBSET relative_frontier t /\ !x. x IN s ==> g x = f x))`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC NULLHOMOTOPIC_INTO_ANR_EXTENSION THEN MP_TAC(ISPEC `t:real^N->bool` ANR_RELATIVE_FRONTIER_CONVEX) THEN ASM_REWRITE_TAC[]);;
let NULLHOMOTOPIC_INTO_SPHERE_EXTENSION = 
prove (`!f:real^M->real^N s a r. closed s /\ f continuous_on s /\ ~(s = {}) /\ IMAGE f s SUBSET sphere(a,r) ==> ((?c. homotopic_with (\x. T) (s,sphere(a,r)) f (\x. c)) <=> (?g. g continuous_on (:real^M) /\ IMAGE g (:real^M) SUBSET sphere(a,r) /\ !x. x IN s ==> g x = f x))`,
REPEAT GEN_TAC THEN MP_TAC(ISPECL [`a:real^N`; `r:real`] RELATIVE_FRONTIER_CBALL) THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[] THENL [ASM_SIMP_TAC[SPHERE_SING] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC(TAUT `p /\ q ==> (p <=> q)`) THEN CONJ_TAC THENL [EXISTS_TAC `a:real^N` THEN SIMP_TAC[HOMOTOPIC_WITH; PCROSS] THEN EXISTS_TAC `\y:real^(1,M)finite_sum. (a:real^N)`; EXISTS_TAC `(\x. a):real^M->real^N`] THEN REWRITE_TAC[CONTINUOUS_ON_CONST] THEN ASM SET_TAC[]; DISCH_THEN(SUBST1_TAC o SYM) THEN STRIP_TAC THEN MATCH_MP_TAC NULLHOMOTOPIC_INTO_RELATIVE_FRONTIER_EXTENSION THEN ASM_REWRITE_TAC[CONVEX_CBALL; BOUNDED_CBALL]]);;
let ABSOLUTE_RETRACT_CONTRACTIBLE_ANR = 
prove (`!s t. closed s /\ contractible s /\ ~(s = {}) /\ s retract_of t /\ open t ==> s retract_of (:real^N)`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [contractible]) THEN ONCE_REWRITE_TAC[HOMOTOPIC_WITH_SYM] THEN DISCH_THEN(X_CHOOSE_TAC `a:real^N`) THEN REWRITE_TAC[retract_of; retraction; SUBSET_UNIV] THEN MATCH_MP_TAC BORSUK_HOMOTOPY_EXTENSION THEN MAP_EVERY EXISTS_TAC [`(\x. a):real^N->real^N`; `t:real^N->bool`] THEN ASM_SIMP_TAC[CLOSED_UNIV; IMAGE_ID] THEN REWRITE_TAC[SUBSET_REFL; CONTINUOUS_ON_ID; CONTINUOUS_ON_CONST] THEN FIRST_X_ASSUM(MP_TAC o MATCH_MP HOMOTOPIC_WITH_IMP_SUBSET) THEN ASM SET_TAC[]);;
(* ------------------------------------------------------------------------- *) (* Preservation of fixpoints under (more general notion of) retraction. *) (* ------------------------------------------------------------------------- *)
let INVERTIBLE_FIXPOINT_PROPERTY = 
prove (`!s:real^M->bool t:real^N->bool i r. i continuous_on t /\ IMAGE i t SUBSET s /\ r continuous_on s /\ IMAGE r s SUBSET t /\ (!y. y IN t ==> (r(i(y)) = y)) ==> (!f. f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ (f x = x)) ==> !g. g continuous_on t /\ IMAGE g t SUBSET t ==> ?y. y IN t /\ (g y = y)`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `(i:real^N->real^M) o (g:real^N->real^N) o (r:real^M->real^N)`) THEN ANTS_TAC THENL [ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; CONTINUOUS_ON_COMPOSE; IMAGE_SUBSET; SUBSET_TRANS; IMAGE_o]; RULE_ASSUM_TAC(REWRITE_RULE[SUBSET; FORALL_IN_IMAGE]) THEN REWRITE_TAC[o_THM] THEN ASM_MESON_TAC[]]);;
let HOMEOMORPHIC_FIXPOINT_PROPERTY = 
prove (`!s t. s homeomorphic t ==> ((!f. f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ (f x = x)) <=> (!g. g continuous_on t /\ IMAGE g t SUBSET t ==> ?y. y IN t /\ (g y = y)))`,
REWRITE_TAC[homeomorphic; homeomorphism] THEN REPEAT STRIP_TAC THEN EQ_TAC THEN MATCH_MP_TAC INVERTIBLE_FIXPOINT_PROPERTY THEN ASM_MESON_TAC[SUBSET_REFL]);;
let RETRACT_FIXPOINT_PROPERTY = 
prove (`!s t:real^N->bool. t retract_of s /\ (!f. f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ (f x = x)) ==> !g. g continuous_on t /\ IMAGE g t SUBSET t ==> ?y. y IN t /\ (g y = y)`,
REPEAT GEN_TAC THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN MATCH_MP_TAC INVERTIBLE_FIXPOINT_PROPERTY THEN EXISTS_TAC `\x:real^N. x` THEN REWRITE_TAC[CONTINUOUS_ON_ID] THEN POP_ASSUM MP_TAC THEN REWRITE_TAC[retract_of] THEN MATCH_MP_TAC MONO_EXISTS THEN GEN_TAC THEN REWRITE_TAC[retraction] THEN REWRITE_TAC[SUBSET; FORALL_IN_IMAGE]);;
(* ------------------------------------------------------------------------- *) (* So the Brouwer theorem for any set with nonempty interior. *) (* ------------------------------------------------------------------------- *)
let BROUWER_WEAK = 
prove (`!f:real^N->real^N s. compact s /\ convex s /\ ~(interior s = {}) /\ f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ f x = x`,
GEN_TAC THEN ONCE_REWRITE_TAC [TAUT `a /\ b /\ c /\ d ==> e <=> a /\ b /\ c ==> d ==> e`] THEN GEN_TAC THEN STRIP_TAC THEN MP_TAC(ISPECL [`interval[vec 0:real^N,vec 1]`; `s:real^N->bool`] HOMEOMORPHIC_CONVEX_COMPACT) THEN ANTS_TAC THENL [ASM_REWRITE_TAC[CONVEX_INTERVAL; COMPACT_INTERVAL] THEN REWRITE_TAC[INTERIOR_CLOSED_INTERVAL; INTERVAL_EQ_EMPTY] THEN MESON_TAC[VEC_COMPONENT; REAL_ARITH `~(&1 <= &0)`]; DISCH_THEN(MP_TAC o MATCH_MP HOMEOMORPHIC_FIXPOINT_PROPERTY) THEN REWRITE_TAC[BROUWER_CUBE] THEN SIMP_TAC[]]);;
(* ------------------------------------------------------------------------- *) (* And in particular for a closed ball. *) (* ------------------------------------------------------------------------- *)
let BROUWER_BALL = 
prove (`!f:real^N->real^N a e. &0 < e /\ f continuous_on cball(a,e) /\ IMAGE f (cball(a,e)) SUBSET (cball(a,e)) ==> ?x. x IN cball(a,e) /\ (f x = x)`,
(* ------------------------------------------------------------------------- *) (* Still more general form; could derive this directly without using the *) (* rather involved HOMEOMORPHIC_CONVEX_COMPACT theorem, just using *) (* a scaling and translation to put the set inside the unit cube. *) (* ------------------------------------------------------------------------- *)
let BROUWER = 
prove (`!f:real^N->real^N s. compact s /\ convex s /\ ~(s = {}) /\ f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ f x = x`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `?e. &0 < e /\ s SUBSET cball(vec 0:real^N,e)` STRIP_ASSUME_TAC THENL [REWRITE_TAC[SUBSET; IN_CBALL; NORM_ARITH `dist(vec 0,x) = norm(x)`] THEN ASM_MESON_TAC[BOUNDED_POS; COMPACT_IMP_BOUNDED]; ALL_TAC] THEN SUBGOAL_THEN `?x:real^N. x IN cball(vec 0,e) /\ (f o closest_point s) x = x` MP_TAC THENL [MATCH_MP_TAC BROUWER_BALL THEN ASM_REWRITE_TAC[] THEN CONJ_TAC THENL [REWRITE_TAC[ETA_AX] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN ASM_SIMP_TAC[CONTINUOUS_ON_CLOSEST_POINT; COMPACT_IMP_CLOSED] THEN MATCH_MP_TAC CONTINUOUS_ON_SUBSET THEN EXISTS_TAC `s:real^N->bool` THEN ASM_REWRITE_TAC[SUBSET; FORALL_IN_IMAGE]; REWRITE_TAC[SUBSET; FORALL_IN_IMAGE] THEN X_GEN_TAC `x:real^N` THEN REPEAT STRIP_TAC THEN REPEAT(FIRST_X_ASSUM(MATCH_MP_TAC o REWRITE_RULE[SUBSET])) THEN REWRITE_TAC[o_THM; IN_IMAGE] THEN EXISTS_TAC `closest_point s x:real^N` THEN ASM_SIMP_TAC[COMPACT_IMP_CLOSED; CLOSEST_POINT_IN_SET]] THEN ASM_SIMP_TAC[COMPACT_IMP_CLOSED; CLOSEST_POINT_IN_SET]; MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `x:real^N` THEN REWRITE_TAC[o_THM] THEN STRIP_TAC THEN RULE_ASSUM_TAC(REWRITE_RULE[SUBSET; FORALL_IN_IMAGE]) THEN ASM_MESON_TAC[CLOSEST_POINT_SELF; CLOSEST_POINT_IN_SET; COMPACT_IMP_CLOSED]]);;
(* ------------------------------------------------------------------------- *) (* So we get the no-retraction theorem, first for a ball, then more general. *) (* ------------------------------------------------------------------------- *)
let NO_RETRACTION_CBALL = 
prove (`!a:real^N e. &0 < e ==> ~(sphere(a,e) retract_of cball(a,e))`,
REPEAT GEN_TAC THEN DISCH_TAC THEN DISCH_THEN(MP_TAC o MATCH_MP(REWRITE_RULE[IMP_CONJ] RETRACT_FIXPOINT_PROPERTY)) THEN ASM_SIMP_TAC[BROUWER_BALL] THEN REWRITE_TAC[NOT_FORALL_THM] THEN EXISTS_TAC `\x:real^N. &2 % a - x` THEN REWRITE_TAC[NOT_IMP] THEN SIMP_TAC[CONTINUOUS_ON_SUB; CONTINUOUS_ON_ID; CONTINUOUS_ON_CONST] THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; IN_SPHERE] THEN SIMP_TAC[dist; VECTOR_ARITH `a - (&2 % a - x) = --(a - x)`; NORM_NEG] THEN REWRITE_TAC[VECTOR_ARITH `(&2 % a - y = y) <=> (a - y = vec 0)`] THEN ASM_MESON_TAC[NORM_0; REAL_LT_REFL]);;
let FRONTIER_SUBSET_RETRACTION = 
prove (`!s:real^N->bool t r. bounded s /\ frontier s SUBSET t /\ r continuous_on (closure s) /\ IMAGE r s SUBSET t /\ (!x. x IN t ==> r x = x) ==> s SUBSET t`,
ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN REWRITE_TAC[SET_RULE `~(s SUBSET t) <=> ?x. x IN s /\ ~(x IN t)`] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY; LEFT_IMP_EXISTS_THM] THEN REPLICATE_TAC 3 GEN_TAC THEN X_GEN_TAC `a:real^N` THEN REPEAT STRIP_TAC THEN ABBREV_TAC `q = \z:real^N. if z IN closure s then r(z) else z` THEN SUBGOAL_THEN `(q:real^N->real^N) continuous_on closure(s) UNION closure((:real^N) DIFF s)` MP_TAC THENL [EXPAND_TAC "q" THEN MATCH_MP_TAC CONTINUOUS_ON_CASES THEN ASM_REWRITE_TAC[CLOSED_CLOSURE; CONTINUOUS_ON_ID] THEN REWRITE_TAC[TAUT `p /\ ~p <=> F`] THEN X_GEN_TAC `z:real^N` THEN REWRITE_TAC[CLOSURE_COMPLEMENT; IN_DIFF; IN_UNIV] THEN STRIP_TAC THEN FIRST_X_ASSUM MATCH_MP_TAC THEN RULE_ASSUM_TAC(REWRITE_RULE[SUBSET; frontier; IN_DIFF]) THEN ASM_MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `closure(s) UNION closure((:real^N) DIFF s) = (:real^N)` SUBST1_TAC THENL [MATCH_MP_TAC(SET_RULE `s SUBSET closure s /\ t SUBSET closure t /\ s UNION t = UNIV ==> closure s UNION closure t = UNIV`) THEN REWRITE_TAC[CLOSURE_SUBSET] THEN SET_TAC[]; DISCH_TAC] THEN FIRST_ASSUM(X_CHOOSE_THEN `B:real` STRIP_ASSUME_TAC o SPEC `a:real^N` o MATCH_MP BOUNDED_SUBSET_BALL o MATCH_MP BOUNDED_CLOSURE) THEN SUBGOAL_THEN `!x. ~((q:real^N->real^N) x = a)` ASSUME_TAC THENL [GEN_TAC THEN EXPAND_TAC "q" THEN COND_CASES_TAC THENL [ASM_CASES_TAC `(x:real^N) IN s` THENL [ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `(x:real^N) IN t` (fun th -> ASM_MESON_TAC[th]) THEN UNDISCH_TAC `frontier(s:real^N->bool) SUBSET t` THEN REWRITE_TAC[SUBSET; frontier; IN_DIFF] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_MESON_TAC[SUBSET; INTERIOR_SUBSET]; ASM_MESON_TAC[SUBSET; INTERIOR_SUBSET; CLOSURE_SUBSET]]; ALL_TAC] THEN MP_TAC(ISPECL [`a:real^N`; `B:real`] NO_RETRACTION_CBALL) THEN ASM_REWRITE_TAC[retract_of; GSYM FRONTIER_CBALL] THEN EXISTS_TAC `(\y. a + B / norm(y - a) % (y - a)) o (q:real^N->real^N)` THEN REWRITE_TAC[retraction; FRONTIER_SUBSET_EQ; CLOSED_CBALL] THEN REWRITE_TAC[FRONTIER_CBALL; SUBSET; FORALL_IN_IMAGE; FORALL_IN_GSPEC] THEN REWRITE_TAC[IN_SPHERE; DIST_0] THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THENL [ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]; ALL_TAC] THEN MATCH_MP_TAC CONTINUOUS_ON_ADD THEN REWRITE_TAC[CONTINUOUS_ON_CONST] THEN MATCH_MP_TAC CONTINUOUS_ON_MUL THEN SIMP_TAC[CONTINUOUS_ON_SUB; CONTINUOUS_ON_ID; CONTINUOUS_ON_CONST] THEN REWRITE_TAC[o_DEF; real_div; LIFT_CMUL] THEN MATCH_MP_TAC CONTINUOUS_ON_CMUL THEN MATCH_MP_TAC(REWRITE_RULE[o_DEF] CONTINUOUS_ON_INV) THEN ASM_REWRITE_TAC[FORALL_IN_IMAGE; NORM_EQ_0; VECTOR_SUB_EQ] THEN SUBGOAL_THEN `(\x:real^N. lift(norm(x - a))) = (lift o norm) o (\x. x - a)` SUBST1_TAC THENL [REWRITE_TAC[FUN_EQ_THM; o_THM]; ALL_TAC] THEN MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN ASM_SIMP_TAC[CONTINUOUS_ON_SUB; CONTINUOUS_ON_ID; CONTINUOUS_ON_CONST] THEN REWRITE_TAC[CONTINUOUS_ON_LIFT_NORM]; REWRITE_TAC[o_THM; NORM_MUL; REAL_ABS_DIV; REAL_ABS_NORM; NORM_ARITH `dist(a,a + b) = norm b`] THEN ASM_SIMP_TAC[REAL_DIV_RMUL; VECTOR_SUB_EQ; NORM_EQ_0] THEN ASM_REAL_ARITH_TAC; X_GEN_TAC `x:real^N` THEN DISCH_TAC THEN REWRITE_TAC[o_THM] THEN EXPAND_TAC "q" THEN REWRITE_TAC[] THEN COND_CASES_TAC THENL [RULE_ASSUM_TAC(REWRITE_RULE[SUBSET; IN_BALL]) THEN ASM_MESON_TAC[REAL_LT_REFL]; REWRITE_TAC[NORM_ARITH `norm(x - a) = dist(a,x)`] THEN ASM_SIMP_TAC[REAL_DIV_REFL; REAL_LT_IMP_NZ; VECTOR_MUL_LID] THEN VECTOR_ARITH_TAC]]);;
let NO_RETRACTION_FRONTIER_BOUNDED = 
prove (`!s:real^N->bool. bounded s /\ ~(interior s = {}) ==> ~((frontier s) retract_of s)`,
GEN_TAC THEN STRIP_TAC THEN REWRITE_TAC[retract_of; retraction] THEN REWRITE_TAC[FRONTIER_SUBSET_EQ] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^N->real^N` STRIP_ASSUME_TAC) THEN MP_TAC(ISPECL [`s:real^N->bool`; `frontier s:real^N->bool`; `r:real^N->real^N`] FRONTIER_SUBSET_RETRACTION) THEN ASM_SIMP_TAC[CLOSURE_CLOSED; SUBSET_REFL] THEN REWRITE_TAC[frontier] THEN MP_TAC(ISPEC `s:real^N->bool` INTERIOR_SUBSET) THEN ASM SET_TAC[]);;
let COMPACT_SUBSET_FRONTIER_RETRACTION = 
prove (`!f:real^N->real^N s. compact s /\ f continuous_on s /\ (!x. x IN frontier s ==> f x = x) ==> s SUBSET IMAGE f s`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`s UNION (IMAGE f s):real^N->bool`; `vec 0:real^N`] BOUNDED_SUBSET_BALL) THEN ASM_SIMP_TAC[BOUNDED_UNION; COMPACT_IMP_BOUNDED; COMPACT_CONTINUOUS_IMAGE; UNION_SUBSET] THEN DISCH_THEN(X_CHOOSE_THEN `r:real` STRIP_ASSUME_TAC) THEN ABBREV_TAC `g = \x:real^N. if x IN s then f(x) else x` THEN SUBGOAL_THEN `(g:real^N->real^N) continuous_on (:real^N)` ASSUME_TAC THENL [SUBGOAL_THEN `(:real^N) = s UNION closure((:real^N) DIFF s)` SUBST1_TAC THENL [MATCH_MP_TAC(SET_RULE `UNIV DIFF s SUBSET t ==> UNIV = s UNION t`) THEN REWRITE_TAC[CLOSURE_SUBSET]; ALL_TAC] THEN EXPAND_TAC "g" THEN MATCH_MP_TAC CONTINUOUS_ON_CASES THEN ASM_SIMP_TAC[CLOSED_CLOSURE; CONTINUOUS_ON_ID; COMPACT_IMP_CLOSED] THEN REWRITE_TAC[CLOSURE_COMPLEMENT; IN_DIFF; IN_UNIV] THEN REWRITE_TAC[TAUT `p /\ ~p <=> F`] THEN REPEAT STRIP_TAC THEN FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[frontier; IN_DIFF] THEN ASM_SIMP_TAC[CLOSURE_CLOSED; COMPACT_IMP_CLOSED]; ALL_TAC] THEN REWRITE_TAC[SUBSET] THEN X_GEN_TAC `p:real^N` THEN DISCH_TAC THEN SUBGOAL_THEN `?h:real^N->real^N. retraction (UNIV DELETE p,sphere(vec 0,r)) h` STRIP_ASSUME_TAC THENL [REWRITE_TAC[GSYM retract_of] THEN MATCH_MP_TAC SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE_GEN THEN ASM SET_TAC[]; ALL_TAC] THEN MP_TAC(ISPECL [`vec 0:real^N`; `r:real`] NO_RETRACTION_CBALL) THEN ASM_REWRITE_TAC[retract_of; NOT_EXISTS_THM] THEN DISCH_THEN(MP_TAC o SPEC `(h:real^N->real^N) o (g:real^N->real^N)`) THEN ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN DISCH_TAC THEN REWRITE_TAC[] THEN REWRITE_TAC[retraction] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retraction]) THEN SIMP_TAC[SUBSET; IN_SPHERE; IN_CBALL; REAL_EQ_IMP_LE] THEN REWRITE_TAC[FORALL_IN_IMAGE; IN_DELETE; IN_UNIV; o_THM] THEN STRIP_TAC THEN SUBGOAL_THEN `!x. x IN cball (vec 0,r) ==> ~((g:real^N->real^N) x = p)` ASSUME_TAC THENL [X_GEN_TAC `x:real^N` THEN DISCH_TAC THEN EXPAND_TAC "g" THEN COND_CASES_TAC THEN ASM SET_TAC[]; ALL_TAC] THEN ASM_SIMP_TAC[] THEN REPEAT STRIP_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM_REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; IN_UNIV; IN_DELETE]; SUBGOAL_THEN `(g:real^N->real^N) x = x` (fun th -> ASM_SIMP_TAC[th]) THEN EXPAND_TAC "g" THEN COND_CASES_TAC THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[IN_BALL; REAL_LT_REFL; SUBSET]]);;
let NOT_ABSOLUTE_RETRACT_COBOUNDED = 
prove (`!s. bounded s /\ ((:real^N) DIFF s) retract_of (:real^N) ==> s = {}`,
GEN_TAC THEN DISCH_TAC THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> F) ==> s = {}`) THEN X_GEN_TAC `a:real^N` THEN POP_ASSUM MP_TAC THEN GEOM_ORIGIN_TAC `a:real^N` THEN REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `vec 0:real^N` o MATCH_MP BOUNDED_SUBSET_BALL) THEN DISCH_THEN(X_CHOOSE_THEN `r:real` STRIP_ASSUME_TAC) THEN FIRST_ASSUM(MP_TAC o SPEC `vec 0:real^N` o MATCH_MP NO_RETRACTION_CBALL) THEN REWRITE_TAC[] THEN MATCH_MP_TAC RETRACT_OF_SUBSET THEN EXISTS_TAC `(:real^N)` THEN SIMP_TAC[SUBSET_UNIV; SPHERE_SUBSET_CBALL] THEN MATCH_MP_TAC RETRACT_OF_TRANS THEN EXISTS_TAC `(:real^N) DIFF s` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC RETRACT_OF_SUBSET THEN EXISTS_TAC `(:real^N) DELETE (vec 0)` THEN ASM_SIMP_TAC[SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [SUBSET]) THEN REWRITE_TAC[SUBSET; IN_BALL; IN_SPHERE; IN_DIFF; IN_UNIV] THEN MESON_TAC[REAL_LT_REFL]);;
(* ------------------------------------------------------------------------- *) (* Similarly we get noncontractibility of a non-trivial sphere. *) (* ------------------------------------------------------------------------- *)
let CONTRACTIBLE_SPHERE = 
prove (`!a:real^N r. contractible(sphere(a,r)) <=> r <= &0`,
REPEAT GEN_TAC THEN REWRITE_TAC[contractible; GSYM REAL_NOT_LT] THEN REWRITE_TAC[NULLHOMOTOPIC_FROM_SPHERE_EXTENSION] THEN ASM_CASES_TAC `&0 < r` THEN ASM_REWRITE_TAC[] THENL [FIRST_ASSUM(MP_TAC o ISPEC `a:real^N` o MATCH_MP NO_RETRACTION_CBALL) THEN SIMP_TAC[retract_of; retraction; SPHERE_SUBSET_CBALL]; RULE_ASSUM_TAC(REWRITE_RULE[REAL_NOT_LT]) THEN EXISTS_TAC `\x:real^N. x` THEN REWRITE_TAC[CONTINUOUS_ON_ID; IMAGE_ID] THEN REWRITE_TAC[SUBSET; IN_CBALL; IN_SPHERE; IN_ELIM_THM] THEN POP_ASSUM MP_TAC THEN NORM_ARITH_TAC]);;
(* ------------------------------------------------------------------------- *) (* We also get fixpoint properties for suitable ANRs. *) (* ------------------------------------------------------------------------- *)
let BROUWER_INESSENTIAL_ANR = 
prove (`!f:real^N->real^N s t. compact s /\ ~(s = {}) /\ s retract_of t /\ open t /\ f continuous_on s /\ IMAGE f s SUBSET s /\ (?a. homotopic_with (\x. T) (s,s) f (\x. a)) ==> ?x. x IN s /\ f x = x`,
ONCE_REWRITE_TAC[HOMOTOPIC_WITH_SYM] THEN REPEAT STRIP_TAC THEN FIRST_ASSUM(X_CHOOSE_TAC `r:real` o SPEC `vec 0:real^N` o MATCH_MP BOUNDED_SUBSET_CBALL o MATCH_MP COMPACT_IMP_BOUNDED) THEN MP_TAC(ISPECL [`(\x. a):real^N->real^N`; `f:real^N->real^N`; `s:real^N->bool`; `cball(vec 0:real^N,r)`; `s:real^N->bool`; `t:real^N->bool`] BORSUK_HOMOTOPY_EXTENSION) THEN ASM_SIMP_TAC[COMPACT_IMP_CLOSED; CONTINUOUS_ON_CONST; CLOSED_CBALL] THEN FIRST_X_ASSUM(ASSUME_TAC o MATCH_MP HOMOTOPIC_WITH_IMP_SUBSET) THEN ANTS_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `g:real^N->real^N` STRIP_ASSUME_TAC) THEN MP_TAC(ISPECL [`g:real^N->real^N`; `cball(vec 0:real^N,r)`] BROUWER) THEN ASM_SIMP_TAC[COMPACT_CBALL; CONVEX_CBALL; CBALL_EQ_EMPTY] THEN ASM_SIMP_TAC[REAL_ARITH `&0 < r ==> ~(r < &0)`] THEN ASM SET_TAC[]);;
let BROUWER_CONTRACTIBLE_ANR = 
prove (`!f:real^N->real^N s t. compact s /\ contractible s /\ ~(s = {}) /\ s retract_of t /\ open t /\ f continuous_on s /\ IMAGE f s SUBSET s ==> ?x. x IN s /\ f x = x`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC BROUWER_INESSENTIAL_ANR THEN EXISTS_TAC `t:real^N->bool` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC NULLHOMOTOPIC_FROM_CONTRACTIBLE THEN ASM_REWRITE_TAC[]);;
let FIXED_POINT_INESSENTIAL_SPHERE_MAP = 
prove (`!f a:real^N r c. &0 < r /\ homotopic_with (\x. T) (sphere(a,r),sphere(a,r)) f (\x. c) ==> ?x. x IN sphere(a,r) /\ f x = x`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC BROUWER_INESSENTIAL_ANR THEN EXISTS_TAC `(:real^N) DELETE a` THEN ASM_SIMP_TAC[SPHERE_RETRACT_OF_PUNCTURED_UNIVERSE] THEN SIMP_TAC[SPHERE_EQ_EMPTY; COMPACT_SPHERE; OPEN_DELETE; OPEN_UNIV] THEN FIRST_ASSUM(STRIP_ASSUME_TAC o MATCH_MP HOMOTOPIC_WITH_IMP_CONTINUOUS) THEN FIRST_ASSUM(STRIP_ASSUME_TAC o MATCH_MP HOMOTOPIC_WITH_IMP_SUBSET) THEN ASM_SIMP_TAC[REAL_NOT_LT; REAL_LT_IMP_LE] THEN ASM_MESON_TAC[]);;
(* ------------------------------------------------------------------------- *) (* Some other related fixed-point theorems. *) (* ------------------------------------------------------------------------- *)
let SCHAUDER_PROJECTION = 
prove (`!s:real^N->bool e. compact s /\ &0 < e ==> ?t f. FINITE t /\ t SUBSET s /\ f continuous_on s /\ IMAGE f s SUBSET (convex hull t) /\ (!x. x IN s ==> norm(f x - x) < e)`,
REPEAT STRIP_TAC THEN FIRST_ASSUM (MP_TAC o SPEC `e:real` o MATCH_MP COMPACT_IMP_TOTALLY_BOUNDED) THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `t:real^N->bool` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN ABBREV_TAC `g = \p x:real^N. max (&0) (e - norm(x - p))` THEN SUBGOAL_THEN `!x. x IN s ==> &0 < sum t (\p. (g:real^N->real^N->real) p x)` ASSUME_TAC THENL [REPEAT STRIP_TAC THEN MATCH_MP_TAC SUM_POS_LT THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "g" THEN REWRITE_TAC[REAL_ARITH `&0 <= max (&0) b`] THEN REWRITE_TAC[REAL_ARITH `&0 < max (&0) b <=> &0 < b`; REAL_SUB_LT] THEN UNDISCH_TAC `s SUBSET UNIONS (IMAGE (\x:real^N. ball(x,e)) t)` THEN REWRITE_TAC[SUBSET; UNIONS_IMAGE; IN_BALL; IN_ELIM_THM] THEN DISCH_THEN(MP_TAC o SPEC `x:real^N`) THEN ASM_REWRITE_TAC[dist; NORM_SUB]; ALL_TAC] THEN EXISTS_TAC `(\x. inv(sum t (\p. g p x)) % vsum t (\p. g p x % p)):real^N->real^N` THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_MUL THEN REWRITE_TAC[o_DEF] THEN CONJ_TAC THENL [MATCH_MP_TAC(REWRITE_RULE[o_DEF] CONTINUOUS_ON_INV) THEN ASM_SIMP_TAC[REAL_LT_IMP_NZ; LIFT_SUM; o_DEF]; ALL_TAC] THEN MATCH_MP_TAC CONTINUOUS_ON_VSUM THEN ASM_REWRITE_TAC[] THEN X_GEN_TAC `y:real^N` THEN DISCH_TAC THENL [ALL_TAC; MATCH_MP_TAC CONTINUOUS_ON_MUL] THEN REWRITE_TAC[o_DEF; CONTINUOUS_ON_CONST] THEN EXPAND_TAC "g" THEN (SUBGOAL_THEN `(\x. lift (max (&0) (e - norm (x - y:real^N)))) = (\x. (lambda i. max (lift(&0)$i) (lift(e - norm (x - y))$i)))` SUBST1_TAC THENL [SIMP_TAC[CART_EQ; LAMBDA_BETA; FUN_EQ_THM] THEN REWRITE_TAC[DIMINDEX_1; FORALL_1; GSYM drop; LIFT_DROP]; MATCH_MP_TAC CONTINUOUS_ON_MAX] THEN REWRITE_TAC[CONTINUOUS_ON_CONST; LIFT_SUB] THEN MATCH_MP_TAC CONTINUOUS_ON_SUB THEN REWRITE_TAC[CONTINUOUS_ON_CONST] THEN REWRITE_TAC[ONCE_REWRITE_RULE[NORM_SUB] (GSYM dist)] THEN REWRITE_TAC[REWRITE_RULE[o_DEF] CONTINUOUS_ON_LIFT_DIST]); REWRITE_TAC[SUBSET; FORALL_IN_IMAGE; GSYM VSUM_LMUL; VECTOR_MUL_ASSOC] THEN X_GEN_TAC `x:real^N` THEN DISCH_TAC THEN MATCH_MP_TAC CONVEX_VSUM THEN ASM_SIMP_TAC[HULL_INC; CONVEX_CONVEX_HULL; SUM_LMUL] THEN ASM_SIMP_TAC[REAL_LT_IMP_NZ; REAL_MUL_LINV] THEN X_GEN_TAC `y:real^N` THEN DISCH_TAC THEN MATCH_MP_TAC REAL_LE_MUL THEN ASM_SIMP_TAC[REAL_LE_INV_EQ; REAL_LT_IMP_LE] THEN EXPAND_TAC "g" THEN REAL_ARITH_TAC; X_GEN_TAC `x:real^N` THEN DISCH_TAC THEN REWRITE_TAC[] THEN ONCE_REWRITE_TAC[NORM_SUB] THEN REWRITE_TAC[REWRITE_RULE[dist] (GSYM IN_BALL)] THEN REWRITE_TAC[GSYM VSUM_LMUL; VECTOR_MUL_ASSOC] THEN MATCH_MP_TAC CONVEX_VSUM_STRONG THEN ASM_REWRITE_TAC[CONVEX_BALL; SUM_LMUL; REAL_ENTIRE] THEN ASM_SIMP_TAC[REAL_LT_IMP_NZ; REAL_MUL_LINV; REAL_LT_INV_EQ; REAL_LE_MUL_EQ] THEN X_GEN_TAC `y:real^N` THEN DISCH_TAC THEN EXPAND_TAC "g" THEN REWRITE_TAC[IN_BALL; dist; NORM_SUB] THEN REAL_ARITH_TAC]);;
let SCHAUDER = 
prove (`!f s t:real^N->bool. convex s /\ ~(s = {}) /\ t SUBSET s /\ compact t /\ f continuous_on s /\ IMAGE f s SUBSET t ==> ?x. x IN s /\ f x = x`,
REPEAT STRIP_TAC THEN MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN MP_TAC(ISPECL [`\x:real^N. f x - x`; `convex hull t:real^N->bool`] BROUWER_COMPACTNESS_LEMMA) THEN SUBGOAL_THEN `(t:real^N->bool) SUBSET convex hull t` ASSUME_TAC THENL [REWRITE_TAC[HULL_SUBSET]; ALL_TAC] THEN SUBGOAL_THEN `(convex hull t:real^N->bool) SUBSET s` ASSUME_TAC THENL [ASM_SIMP_TAC[SUBSET_HULL] THEN ASM SET_TAC[]; ALL_TAC] THEN REWRITE_TAC[NOT_IMP] THEN CONJ_TAC THENL [ASM_SIMP_TAC[COMPACT_CONVEX_HULL; VECTOR_SUB_EQ] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN MATCH_MP_TAC CONTINUOUS_ON_SUB THEN REWRITE_TAC[CONTINUOUS_ON_ID; ETA_AX] THEN ASM_MESON_TAC[CONTINUOUS_ON_SUBSET]; DISCH_THEN(X_CHOOSE_THEN `e:real` STRIP_ASSUME_TAC)] THEN MP_TAC(ISPECL [`convex hull t:real^N->bool`; `e:real`] SCHAUDER_PROJECTION) THEN ASM_SIMP_TAC[NOT_EXISTS_THM; COMPACT_CONVEX_HULL] THEN MAP_EVERY X_GEN_TAC [`k:real^N->bool`; `g:real^N->real^N`] THEN STRIP_TAC THEN SUBGOAL_THEN `(convex hull k:real^N->bool) SUBSET s` ASSUME_TAC THENL [ASM_SIMP_TAC[SUBSET_HULL] THEN ASM SET_TAC[]; ALL_TAC] THEN MP_TAC(ISPECL [`(g:real^N->real^N) o (f:real^N->real^N)`; `convex hull k:real^N->bool`] BROUWER) THEN ASM_SIMP_TAC[COMPACT_CONVEX_HULL; FINITE_IMP_COMPACT] THEN REWRITE_TAC[CONVEX_CONVEX_HULL; NOT_IMP; GSYM CONJ_ASSOC; o_THM] THEN SUBGOAL_THEN `~(k:real^N->bool = {})` ASSUME_TAC THENL [ASM_MESON_TAC[IMAGE_EQ_EMPTY; SUBSET_EMPTY; CONVEX_HULL_EQ_EMPTY]; ASM_REWRITE_TAC[CONVEX_HULL_EQ_EMPTY]] THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN CONJ_TAC THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (REWRITE_RULE[IMP_CONJ] CONTINUOUS_ON_SUBSET)) THEN ASM SET_TAC[]; REWRITE_TAC[IMAGE_o] THEN ASM SET_TAC[]; DISCH_THEN(X_CHOOSE_THEN `x:real^N` STRIP_ASSUME_TAC) THEN FIRST_X_ASSUM(MP_TAC o SPEC `(f:real^N->real^N) x`) THEN ASM_REWRITE_TAC[NOT_IMP; NORM_ARITH `norm(x - y:real^N) < e <=> ~(e <= norm(y - x))`] THEN CONJ_TAC THENL [ASM SET_TAC[]; FIRST_X_ASSUM MATCH_MP_TAC] THEN UNDISCH_TAC `(x:real^N) IN convex hull k` THEN SPEC_TAC(`x:real^N`,`x:real^N`) THEN REWRITE_TAC[GSYM SUBSET] THEN ASM_SIMP_TAC[SUBSET_HULL; CONVEX_CONVEX_HULL]]);;
let SCHAUDER_UNIV = 
prove (`!f:real^N->real^N. f continuous_on (:real^N) /\ bounded (IMAGE f (:real^N)) ==> ?x. f x = x`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`f:real^N->real^N`; `(:real^N)`; `closure(IMAGE (f:real^N->real^N) (:real^N))`] SCHAUDER) THEN ASM_REWRITE_TAC[UNIV_NOT_EMPTY; CONVEX_UNIV; COMPACT_CLOSURE; IN_UNIV] THEN REWRITE_TAC[SUBSET_UNIV; CLOSURE_SUBSET]);;
let ROTHE = 
prove (`!f s:real^N->bool. closed s /\ convex s /\ ~(s = {}) /\ f continuous_on s /\ bounded(IMAGE f s) /\ IMAGE f (frontier s) SUBSET s ==> ?x. x IN s /\ f x = x`,
REPEAT STRIP_TAC THEN MP_TAC(ISPECL [`s:real^N->bool`; `(:real^N)`] ABSOLUTE_RETRACTION_CONVEX_CLOSED) THEN ASM_REWRITE_TAC[retraction; SUBSET_UNIV] THEN DISCH_THEN(X_CHOOSE_THEN `r:real^N->real^N` STRIP_ASSUME_TAC) THEN MP_TAC(ISPECL [`(r:real^N->real^N) o (f:real^N->real^N)`; `s:real^N->bool`; `IMAGE (r:real^N->real^N) (closure(IMAGE (f:real^N->real^N) s))`] SCHAUDER) THEN ANTS_TAC THENL [ASM_SIMP_TAC[CLOSURE_SUBSET; IMAGE_SUBSET; IMAGE_o] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN CONJ_TAC THENL [MATCH_MP_TAC COMPACT_CONTINUOUS_IMAGE THEN ASM_REWRITE_TAC[COMPACT_CLOSURE]; MATCH_MP_TAC CONTINUOUS_ON_COMPOSE] THEN ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]; MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `y:real^N` THEN REWRITE_TAC[o_THM] THEN STRIP_TAC THEN ASM SET_TAC[]]);;
(* ------------------------------------------------------------------------- *) (* Bijections between intervals. *) (* ------------------------------------------------------------------------- *)
let interval_bij = new_definition
 `interval_bij (a:real^N,b:real^N) (u:real^N,v:real^N) (x:real^N) =
    (lambda i. u$i + (x$i - a$i) / (b$i - a$i) * (v$i - u$i)):real^N`;;
let INTERVAL_BIJ_AFFINE = 
prove (`interval_bij (a,b) (u,v) = \x. (lambda i. (v$i - u$i) / (b$i - a$i) * x$i) + (lambda i. u$i - (v$i - u$i) / (b$i - a$i) * a$i)`,
SIMP_TAC[FUN_EQ_THM; CART_EQ; VECTOR_ADD_COMPONENT; LAMBDA_BETA; interval_bij] THEN REAL_ARITH_TAC);;
let CONTINUOUS_INTERVAL_BIJ = 
prove (`!a b u v x. (interval_bij (a:real^N,b:real^N) (u:real^N,v:real^N)) continuous at x`,
REPEAT GEN_TAC THEN REWRITE_TAC[INTERVAL_BIJ_AFFINE] THEN MATCH_MP_TAC CONTINUOUS_ADD THEN REWRITE_TAC[CONTINUOUS_CONST] THEN MATCH_MP_TAC LINEAR_CONTINUOUS_AT THEN SIMP_TAC[linear; CART_EQ; LAMBDA_BETA; VECTOR_ADD_COMPONENT; VECTOR_MUL_COMPONENT] THEN REAL_ARITH_TAC);;
let CONTINUOUS_ON_INTERVAL_BIJ = 
prove (`!a b u v s. interval_bij (a,b) (u,v) continuous_on s`,
REPEAT GEN_TAC THEN MATCH_MP_TAC CONTINUOUS_AT_IMP_CONTINUOUS_ON THEN REWRITE_TAC[CONTINUOUS_INTERVAL_BIJ]);;
let IN_INTERVAL_INTERVAL_BIJ = 
prove (`!a b u v x:real^N. x IN interval[a,b] /\ ~(interval[u,v] = {}) ==> (interval_bij (a,b) (u,v) x) IN interval[u,v]`,
SIMP_TAC[IN_INTERVAL; interval_bij; LAMBDA_BETA; INTERVAL_NE_EMPTY] THEN REWRITE_TAC[REAL_ARITH `u <= u + x <=> &0 <= x`; REAL_ARITH `u + x <= v <=> x <= &1 * (v - u)`] THEN REPEAT STRIP_TAC THENL [MATCH_MP_TAC REAL_LE_MUL THEN CONJ_TAC THEN TRY(MATCH_MP_TAC REAL_LE_DIV) THEN ASM_SIMP_TAC[REAL_SUB_LE] THEN ASM_MESON_TAC[REAL_LE_TRANS]; MATCH_MP_TAC REAL_LE_RMUL THEN ASM_SIMP_TAC[REAL_SUB_LE] THEN SUBGOAL_THEN `(a:real^N)$i <= (b:real^N)$i` MP_TAC THENL [ASM_MESON_TAC[REAL_LE_TRANS]; ALL_TAC] THEN GEN_REWRITE_TAC LAND_CONV [REAL_LE_LT] THEN STRIP_TAC THENL [ASM_SIMP_TAC[REAL_LE_LDIV_EQ; REAL_SUB_LT] THEN ASM_SIMP_TAC[REAL_ARITH `a <= x /\ x <= b ==> x - a <= &1 * (b - a)`]; ASM_REWRITE_TAC[real_div; REAL_SUB_REFL; REAL_INV_0] THEN REAL_ARITH_TAC]]);;
let INTERVAL_BIJ_BIJ = 
prove (`!a b u v x:real^N. (!i. 1 <= i /\ i <= dimindex(:N) ==> a$i < b$i /\ u$i < v$i) ==> interval_bij (a,b) (u,v) (interval_bij (u,v) (a,b) x) = x`,
SIMP_TAC[interval_bij; CART_EQ; LAMBDA_BETA; REAL_ADD_SUB] THEN REPEAT GEN_TAC THEN REWRITE_TAC[AND_FORALL_THM] THEN MATCH_MP_TAC MONO_FORALL THEN GEN_TAC THEN MATCH_MP_TAC MONO_IMP THEN REWRITE_TAC[] THEN CONV_TAC REAL_FIELD);;
(* ------------------------------------------------------------------------- *) (* Fashoda meet theorem. *) (* ------------------------------------------------------------------------- *)
let INFNORM_2 = 
prove (`infnorm (x:real^2) = max (abs(x$1)) (abs(x$2))`,
REWRITE_TAC[infnorm; INFNORM_SET_IMAGE; NUMSEG_CONV `1..2`; DIMINDEX_2] THEN REWRITE_TAC[IMAGE_CLAUSES; GSYM REAL_MAX_SUP]);;
let INFNORM_EQ_1_2 = 
prove (`infnorm (x:real^2) = &1 <=> abs(x$1) <= &1 /\ abs(x$2) <= &1 /\ (x$1 = -- &1 \/ x$1 = &1 \/ x$2 = -- &1 \/ x$2 = &1)`,
REWRITE_TAC[INFNORM_2] THEN REAL_ARITH_TAC);;
let INFNORM_EQ_1_IMP = 
prove (`infnorm (x:real^2) = &1 ==> abs(x$1) <= &1 /\ abs(x$2) <= &1`,
SIMP_TAC[INFNORM_EQ_1_2]);;
let FASHODA_UNIT = 
prove (`!f:real^1->real^2 g:real^1->real^2. IMAGE f (interval[--vec 1,vec 1]) SUBSET interval[--vec 1,vec 1] /\ IMAGE g (interval[--vec 1,vec 1]) SUBSET interval[--vec 1,vec 1] /\ f continuous_on interval[--vec 1,vec 1] /\ g continuous_on interval[--vec 1,vec 1] /\ f(--vec 1)$1 = -- &1 /\ f(vec 1)$1 = &1 /\ g(--vec 1)$2 = -- &1 /\ g(vec 1)$2 = &1 ==> ?s t. s IN interval[--vec 1,vec 1] /\ t IN interval[--vec 1,vec 1] /\ f(s) = g(t)`,
REPEAT STRIP_TAC THEN GEN_REWRITE_TAC I [TAUT `p <=> ~ ~p`] THEN DISCH_THEN(MP_TAC o REWRITE_RULE[NOT_EXISTS_THM]) THEN REWRITE_TAC[TAUT `~(a /\ b /\ c) <=> a /\ b ==> ~c`] THEN DISCH_TAC THEN ABBREV_TAC `sqprojection = \z:real^2. inv(infnorm z) % z` THEN ABBREV_TAC `(negatex:real^2->real^2) = \x. vector[--(x$1); x$2]` THEN SUBGOAL_THEN `!z:real^2. infnorm(negatex z:real^2) = infnorm z` ASSUME_TAC THENL [EXPAND_TAC "negatex" THEN SIMP_TAC[VECTOR_2; INFNORM_2] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `!z. ~(z = vec 0) ==> infnorm((sqprojection:real^2->real^2) z) = &1` ASSUME_TAC THENL [EXPAND_TAC "sqprojection" THEN REWRITE_TAC[INFNORM_MUL; REAL_ABS_INFNORM; REAL_ABS_INV] THEN SIMP_TAC[REAL_MUL_LINV; INFNORM_EQ_0]; ALL_TAC] THEN MP_TAC(ISPECL [`(\w. (negatex:real^2->real^2) (sqprojection(f(lift(w$1)) - g(lift(w$2)):real^2))) :real^2->real^2`; `interval[--vec 1,vec 1]:real^2->bool`] BROUWER_WEAK) THEN REWRITE_TAC[NOT_IMP; COMPACT_INTERVAL; CONVEX_INTERVAL] THEN REPEAT CONJ_TAC THENL [REWRITE_TAC[INTERIOR_CLOSED_INTERVAL; INTERVAL_NE_EMPTY] THEN SIMP_TAC[VEC_COMPONENT; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; MATCH_MP_TAC(REWRITE_RULE[o_DEF] CONTINUOUS_ON_COMPOSE) THEN CONJ_TAC THENL [ALL_TAC; MATCH_MP_TAC LINEAR_CONTINUOUS_ON THEN EXPAND_TAC "negatex" THEN SIMP_TAC[linear; VECTOR_2; CART_EQ; FORALL_2; DIMINDEX_2; VECTOR_MUL_COMPONENT; VECTOR_NEG_COMPONENT; VECTOR_ADD_COMPONENT; ARITH] THEN REAL_ARITH_TAC] THEN MATCH_MP_TAC(REWRITE_RULE[o_DEF] CONTINUOUS_ON_COMPOSE) THEN CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_SUB THEN CONJ_TAC THEN MATCH_MP_TAC(REWRITE_RULE[o_DEF] CONTINUOUS_ON_COMPOSE) THEN SIMP_TAC[CONTINUOUS_ON_LIFT_COMPONENT; DIMINDEX_2; ARITH] THEN MATCH_MP_TAC CONTINUOUS_ON_SUBSET THEN EXISTS_TAC `interval[--vec 1:real^1,vec 1]`; MATCH_MP_TAC CONTINUOUS_AT_IMP_CONTINUOUS_ON THEN EXPAND_TAC "sqprojection" THEN REWRITE_TAC[FORALL_IN_IMAGE] THEN X_GEN_TAC `x:real^2` THEN STRIP_TAC THEN MATCH_MP_TAC CONTINUOUS_MUL THEN REWRITE_TAC[CONTINUOUS_AT_ID] THEN GEN_REWRITE_TAC (LAND_CONV o RAND_CONV) [GSYM o_DEF] THEN MATCH_MP_TAC CONTINUOUS_AT_INV THEN REWRITE_TAC[CONTINUOUS_AT_LIFT_INFNORM; INFNORM_EQ_0; VECTOR_SUB_EQ] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [IN_INTERVAL])] THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; IN_INTERVAL_1] THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VEC_COMPONENT; ARITH; VECTOR_NEG_COMPONENT; DROP_NEG; DROP_VEC; LIFT_DROP]; REWRITE_TAC[SUBSET; FORALL_IN_IMAGE] THEN X_GEN_TAC `x:real^2` THEN STRIP_TAC THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; REAL_BOUNDS_LE; VECTOR_NEG_COMPONENT; VEC_COMPONENT; ARITH] THEN MATCH_MP_TAC INFNORM_EQ_1_IMP THEN ASM_REWRITE_TAC[] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN REWRITE_TAC[VECTOR_SUB_EQ] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [IN_INTERVAL]) THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; IN_INTERVAL_1] THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VEC_COMPONENT; ARITH; VECTOR_NEG_COMPONENT; DROP_NEG; DROP_VEC; LIFT_DROP]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `x:real^2` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `infnorm(x:real^2) = &1` MP_TAC THENL [FIRST_X_ASSUM(fun th -> GEN_REWRITE_TAC (LAND_CONV o RAND_CONV) [SYM th]) THEN ASM_REWRITE_TAC[] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN REWRITE_TAC[VECTOR_SUB_EQ] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN REWRITE_TAC[IN_INTERVAL_1] THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [IN_INTERVAL]) THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VEC_COMPONENT; ARITH; VECTOR_NEG_COMPONENT; DROP_NEG; DROP_VEC; LIFT_DROP]; ALL_TAC] THEN SUBGOAL_THEN `(!x i. 1 <= i /\ i <= 2 /\ ~(x = vec 0) ==> (&0 < ((sqprojection:real^2->real^2) x)$i <=> &0 < x$i)) /\ (!x i. 1 <= i /\ i <= 2 /\ ~(x = vec 0) ==> ((sqprojection x)$i < &0 <=> x$i < &0))` STRIP_ASSUME_TAC THENL [EXPAND_TAC "sqprojection" THEN SIMP_TAC[VECTOR_MUL_COMPONENT; DIMINDEX_2; ARITH] THEN REWRITE_TAC[GSYM(ONCE_REWRITE_RULE[REAL_MUL_SYM] real_div)] THEN SIMP_TAC[REAL_LT_LDIV_EQ; REAL_LT_RDIV_EQ; INFNORM_POS_LT] THEN REWRITE_TAC[REAL_MUL_LZERO]; ALL_TAC] THEN REWRITE_TAC[INFNORM_EQ_1_2; CONJ_ASSOC] THEN DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC (REPEAT_TCL DISJ_CASES_THEN (fun th -> ASSUME_TAC th THEN MP_TAC th))) THEN MAP_EVERY EXPAND_TAC ["x";
"negatex"] THEN REWRITE_TAC[VECTOR_2] THENL [DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `--x = -- &1 ==> &0 < x`)); DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `--x = &1 ==> x < &0`)); DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `x = -- &1 ==> x < &0`)); DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `x = &1 ==> &0 < x`))] THEN W(fun (_,w) -> FIRST_X_ASSUM(fun th -> MP_TAC(PART_MATCH (lhs o rand) th (lhand w)))) THEN (ANTS_TAC THENL [REWRITE_TAC[VECTOR_SUB_EQ; ARITH] THEN FIRST_X_ASSUM MATCH_MP_TAC THEN FIRST_X_ASSUM(MP_TAC o GEN_REWRITE_RULE I [IN_INTERVAL]) THEN ASM_SIMP_TAC[SUBSET; FORALL_IN_IMAGE; IN_INTERVAL_1] THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VEC_COMPONENT; ARITH; VECTOR_NEG_COMPONENT; DROP_NEG; DROP_VEC; LIFT_DROP] THEN REAL_ARITH_TAC; DISCH_THEN SUBST1_TAC]) THEN ASM_SIMP_TAC[VECTOR_SUB_COMPONENT; DIMINDEX_2; ARITH; LIFT_NEG; LIFT_NUM] THENL [MATCH_MP_TAC(REAL_ARITH `abs(x$1) <= &1 /\ abs(x$2) <= &1 ==> ~(&0 < -- &1 - x$1)`); MATCH_MP_TAC(REAL_ARITH `abs(x$1) <= &1 /\ abs(x$2) <= &1 ==> ~(&1 - x$1 < &0)`); MATCH_MP_TAC(REAL_ARITH `abs(x$1) <= &1 /\ abs(x$2) <= &1 ==> ~(x$2 - -- &1 < &0)`); MATCH_MP_TAC(REAL_ARITH `abs(x$1) <= &1 /\ abs(x$2) <= &1 ==> ~(&0 < x$2 - &1)`)] THEN (SUBGOAL_THEN `!z:real^2. abs(z$1) <= &1 /\ abs(z$2) <= &1 <=> z IN interval[--vec 1,vec 1]` (fun th -> REWRITE_TAC[th]) THENL [SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VEC_COMPONENT; ARITH; VECTOR_NEG_COMPONENT; DROP_NEG; DROP_VEC; LIFT_DROP] THEN REAL_ARITH_TAC; ALL_TAC]) THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `IMAGE f s SUBSET t ==> x IN s ==> f x IN t`)) THEN REWRITE_TAC[IN_INTERVAL_1; DROP_NEG; DROP_VEC; LIFT_DROP] THEN ASM_REWRITE_TAC[REAL_BOUNDS_LE]);;
let FASHODA_UNIT_PATH = 
prove (`!f:real^1->real^2 g:real^1->real^2. path f /\ path g /\ path_image f SUBSET interval[--vec 1,vec 1] /\ path_image g SUBSET interval[--vec 1,vec 1] /\ (pathstart f)$1 = -- &1 /\ (pathfinish f)$1 = &1 /\ (pathstart g)$2 = -- &1 /\ (pathfinish g)$2 = &1 ==> ?z. z IN path_image f /\ z IN path_image g`,
SIMP_TAC[path; path_image; pathstart; pathfinish] THEN REPEAT STRIP_TAC THEN ABBREV_TAC `iscale = \z:real^1. inv(&2) % (z + vec 1)` THEN MP_TAC(ISPECL [`(f:real^1->real^2) o (iscale:real^1->real^1)`; `(g:real^1->real^2) o (iscale:real^1->real^1)`] FASHODA_UNIT) THEN SUBGOAL_THEN `IMAGE (iscale:real^1->real^1) (interval[--vec 1,vec 1]) SUBSET interval[vec 0,vec 1]` ASSUME_TAC THENL [REWRITE_TAC[SUBSET; FORALL_IN_IMAGE] THEN EXPAND_TAC "iscale" THEN REWRITE_TAC[IN_INTERVAL_1; DROP_NEG; DROP_VEC; DROP_CMUL; DROP_ADD] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `(iscale:real^1->real^1) continuous_on interval [--vec 1,vec 1]` ASSUME_TAC THENL [EXPAND_TAC "iscale" THEN SIMP_TAC[CONTINUOUS_ON_CMUL; CONTINUOUS_ON_ID; CONTINUOUS_ON_ADD; CONTINUOUS_ON_CONST]; ALL_TAC] THEN ASM_REWRITE_TAC[IMAGE_o] THEN ANTS_TAC THENL [REPLICATE_TAC 2 (CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC]) THEN REPLICATE_TAC 2 (CONJ_TAC THENL [MATCH_MP_TAC CONTINUOUS_ON_COMPOSE THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[CONTINUOUS_ON_SUBSET]; ALL_TAC]) THEN EXPAND_TAC "iscale" THEN REWRITE_TAC[o_THM] THEN ASM_REWRITE_TAC[VECTOR_ARITH `inv(&2) % (--x + x) = vec 0`; VECTOR_ARITH `inv(&2) % (x + x) = x`]; REWRITE_TAC[o_THM; LEFT_IMP_EXISTS_THM; IN_IMAGE] THEN ASM SET_TAC[]]);;
let FASHODA = 
prove (`!f g a b:real^2. path f /\ path g /\ path_image f SUBSET interval[a,b] /\ path_image g SUBSET interval[a,b] /\ (pathstart f)$1 = a$1 /\ (pathfinish f)$1 = b$1 /\ (pathstart g)$2 = a$2 /\ (pathfinish g)$2 = b$2 ==> ?z. z IN path_image f /\ z IN path_image g`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `~(interval[a:real^2,b] = {})` MP_TAC THENL [FIRST_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `s SUBSET t ==> ~(s = {}) ==> ~(t = {})`)) THEN REWRITE_TAC[PATH_IMAGE_NONEMPTY]; ALL_TAC] THEN REWRITE_TAC[INTERVAL_NE_EMPTY; DIMINDEX_2; FORALL_2] THEN STRIP_TAC THEN MP_TAC(ASSUME `(a:real^2)$1 <= (b:real^2)$1`) THEN REWRITE_TAC[REAL_ARITH `a <= b <=> b = a \/ a < b`] THEN STRIP_TAC THENL [SUBGOAL_THEN `?z:real^2. z IN path_image g /\ z$2 = (pathstart f:real^2)$2` MP_TAC THENL [MATCH_MP_TAC CONNECTED_IVT_COMPONENT THEN MAP_EVERY EXISTS_TAC [`pathstart(g:real^1->real^2)`; `pathfinish(g:real^1->real^2)`] THEN ASM_SIMP_TAC[CONNECTED_PATH_IMAGE; PATHSTART_IN_PATH_IMAGE; REAL_LE_REFL; PATHFINISH_IN_PATH_IMAGE; DIMINDEX_2; ARITH] THEN UNDISCH_TAC `path_image f SUBSET interval[a:real^2,b]` THEN REWRITE_TAC[SUBSET; path_image; IN_INTERVAL_1; FORALL_IN_IMAGE] THEN DISCH_THEN(MP_TAC o SPEC `vec 0:real^1`) THEN SIMP_TAC[pathstart] THEN SIMP_TAC[DROP_VEC; REAL_POS; IN_INTERVAL; FORALL_2; DIMINDEX_2]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `z:real^2` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SIMP_TAC[path_image; IN_IMAGE] THEN EXISTS_TAC `vec 0:real^1` THEN REWRITE_TAC[IN_INTERVAL_1; DROP_VEC; REAL_POS] THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2; pathstart] THEN SUBGOAL_THEN `(z:real^2) IN interval[a,b] /\ f(vec 0:real^1) IN interval[a,b]` MP_TAC THENL [ASM_MESON_TAC[SUBSET; path_image; IN_IMAGE; PATHSTART_IN_PATH_IMAGE; pathstart]; ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN REAL_ARITH_TAC]; ALL_TAC] THEN MP_TAC(ASSUME `(a:real^2)$2 <= (b:real^2)$2`) THEN REWRITE_TAC[REAL_ARITH `a <= b <=> b = a \/ a < b`] THEN STRIP_TAC THENL [SUBGOAL_THEN `?z:real^2. z IN path_image f /\ z$1 = (pathstart g:real^2)$1` MP_TAC THENL [MATCH_MP_TAC CONNECTED_IVT_COMPONENT THEN MAP_EVERY EXISTS_TAC [`pathstart(f:real^1->real^2)`; `pathfinish(f:real^1->real^2)`] THEN ASM_SIMP_TAC[CONNECTED_PATH_IMAGE; PATHSTART_IN_PATH_IMAGE; REAL_LE_REFL; PATHFINISH_IN_PATH_IMAGE; DIMINDEX_2; ARITH] THEN UNDISCH_TAC `path_image g SUBSET interval[a:real^2,b]` THEN REWRITE_TAC[SUBSET; path_image; IN_INTERVAL_1; FORALL_IN_IMAGE] THEN DISCH_THEN(MP_TAC o SPEC `vec 0:real^1`) THEN SIMP_TAC[pathstart] THEN SIMP_TAC[DROP_VEC; REAL_POS; IN_INTERVAL; FORALL_2; DIMINDEX_2]; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `z:real^2` THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN SIMP_TAC[path_image; IN_IMAGE] THEN EXISTS_TAC `vec 0:real^1` THEN REWRITE_TAC[IN_INTERVAL_1; DROP_VEC; REAL_POS] THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2; pathstart] THEN SUBGOAL_THEN `(z:real^2) IN interval[a,b] /\ g(vec 0:real^1) IN interval[a,b]` MP_TAC THENL [ASM_MESON_TAC[SUBSET; path_image; IN_IMAGE; PATHSTART_IN_PATH_IMAGE; pathstart]; ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN REAL_ARITH_TAC]; ALL_TAC] THEN MP_TAC(ISPECL [`interval_bij (a,b) (--vec 1,vec 1) o (f:real^1->real^2)`; `interval_bij (a,b) (--vec 1,vec 1) o (g:real^1->real^2)`] FASHODA_UNIT_PATH) THEN RULE_ASSUM_TAC(REWRITE_RULE[path; path_image; pathstart; pathfinish]) THEN ASM_REWRITE_TAC[path; path_image; pathstart; pathfinish; o_THM] THEN ANTS_TAC THENL [ASM_SIMP_TAC[CONTINUOUS_ON_COMPOSE; CONTINUOUS_ON_INTERVAL_BIJ] THEN REWRITE_TAC[IMAGE_o] THEN REPLICATE_TAC 2 (CONJ_TAC THENL [REWRITE_TAC[SUBSET] THEN ONCE_REWRITE_TAC[FORALL_IN_IMAGE] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC IN_INTERVAL_INTERVAL_BIJ THEN SIMP_TAC[INTERVAL_NE_EMPTY; VECTOR_NEG_COMPONENT; VEC_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN ASM SET_TAC[]; ALL_TAC]) THEN ASM_SIMP_TAC[interval_bij; LAMBDA_BETA; DIMINDEX_2; ARITH] THEN ASM_SIMP_TAC[REAL_DIV_REFL; REAL_LT_IMP_NZ; REAL_SUB_LT] THEN REWRITE_TAC[real_div; REAL_SUB_REFL; REAL_MUL_LZERO] THEN SIMP_TAC[VECTOR_NEG_COMPONENT; VEC_COMPONENT; DIMINDEX_2; ARITH] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `z:real^2` (fun th -> EXISTS_TAC `interval_bij (--vec 1,vec 1) (a,b) (z:real^2)` THEN MP_TAC th)) THEN MATCH_MP_TAC MONO_AND THEN CONJ_TAC THEN REWRITE_TAC[IMAGE_o] THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> g(f(x)) = x) ==> x IN IMAGE f s ==> g x IN s`) THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC INTERVAL_BIJ_BIJ THEN ASM_SIMP_TAC[FORALL_2; DIMINDEX_2; VECTOR_NEG_COMPONENT; VEC_COMPONENT; ARITH] THEN CONV_TAC REAL_RAT_REDUCE_CONV);;
(* ------------------------------------------------------------------------- *) (* Some slightly ad hoc lemmas I use below *) (* ------------------------------------------------------------------------- *)
let SEGMENT_VERTICAL = 
prove (`!a:real^2 b:real^2 x:real^2. a$1 = b$1 ==> (x IN segment[a,b] <=> x$1 = a$1 /\ x$1 = b$1 /\ (a$2 <= x$2 /\ x$2 <= b$2 \/ b$2 <= x$2 /\ x$2 <= a$2))`,
GEOM_ORIGIN_TAC `a:real^2` THEN REWRITE_TAC[VECTOR_ADD_COMPONENT; VEC_COMPONENT; REAL_LE_LADD; REAL_EQ_ADD_LCANCEL] THEN REPEAT GEN_TAC THEN DISCH_THEN(ASSUME_TAC o SYM) THEN SUBST1_TAC(SYM(ISPEC `b:real^2` BASIS_EXPANSION)) THEN ASM_REWRITE_TAC[DIMINDEX_2; VSUM_2; VECTOR_MUL_LZERO; VECTOR_ADD_LID] THEN SUBST1_TAC(VECTOR_ARITH `vec 0:real^2 = &0 % basis 2`) THEN REWRITE_TAC[SEGMENT_SCALAR_MULTIPLE; IN_ELIM_THM; CART_EQ] THEN REWRITE_TAC[DIMINDEX_2; FORALL_2; VECTOR_MUL_COMPONENT] THEN SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; REAL_MUL_RZERO; REAL_MUL_RID] THEN MESON_TAC[]);;
let SEGMENT_HORIZONTAL = 
prove (`!a:real^2 b:real^2 x:real^2. a$2 = b$2 ==> (x IN segment[a,b] <=> x$2 = a$2 /\ x$2 = b$2 /\ (a$1 <= x$1 /\ x$1 <= b$1 \/ b$1 <= x$1 /\ x$1 <= a$1))`,
GEOM_ORIGIN_TAC `a:real^2` THEN REWRITE_TAC[VECTOR_ADD_COMPONENT; VEC_COMPONENT; REAL_LE_LADD; REAL_EQ_ADD_LCANCEL] THEN REPEAT GEN_TAC THEN DISCH_THEN(ASSUME_TAC o SYM) THEN SUBST1_TAC(SYM(ISPEC `b:real^2` BASIS_EXPANSION)) THEN ASM_REWRITE_TAC[DIMINDEX_2; VSUM_2; VECTOR_MUL_LZERO; VECTOR_ADD_RID] THEN SUBST1_TAC(VECTOR_ARITH `vec 0:real^2 = &0 % basis 1`) THEN REWRITE_TAC[SEGMENT_SCALAR_MULTIPLE; IN_ELIM_THM; CART_EQ] THEN REWRITE_TAC[DIMINDEX_2; FORALL_2; VECTOR_MUL_COMPONENT] THEN SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; REAL_MUL_RZERO; REAL_MUL_RID] THEN MESON_TAC[]);;
(* ------------------------------------------------------------------------- *) (* Useful Fashoda corollary pointed out to me by Tom Hales. *) (* ------------------------------------------------------------------------- *)
let FASHODA_INTERLACE = 
prove (`!f g a b:real^2. path f /\ path g /\ path_image f SUBSET interval[a,b] /\ path_image g SUBSET interval[a,b] /\ (pathstart f)$2 = a$2 /\ (pathfinish f)$2 = a$2 /\ (pathstart g)$2 = a$2 /\ (pathfinish g)$2 = a$2 /\ (pathstart f)$1 < (pathstart g)$1 /\ (pathstart g)$1 < (pathfinish f)$1 /\ (pathfinish f)$1 < (pathfinish g)$1 ==> ?z. z IN path_image f /\ z IN path_image g`,
REPEAT STRIP_TAC THEN SUBGOAL_THEN `~(interval[a:real^2,b] = {})` MP_TAC THENL [FIRST_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `s SUBSET t ==> ~(s = {}) ==> ~(t = {})`)) THEN REWRITE_TAC[PATH_IMAGE_NONEMPTY]; ALL_TAC] THEN SUBGOAL_THEN `pathstart (f:real^1->real^2) IN interval[a,b] /\ pathfinish f IN interval[a,b] /\ pathstart g IN interval[a,b] /\ pathfinish g IN interval[a,b]` MP_TAC THENL [ASM_MESON_TAC[SUBSET; PATHSTART_IN_PATH_IMAGE; PATHFINISH_IN_PATH_IMAGE]; ALL_TAC] THEN REWRITE_TAC[INTERVAL_NE_EMPTY; IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN REPEAT STRIP_TAC THEN MP_TAC(SPECL [`linepath(vector[a$1 - &2;a$2 - &2],vector[(pathstart f)$1;a$2 - &2]) ++ linepath(vector[(pathstart f)$1;(a:real^2)$2 - &2],pathstart f) ++ (f:real^1->real^2) ++ linepath(pathfinish f,vector[(pathfinish f)$1;a$2 - &2]) ++ linepath(vector[(pathfinish f)$1;a$2 - &2], vector[(b:real^2)$1 + &2;a$2 - &2])`; `linepath(vector[(pathstart g)$1; (pathstart g)$2 - &3],pathstart g) ++ (g:real^1->real^2) ++ linepath(pathfinish g,vector[(pathfinish g)$1;(a:real^2)$2 - &1]) ++ linepath(vector[(pathfinish g)$1;a$2 - &1],vector[b$1 + &1;a$2 - &1]) ++ linepath(vector[b$1 + &1;a$2 - &1],vector[(b:real^2)$1 + &1;b$2 + &3])`; `vector[(a:real^2)$1 - &2; a$2 - &3]:real^2`; `vector[(b:real^2)$1 + &2; b$2 + &3]:real^2`] FASHODA) THEN ASM_SIMP_TAC[PATH_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_IMAGE_JOIN; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_LINEPATH] THEN REWRITE_TAC[VECTOR_2] THEN ANTS_TAC THENL [CONJ_TAC THEN REPEAT(MATCH_MP_TAC (SET_RULE `s SUBSET u /\ t SUBSET u ==> (s UNION t) SUBSET u`) THEN CONJ_TAC) THEN TRY(REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN MATCH_MP_TAC(REWRITE_RULE[CONVEX_CONTAINS_SEGMENT] (CONJUNCT1 (SPEC_ALL CONVEX_INTERVAL))) THEN ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2] THEN ASM_REAL_ARITH_TAC) THEN MATCH_MP_TAC SUBSET_TRANS THEN EXISTS_TAC `interval[a:real^2,b:real^2]` THEN ASM_REWRITE_TAC[SUBSET_REFL] THEN REWRITE_TAC[SUBSET_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2] THEN ASM_REAL_ARITH_TAC; ALL_TAC] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `z:real^2` THEN REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN SUBGOAL_THEN `!f s:real^2->bool. path_image f UNION s = path_image f UNION (s DIFF {pathstart f,pathfinish f})` (fun th -> ONCE_REWRITE_TAC[th] THEN REWRITE_TAC[GSYM UNION_ASSOC] THEN ONCE_REWRITE_TAC[SET_RULE `(s UNION t) UNION u = u UNION t UNION s`] THEN ONCE_REWRITE_TAC[th]) THENL [REWRITE_TAC[EXTENSION; IN_UNION; IN_DIFF; IN_INSERT; NOT_IN_EMPTY] THEN ASM_MESON_TAC[PATHSTART_IN_PATH_IMAGE; PATHFINISH_IN_PATH_IMAGE]; ALL_TAC] THEN REWRITE_TAC[IN_UNION; IN_DIFF; GSYM DISJ_ASSOC; LEFT_OR_DISTRIB; RIGHT_OR_DISTRIB; GSYM CONJ_ASSOC; SET_RULE `~(z IN {x,y}) <=> ~(z = x) /\ ~(z = y)`] THEN DISCH_THEN(REPEAT_TCL DISJ_CASES_THEN MP_TAC) THEN ASM_SIMP_TAC[SEGMENT_VERTICAL; SEGMENT_HORIZONTAL; VECTOR_2] THEN REPEAT STRIP_TAC THEN ASM_REWRITE_TAC[] THEN UNDISCH_TAC `path_image (f:real^1->real^2) SUBSET interval [a,b]` THEN REWRITE_TAC[SUBSET] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN UNDISCH_TAC `path_image (g:real^1->real^2) SUBSET interval [a,b]` THEN REWRITE_TAC[SUBSET] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN REPEAT(DISCH_THEN(fun th -> if is_imp(concl th) then ALL_TAC else ASSUME_TAC th)) THEN REPEAT(POP_ASSUM MP_TAC) THEN TRY REAL_ARITH_TAC THEN REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2] THEN REAL_ARITH_TAC);;
(* ------------------------------------------------------------------------- *) (* Complement in dimension N >= 2 of set homeomorphic to any interval in *) (* any dimension is (path-)connected. This naively generalizes the argument *) (* in Ryuji Maehara's paper "The Jordan curve theorem via the Brouwer *) (* fixed point theorem", American Mathematical Monthly 1984. *) (* ------------------------------------------------------------------------- *)
let UNBOUNDED_COMPONENTS_COMPLEMENT_ABSOLUTE_RETRACT = 
prove (`!s c. compact s /\ s retract_of (:real^N) /\ c IN components((:real^N) DIFF s) ==> ~bounded c`,
REWRITE_TAC[IMP_CONJ; RIGHT_FORALL_IMP_THM; components; FORALL_IN_GSPEC] THEN GEN_TAC THEN DISCH_TAC THEN DISCH_TAC THEN X_GEN_TAC `y:real^N` THEN REWRITE_TAC[IN_DIFF; IN_UNIV] THEN REPEAT STRIP_TAC THEN SUBGOAL_THEN `open((:real^N) DIFF s)` ASSUME_TAC THENL [ASM_SIMP_TAC[GSYM closed; COMPACT_IMP_CLOSED]; ALL_TAC] THEN FIRST_ASSUM(MP_TAC o GEN_REWRITE_RULE I [retract_of]) THEN REWRITE_TAC[retraction; LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `r:real^N->real^N` THEN STRIP_TAC THEN MP_TAC(ISPECL [`connected_component ((:real^N) DIFF s) y`; `s:real^N->bool`; `r:real^N->real^N`] FRONTIER_SUBSET_RETRACTION) THEN ASM_SIMP_TAC[NOT_IMP; INTERIOR_OPEN; OPEN_CONNECTED_COMPONENT] THEN REPEAT CONJ_TAC THENL [REWRITE_TAC[frontier] THEN ASM_SIMP_TAC[INTERIOR_OPEN; OPEN_CONNECTED_COMPONENT] THEN REWRITE_TAC[SUBSET; IN_DIFF] THEN X_GEN_TAC `z:real^N` THEN ASM_CASES_TAC `(z:real^N) IN s` THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[IN_CLOSURE_CONNECTED_COMPONENT; IN_UNIV; IN_DIFF] THEN CONV_TAC TAUT; ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]; ASM SET_TAC[]; MATCH_MP_TAC(SET_RULE `~(c = {}) /\ c SUBSET (:real^N) DIFF s ==> ~(c SUBSET s)`) THEN REWRITE_TAC[CONNECTED_COMPONENT_SUBSET; CONNECTED_COMPONENT_EQ_EMPTY] THEN ASM_REWRITE_TAC[IN_UNIV; IN_DIFF]]);;
let CONNECTED_COMPLEMENT_ABSOLUTE_RETRACT = 
prove (`!s. 2 <= dimindex(:N) /\ compact s /\ s retract_of (:real^N) ==> connected((:real^N) DIFF s)`,
REPEAT STRIP_TAC THEN REWRITE_TAC[CONNECTED_EQ_CONNECTED_COMPONENT_EQ] THEN REPEAT STRIP_TAC THEN MATCH_MP_TAC COBOUNDED_UNIQUE_UNBOUNDED_COMPONENT THEN ASM_SIMP_TAC[SET_RULE`UNIV DIFF (UNIV DIFF s) = s`; COMPACT_IMP_BOUNDED] THEN CONJ_TAC THEN MATCH_MP_TAC UNBOUNDED_COMPONENTS_COMPLEMENT_ABSOLUTE_RETRACT THEN EXISTS_TAC `s:real^N->bool` THEN ASM_REWRITE_TAC[IN_COMPONENTS] THEN ASM_MESON_TAC[]);;
let PATH_CONNECTED_COMPLEMENT_ABSOLUTE_RETRACT = 
prove (`!s:real^N->bool. 2 <= dimindex(:N) /\ compact s /\ s retract_of (:real^N) ==> path_connected((:real^N) DIFF s)`,
REPEAT GEN_TAC THEN DISCH_TAC THEN FIRST_ASSUM (MP_TAC o MATCH_MP CONNECTED_COMPLEMENT_ABSOLUTE_RETRACT) THEN MATCH_MP_TAC EQ_IMP THEN CONV_TAC SYM_CONV THEN MATCH_MP_TAC PATH_CONNECTED_EQ_CONNECTED THEN REWRITE_TAC[GSYM closed] THEN ASM_MESON_TAC[HOMEOMORPHIC_COMPACTNESS; COMPACT_INTERVAL; COMPACT_IMP_CLOSED]);;
let CONNECTED_COMPLEMENT_HOMEOMORPHIC_CONVEX_COMPACT = 
prove (`!s:real^N->bool t:real^M->bool. 2 <= dimindex(:N) /\ s homeomorphic t /\ convex t /\ compact t ==> connected((:real^N) DIFF s)`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `s:real^N->bool = {}` THEN ASM_REWRITE_TAC[DIFF_EMPTY; CONNECTED_UNIV] THEN MATCH_MP_TAC CONNECTED_COMPLEMENT_ABSOLUTE_RETRACT THEN ASM_REWRITE_TAC[] THEN CONJ_TAC THENL [ASM_MESON_TAC[HOMEOMORPHIC_COMPACTNESS]; ALL_TAC] THEN MATCH_MP_TAC ABSOLUTE_RETRACT_HOMEOMORPHIC_CONVEX_COMPACT THEN EXISTS_TAC `t:real^M->bool` THEN ASM_REWRITE_TAC[SUBSET_UNIV]);;
let PATH_CONNECTED_COMPLEMENT_HOMEOMORPHIC_CONVEX_COMPACT = 
prove (`!s:real^N->bool t:real^M->bool. 2 <= dimindex(:N) /\ s homeomorphic t /\ convex t /\ compact t ==> path_connected((:real^N) DIFF s)`,
REPEAT GEN_TAC THEN DISCH_TAC THEN FIRST_ASSUM (MP_TAC o MATCH_MP CONNECTED_COMPLEMENT_HOMEOMORPHIC_CONVEX_COMPACT) THEN MATCH_MP_TAC EQ_IMP THEN CONV_TAC SYM_CONV THEN MATCH_MP_TAC PATH_CONNECTED_EQ_CONNECTED THEN REWRITE_TAC[GSYM closed] THEN ASM_MESON_TAC[HOMEOMORPHIC_COMPACTNESS; COMPACT_INTERVAL; COMPACT_IMP_CLOSED]);;
(* ------------------------------------------------------------------------- *) (* In particular, apply all these to the special case of an arc. *) (* ------------------------------------------------------------------------- *)
let RETRACTION_ARC = 
prove (`!p. arc p ==> ?f. f continuous_on (:real^N) /\ IMAGE f (:real^N) SUBSET path_image p /\ (!x. x IN path_image p ==> f x = x)`,
REPEAT STRIP_TAC THEN FIRST_X_ASSUM(MP_TAC o SPEC `(:real^N)` o MATCH_MP (REWRITE_RULE[IMP_CONJ] ABSOLUTE_RETRACT_PATH_IMAGE_ARC)) THEN REWRITE_TAC[SUBSET_UNIV; retract_of; retraction]);;
let PATH_CONNECTED_ARC_COMPLEMENT = 
prove (`!p. 2 <= dimindex(:N) /\ arc p ==> path_connected((:real^N) DIFF path_image p)`,
REWRITE_TAC[arc; path] THEN REPEAT STRIP_TAC THEN SIMP_TAC[path_image] THEN MP_TAC(ISPECL [`path_image p:real^N->bool`; `interval[vec 0:real^1,vec 1]`] PATH_CONNECTED_COMPLEMENT_HOMEOMORPHIC_CONVEX_COMPACT) THEN ASM_REWRITE_TAC[CONVEX_INTERVAL; COMPACT_INTERVAL; path_image] THEN DISCH_THEN MATCH_MP_TAC THEN ONCE_REWRITE_TAC[HOMEOMORPHIC_SYM] THEN MATCH_MP_TAC HOMEOMORPHIC_COMPACT THEN EXISTS_TAC `p:real^1->real^N` THEN ASM_REWRITE_TAC[COMPACT_INTERVAL]);;
let CONNECTED_ARC_COMPLEMENT = 
prove (`!p. 2 <= dimindex(:N) /\ arc p ==> connected((:real^N) DIFF path_image p)`,
let INSIDE_ARC_EMPTY = 
prove (`!p:real^1->real^N. arc p ==> inside(path_image p) = {}`,
REPEAT STRIP_TAC THEN ASM_CASES_TAC `dimindex(:N) = 1` THENL [MATCH_MP_TAC INSIDE_CONVEX THEN ASM_SIMP_TAC[CONVEX_CONNECTED_1_GEN; CONNECTED_PATH_IMAGE; ARC_IMP_PATH]; MATCH_MP_TAC INSIDE_BOUNDED_COMPLEMENT_CONNECTED_EMPTY THEN ASM_SIMP_TAC[BOUNDED_PATH_IMAGE; ARC_IMP_PATH] THEN MATCH_MP_TAC CONNECTED_ARC_COMPLEMENT THEN ASM_REWRITE_TAC[ARITH_RULE `2 <= n <=> 1 <= n /\ ~(n = 1)`] THEN REWRITE_TAC[DIMINDEX_GE_1]]);;
let INSIDE_SIMPLE_CURVE_IMP_CLOSED = 
prove (`!g x:real^N. simple_path g /\ x IN inside(path_image g) ==> pathfinish g = pathstart g`,
(* ------------------------------------------------------------------------- *) (* The Jordan curve theorem, again approximately following Maehara. *) (* ------------------------------------------------------------------------- *)
let JORDAN_CURVE_THEOREM = 
prove (`!c:real^1->real^2. simple_path c /\ pathfinish c = pathstart c ==> ?ins out. ~(ins = {}) /\ open ins /\ connected ins /\ ~(out = {}) /\ open out /\ connected out /\ bounded ins /\ ~bounded out /\ ins INTER out = {} /\ ins UNION out = (:real^2) DIFF path_image c /\ frontier ins = path_image c /\ frontier out = path_image c`,
REPEAT STRIP_TAC THEN MP_TAC(ISPEC `path_image(c:real^1->real^2)` DIAMETER_BOUNDED_BOUND) THEN MP_TAC(ISPEC `path_image(c:real^1->real^2)` DIAMETER_COMPACT_ATTAINED) THEN ASM_SIMP_TAC[COMPACT_PATH_IMAGE; PATH_IMAGE_NONEMPTY; SIMPLE_PATH_IMP_PATH; COMPACT_IMP_BOUNDED; LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`a:real^2`; `b:real^2`] THEN ABBREV_TAC `m:real^2 = midpoint(a,b)` THEN POP_ASSUM_LIST(MP_TAC o end_itlist CONJ) THEN REWRITE_TAC[IMP_IMP] THEN GEOM_ORIGIN_TAC `m:real^2` THEN REPEAT GEN_TAC THEN REWRITE_TAC[midpoint; VECTOR_ARITH `inv(&2) % (a + b):real^N = vec 0 <=> a = --b`] THEN ASM_CASES_TAC `a:real^2 = --b` THEN ASM_REWRITE_TAC[] THEN POP_ASSUM(K ALL_TAC) THEN REWRITE_TAC[NORM_ARITH `norm(--b - b) = &2 * norm(b)`] THEN ASM_CASES_TAC `diameter(path_image(c:real^1->real^2)) = &2 * norm(b:real^2)` THEN ASM_REWRITE_TAC[] THEN POP_ASSUM(K ALL_TAC) THEN GEOM_NORMALIZE_TAC `b:real^2` THEN CONJ_TAC THENL [REWRITE_TAC[NORM_0; REAL_MUL_RZERO] THEN REWRITE_TAC[NORM_ARITH `norm(x - y) <= &0 <=> x = y`] THEN REWRITE_TAC[simple_path; path_image; IMP_CONJ; RIGHT_FORALL_IMP_THM] THEN REWRITE_TAC[FORALL_IN_IMAGE] THEN SUBGOAL_THEN `(vec 0:real^1) IN interval[vec 0,vec 1] /\ lift(&1 / &2) IN interval[vec 0,vec 1] /\ ~(lift(&1 / &2) = vec 0) /\ ~(lift(&1 / &2) = vec 1)` (fun th -> MESON_TAC[th]) THEN REWRITE_TAC[IN_INTERVAL_1; DROP_VEC; LIFT_DROP; GSYM DROP_EQ] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN GEOM_BASIS_MULTIPLE_TAC 1 `b:real^2` THEN GEN_TAC THEN DISCH_TAC THEN SIMP_TAC[NORM_MUL; NORM_BASIS; DIMINDEX_2; ARITH] THEN ASM_SIMP_TAC[REAL_ARITH `&0 <= b ==> (abs b * &1 = &1 <=> b = &1)`] THEN DISCH_TAC THEN POP_ASSUM_LIST(K ALL_TAC) THEN REWRITE_TAC[VECTOR_MUL_LID; REAL_MUL_RID; GSYM CONJ_ASSOC] THEN X_GEN_TAC `c:real^1->real^2` THEN STRIP_TAC THEN SUBGOAL_THEN `(!z:real^2. z IN path_image c /\ z$1 = -- &1 <=> z = --basis 1) /\ (!z:real^2. z IN path_image c /\ z$1 = &1 <=> z = basis 1)` (CONJUNCTS_THEN2 (LABEL_TAC "touchleft") (LABEL_TAC "touchright")) THENL [CONJ_TAC THEN X_GEN_TAC `z:real^2` THEN EQ_TAC THEN ASM_SIMP_TAC[BASIS_COMPONENT; VECTOR_NEG_COMPONENT; DIMINDEX_2; ARITH] THEN STRIP_TAC THENL [FIRST_X_ASSUM(MP_TAC o ISPECL [`basis 1:real^2`; `z:real^2`]); FIRST_X_ASSUM(MP_TAC o ISPECL [`--basis 1:real^2`; `z:real^2`])] THEN ASM_REWRITE_TAC[NORM_LE_SQUARE; DOT_2; VECTOR_SUB_COMPONENT] THEN ASM_SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT; CART_EQ; FORALL_2] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN REWRITE_TAC[REAL_ARITH `&4 + (&0 - x) * (&0 - x) <= &4 <=> x pow 2 <= &0`] THEN ONCE_REWRITE_TAC[REAL_RING `z = &0 <=> z * z = &0`] THEN SIMP_TAC[REAL_POW_2; REAL_LE_SQUARE; GSYM REAL_LE_ANTISYM]; ALL_TAC] THEN SUBGOAL_THEN `!z:real^2. z IN path_image c ==> abs(z$1) <= &1` (LABEL_TAC "xbound") THENL [X_GEN_TAC `z:real^2` THEN STRIP_TAC THEN REWRITE_TAC[REAL_ABS_BOUNDS] THEN CONJ_TAC THENL [FIRST_X_ASSUM(MP_TAC o ISPECL [`basis 1:real^2`; `z:real^2`]); FIRST_X_ASSUM(MP_TAC o ISPECL [`z:real^2`; `--basis 1:real^2`])] THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[GSYM CONTRAPOS_THM] THEN REWRITE_TAC[REAL_NOT_LE] THEN DISCH_TAC THEN MATCH_MP_TAC(REAL_ARITH `abs(z$1) <= norm z /\ &2 < z$1 ==> &2 < norm z`) THEN SIMP_TAC[COMPONENT_LE_NORM; DIMINDEX_2; ARITH] THEN SIMP_TAC[VECTOR_SUB_COMPONENT; VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH] THEN ASM_REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `!z:real^2. z IN path_image c ==> abs(z$2) <= &9 / &5` (LABEL_TAC "ybound") THENL [REPEAT STRIP_TAC THEN FIRST_X_ASSUM(fun th -> MP_TAC(ISPECL [`basis 1:real^2`; `z:real^2`] th) THEN MP_TAC(ISPECL [`--basis 1:real^2`; `z:real^2`] th)) THEN SUBST1_TAC(REAL_ARITH `&9 / &5 = abs(&9 / &5)`) THEN ASM_REWRITE_TAC[NORM_LE_SQUARE; REAL_LE_SQUARE_ABS] THEN SIMP_TAC[DOT_2; VECTOR_SUB_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN MP_TAC(ISPEC `(z:real^2)$1` REAL_LE_SQUARE) THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `!z:real^2. z IN path_image c ==> abs(z$2) < &2` (LABEL_TAC "ybounds") THENL [ASM_MESON_TAC[REAL_ARITH `x <= &9 / &5 ==> x < &2`]; ALL_TAC] THEN SUBGOAL_THEN `?t. t$1 = &0 /\ t IN path_image c /\ !z:real^2. z$1 = &0 /\ z IN path_image c ==> z$2 <= t$2` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`\z:real^2. z$2`; `{z:real^2 | z$1 = &0} INTER path_image c`] CONTINUOUS_ATTAINS_SUP) THEN SIMP_TAC[CONTINUOUS_ON_LIFT_COMPONENT; DIMINDEX_2; ARITH; o_DEF] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM; GSYM CONJ_ASSOC] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_SIMP_TAC[CLOSED_INTER_COMPACT; COMPACT_PATH_IMAGE; CLOSED_STANDARD_HYPERPLANE; SIMPLE_PATH_IMP_PATH] THEN ONCE_REWRITE_TAC[INTER_COMM] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM] THEN MATCH_MP_TAC CONNECTED_IVT_COMPONENT THEN ASM_SIMP_TAC[CONNECTED_PATH_IMAGE; SIMPLE_PATH_IMP_PATH] THEN MAP_EVERY EXISTS_TAC [`--basis 1:real^2`; `basis 1:real^2`] THEN ASM_SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `(abs((t:real^2)$2) <= &9 / &5 /\ abs(t$2) < &2) /\ t IN interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ t IN interval[--vector[&1; &2],vector[&1; &2]]` STRIP_ASSUME_TAC THENL [MATCH_MP_TAC(TAUT `a /\ (a ==> b) ==> a /\ b`) THEN CONJ_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `?U D. arc U /\ arc D /\ pathstart U = --basis 1 /\ pathfinish U = basis 1 /\ pathstart D = basis 1 /\ pathfinish D = --basis 1 /\ (t:real^2) IN path_image U /\ (path_image U) INTER (path_image D) = {--basis 1,basis 1} /\ path_image (c:real^1->real^2) = path_image(U) UNION path_image(D)` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`c:real^1->real^2`; `--basis 1:real^2`; `basis 1:real^2`] EXISTS_DOUBLE_ARC) THEN ASM_REWRITE_TAC[VECTOR_ARITH `--b = b <=> b = vec 0`] THEN ASM_SIMP_TAC[BASIS_NONZERO; DIMINDEX_GE_1; LE_REFL] THEN REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`U:real^1->real^2`; `D:real^1->real^2`] THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN DISCH_THEN(SUBST_ALL_TAC o SYM) THEN UNDISCH_TAC `(t:real^2) IN path_image U UNION path_image D` THEN ASM_SIMP_TAC[PATH_IMAGE_JOIN; ARC_IMP_PATH; IN_UNION] THEN STRIP_TAC THENL [MAP_EVERY EXISTS_TAC [`U:real^1->real^2`; `D:real^1->real^2`] THEN ASM_REWRITE_TAC[]; MAP_EVERY EXISTS_TAC [`reversepath D:real^1->real^2`; `reversepath U:real^1->real^2`] THEN ONCE_REWRITE_TAC[INTER_COMM] THEN ASM_SIMP_TAC[PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATH_IMAGE_JOIN; PATH_IMAGE_REVERSEPATH; SIMPLE_PATH_JOIN_LOOP; ARC_REVERSEPATH] THEN ASM SET_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `path_image U INTER path_image D SUBSET {--basis 1:real^2, basis 1}` ASSUME_TAC THENL [ASM_REWRITE_TAC[SUBSET_REFL]; ALL_TAC] THEN SUBGOAL_THEN `simple_path(U ++ D:real^1->real^2)` ASSUME_TAC THENL [MATCH_MP_TAC SIMPLE_PATH_JOIN_LOOP THEN ASM_REWRITE_TAC[SUBSET_REFL]; ALL_TAC] THEN SUBGOAL_THEN `path_image(U ++ D:real^1->real^2) = path_image(U) UNION path_image(D)` ASSUME_TAC THENL [ASM_SIMP_TAC[PATH_IMAGE_JOIN; ARC_IMP_PATH]; ALL_TAC] THEN UNDISCH_THEN `path_image c :real^2->bool = path_image U UNION path_image D` SUBST_ALL_TAC THEN POP_ASSUM_LIST(MAP_EVERY (fun th -> if free_in `c:real^1->real^2` (concl th) then ALL_TAC else ASSUME_TAC th) o rev) THEN SUBGOAL_THEN `path_image(U:real^1->real^2) UNION path_image(D) SUBSET interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ path_image(U) UNION path_image(D) SUBSET interval[--vector[&1; &2],vector[&1; &2]]` MP_TAC THENL [MATCH_MP_TAC(SET_RULE `p SUBSET s /\ s SUBSET t ==> p SUBSET s /\ p SUBSET t`) THEN ASM_SIMP_TAC[SUBSET_INTERVAL; SUBSET; DIMINDEX_2; FORALL_2; VECTOR_2; VECTOR_NEG_COMPONENT; IN_INTERVAL; REAL_BOUNDS_LE] THEN CONV_TAC REAL_RAT_REDUCE_CONV; REWRITE_TAC[UNION_SUBSET] THEN STRIP_TAC] THEN SUBGOAL_THEN `?p. p$1 = &0 /\ p IN path_image U /\ !z:real^2. z$1 = &0 /\ z IN path_image U ==> p$2 <= z$2` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`\z:real^2. z$2`; `{z:real^2 | z$1 = &0} INTER path_image U`] CONTINUOUS_ATTAINS_INF) THEN SIMP_TAC[CONTINUOUS_ON_LIFT_COMPONENT; DIMINDEX_2; ARITH; o_DEF] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM; GSYM CONJ_ASSOC] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_SIMP_TAC[CLOSED_INTER_COMPACT; COMPACT_PATH_IMAGE; CLOSED_STANDARD_HYPERPLANE; ARC_IMP_PATH] THEN ONCE_REWRITE_TAC[INTER_COMM] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM] THEN MATCH_MP_TAC CONNECTED_IVT_COMPONENT THEN ASM_SIMP_TAC[CONNECTED_PATH_IMAGE; ARC_IMP_PATH] THEN MAP_EVERY EXISTS_TAC [`pathstart U:real^2`; `pathfinish U:real^2`] THEN REWRITE_TAC[PATHSTART_IN_PATH_IMAGE; PATHFINISH_IN_PATH_IMAGE] THEN ASM_SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `(p:real^2)$2 <= (t:real^2)$2` ASSUME_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `(abs((p:real^2)$2) <= &9 / &5 /\ abs(p$2) < &2) /\ p IN interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ p IN interval[--vector[&1; &2],vector[&1; &2]]` STRIP_ASSUME_TAC THENL [MATCH_MP_TAC(TAUT `a /\ (a ==> b) ==> a /\ b`) THEN CONJ_TAC THENL [ASM_MESON_TAC[IN_UNION]; ALL_TAC] THEN ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN MP_TAC(ISPECL [`U:real^1->real^2`; `t:real^2`; `p:real^2`] EXISTS_SUBPATH_OF_ARC_NOENDS) THEN ANTS_TAC THENL [ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `~(a = c) /\ ~(a = d) /\ ~(b = c) /\ ~(b = d) ==> {a,b} INTER {c,d} = {}`) THEN REPEAT CONJ_TAC THEN DISCH_THEN(MP_TAC o AP_TERM `\z:real^2. z$1`) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH] THEN REAL_ARITH_TAC; DISCH_THEN(X_CHOOSE_THEN `h:real^1->real^2` STRIP_ASSUME_TAC)] THEN SUBGOAL_THEN `~((t:real^2) IN path_image D) /\ ~(p IN path_image D)` STRIP_ASSUME_TAC THENL [REPEAT CONJ_TAC THEN UNDISCH_TAC `path_image U INTER path_image D SUBSET {--basis 1:real^2, basis 1}` THEN MATCH_MP_TAC(SET_RULE `(t IN s ==> t IN u) /\ ~(t IN v) ==> u SUBSET v ==> ~(t IN s)`) THEN ASM_SIMP_TAC[IN_INTER] THEN REWRITE_TAC[IN_INSERT; NOT_IN_EMPTY] THEN DISCH_THEN(DISJ_CASES_THEN (MP_TAC o AP_TERM `\z:real^2. z$1`)) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `?q. q$2 <= (p:real^2)$2 /\ q$1 = &0 /\ q IN path_image D /\ !z:real^2. z$2 <= p$2 /\ z$1 = &0 /\ z IN path_image D ==> z$2 <= q$2` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`\z:real^2. z$2`; `{z:real^2 | z$2 <= (p:real^2)$2} INTER {z:real^2 | z$1 = &0} INTER path_image D`] CONTINUOUS_ATTAINS_SUP) THEN SIMP_TAC[CONTINUOUS_ON_LIFT_COMPONENT; DIMINDEX_2; ARITH; o_DEF] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM; GSYM CONJ_ASSOC] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_SIMP_TAC[CLOSED_INTER_COMPACT; COMPACT_PATH_IMAGE; CLOSED_STANDARD_HYPERPLANE; ARC_IMP_PATH] THEN CONJ_TAC THENL [REWRITE_TAC[GSYM INTER_ASSOC] THEN MATCH_MP_TAC CLOSED_INTER_COMPACT THEN ASM_SIMP_TAC[COMPACT_PATH_IMAGE; ARC_IMP_PATH] THEN ASM_SIMP_TAC[CLOSED_INTER; CLOSED_STANDARD_HYPERPLANE; CLOSED_HALFSPACE_COMPONENT_LE]; ALL_TAC] THEN MP_TAC(ISPECL [`reversepath D:real^1->real^2`; `linepath(--(&2 % basis 2):real^2,p) ++ reversepath h ++ linepath(t,&2 % basis 2)`; `--vector[&1; &2]:real^2`; `vector[&1; &2]:real^2`] FASHODA) THEN ASM_SIMP_TAC[ARC_IMP_PATH; PATH_REVERSEPATH; PATH_JOIN; PATH_IMAGE_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_LINEPATH; PATH_IMAGE_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH] THEN ANTS_TAC THENL [ASM_SIMP_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH; SEGMENT_CONVEX_HULL; SUBSET_HULL; CONVEX_INTERVAL; INSERT_SUBSET; EMPTY_SUBSET] THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_NEG_COMPONENT; VECTOR_MUL_COMPONENT; VECTOR_2; BASIS_COMPONENT; ARITH] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN ASM SET_TAC[]; ALL_TAC] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY; IN_INTER; IN_ELIM_THM] THEN MATCH_MP_TAC MONO_EXISTS THEN X_GEN_TAC `z:real^2` THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ASM_REWRITE_TAC[IN_UNION; PATH_IMAGE_LINEPATH] THEN STRIP_TAC THENL [MP_TAC(ISPECL [`--(&2 % basis 2):real^2`; `p:real^2`; `z:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_MUL_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN ASM_REAL_ARITH_TAC; UNDISCH_TAC `path_image(h:real^1->real^2) SUBSET path_image U DIFF {pathstart U, pathfinish U}` THEN ASM_REWRITE_TAC[SUBSET] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN ASM SET_TAC[]; MP_TAC(ISPECL [`t:real^2`; `&2 % basis 2:real^2`; `z:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_MUL_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN ASM_CASES_TAC `z:real^2 = t` THENL [ASM_MESON_TAC[]; UNDISCH_TAC `~(z:real^2 = t)`] THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2] THEN ASM_CASES_TAC `(z:real^2)$1 = &0` THEN ASM_REWRITE_TAC[REAL_MUL_RZERO] THEN SUBGOAL_THEN `(z:real^2)$2 <= (t:real^2)$2` MP_TAC THENL [ASM SET_TAC[]; ASM_REAL_ARITH_TAC]]; ALL_TAC] THEN SUBGOAL_THEN `?b. b$1 = &0 /\ b IN path_image D /\ !z:real^2. z$1 = &0 /\ z IN path_image D ==> b$2 <= z$2` STRIP_ASSUME_TAC THENL [MP_TAC(ISPECL [`\z:real^2. z$2`; `{z:real^2 | z$1 = &0} INTER path_image D`] CONTINUOUS_ATTAINS_INF) THEN SIMP_TAC[CONTINUOUS_ON_LIFT_COMPONENT; DIMINDEX_2; ARITH; o_DEF] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM; GSYM CONJ_ASSOC] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_SIMP_TAC[CLOSED_INTER_COMPACT; COMPACT_PATH_IMAGE; CLOSED_STANDARD_HYPERPLANE; ARC_IMP_PATH] THEN ONCE_REWRITE_TAC[INTER_COMM] THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN REWRITE_TAC[IN_INTER; IN_ELIM_THM] THEN MATCH_MP_TAC CONNECTED_IVT_COMPONENT THEN ASM_SIMP_TAC[CONNECTED_PATH_IMAGE; ARC_IMP_PATH] THEN MAP_EVERY EXISTS_TAC [`pathfinish D:real^2`; `pathstart D:real^2`] THEN REWRITE_TAC[PATHSTART_IN_PATH_IMAGE; PATHFINISH_IN_PATH_IMAGE] THEN ASM_SIMP_TAC[BASIS_COMPONENT; DIMINDEX_2; ARITH; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN MP_TAC(ISPECL [`D:real^1->real^2`; `q:real^2`; `b:real^2`] EXISTS_SUBPATH_OF_ARC_NOENDS) THEN ANTS_TAC THENL [ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `~(a = c) /\ ~(a = d) /\ ~(b = c) /\ ~(b = d) ==> {a,b} INTER {c,d} = {}`) THEN REPEAT CONJ_TAC THEN DISCH_THEN(MP_TAC o AP_TERM `\z:real^2. z$1`) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH] THEN REAL_ARITH_TAC; DISCH_THEN(X_CHOOSE_THEN `l:real^1->real^2` STRIP_ASSUME_TAC)] THEN SUBGOAL_THEN `~((q:real^2) IN path_image U) /\ ~(b IN path_image U)` STRIP_ASSUME_TAC THENL [REPEAT CONJ_TAC THEN UNDISCH_TAC `path_image U INTER path_image D SUBSET {--basis 1:real^2, basis 1}` THEN MATCH_MP_TAC(SET_RULE `(t IN s ==> t IN u) /\ ~(t IN v) ==> u SUBSET v ==> ~(t IN s)`) THEN ASM_SIMP_TAC[IN_INTER] THEN REWRITE_TAC[IN_INSERT; NOT_IN_EMPTY] THEN DISCH_THEN(DISJ_CASES_THEN (MP_TAC o AP_TERM `\z:real^2. z$1`)) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `~(p:real^2 = q)` ASSUME_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `(q:real^2)$2 < (p:real^2)$2` ASSUME_TAC THENL [ASM_REWRITE_TAC[REAL_LT_LE] THEN DISCH_TAC THEN UNDISCH_TAC `~(p:real^2 = q)` THEN ASM_REWRITE_TAC[DIMINDEX_2; FORALL_2; CART_EQ]; ALL_TAC] THEN ABBREV_TAC `y:real^2 = midpoint(p,q)` THEN EXISTS_TAC `connected_component ((:real^2) DIFF path_image(U ++ D)) y` THEN EXISTS_TAC `connected_component ((:real^2) DIFF path_image(U ++ D)) (&2 % basis 2)` THEN REWRITE_TAC[CONNECTED_COMPONENT_EQ_EMPTY; CONNECTED_CONNECTED_COMPONENT] THEN ABBREV_TAC `K = (:real^2) DIFF path_image(U ++ D)` THEN SUBGOAL_THEN `open(K:real^2->bool)` ASSUME_TAC THENL [EXPAND_TAC "K" THEN REWRITE_TAC[GSYM closed] THEN ASM_SIMP_TAC[COMPACT_PATH_IMAGE; SIMPLE_PATH_IMP_PATH; COMPACT_IMP_CLOSED]; ALL_TAC] THEN ASM_SIMP_TAC[OPEN_CONNECTED_COMPONENT] THEN ABBREV_TAC `n:real^2 = &2 % basis 2` THEN SUBGOAL_THEN `(y:real^2)$1 = &0 /\ (n:real^2)$1 = &0` STRIP_ASSUME_TAC THENL [MAP_EVERY EXPAND_TAC ["y";
"n"] THEN SIMP_TAC[BASIS_COMPONENT; VECTOR_MUL_COMPONENT; DIMINDEX_2; ARITH] THEN REWRITE_TAC[midpoint; VECTOR_ADD_COMPONENT; VECTOR_MUL_COMPONENT] THEN ASM_REWRITE_TAC[] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN SUBGOAL_THEN `(y:real^2)$2 = ((p:real^2)$2 + (q:real^2)$2) / &2` ASSUME_TAC THENL [EXPAND_TAC "y" THEN REWRITE_TAC[midpoint; VECTOR_ADD_COMPONENT; VECTOR_MUL_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `(n:real^2)$2 = &2` ASSUME_TAC THENL [EXPAND_TAC "n" THEN SIMP_TAC[BASIS_COMPONENT; VECTOR_MUL_COMPONENT; DIMINDEX_2; ARITH] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `(b:real^2)$2 <= (q:real^2)$2` ASSUME_TAC THENL [ASM_MESON_TAC[IN_UNION]; ALL_TAC] THEN SUBGOAL_THEN `((abs((q:real^2)$2) <= &9 / &5 /\ abs(q$2) < &2) /\ q IN interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ q IN interval[--vector[&1; &2],vector[&1; &2]]) /\ ((abs((b:real^2)$2) <= &9 / &5 /\ abs(b$2) < &2) /\ b IN interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ b IN interval[--vector[&1; &2],vector[&1; &2]])` STRIP_ASSUME_TAC THENL [CONJ_TAC THEN MATCH_MP_TAC(TAUT `a /\ (a ==> b) ==> a /\ b`) THEN (CONJ_TAC THENL [ASM_MESON_TAC[IN_UNION]; ALL_TAC]) THEN ASM_REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `path_image(h:real^1->real^2) SUBSET interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ path_image h SUBSET interval[--vector[&1; &2],vector[&1; &2]] /\ path_image l SUBSET interval[--vector[&1; &9 / &5],vector[&1; &9 / &5]] /\ path_image(l:real^1->real^2) SUBSET interval[--vector[&1; &2],vector[&1; &2]] /\ ~(basis 1 IN path_image h) /\ ~(basis 1 IN path_image l) /\ ~(--basis 1 IN path_image h) /\ ~(--basis 1 IN path_image l)` STRIP_ASSUME_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `(n:real^2) IN interval[--vector[&1; &2],vector[&1; &2]] /\ (--n) IN interval[--vector[&1; &2],vector[&1; &2]]` STRIP_ASSUME_TAC THENL [ASM_REWRITE_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN SUBGOAL_THEN `(q:real^2)$2 < (y:real^2)$2 /\ (y:real^2)$2 < (p:real^2)$2 /\ (q:real^2)$2 <= (y:real^2)$2 /\ (y:real^2)$2 <= (p:real^2)$2` STRIP_ASSUME_TAC THENL [ASM_REAL_ARITH_TAC; ALL_TAC] THEN REWRITE_TAC[SYM(ASSUME `path_image(U ++ D):real^2->bool = path_image U UNION path_image D`)] THEN ASM_REWRITE_TAC[] THEN SUBGOAL_THEN `(y:real^2) IN K` ASSUME_TAC THENL [EXPAND_TAC "K" THEN REWRITE_TAC[IN_UNIV; IN_DIFF] THEN ASM_REWRITE_TAC[IN_UNION; DE_MORGAN_THM] THEN ASM_MESON_TAC[REAL_NOT_LE]; ASM_REWRITE_TAC[]] THEN SUBGOAL_THEN `(n:real^2) IN K` ASSUME_TAC THENL [EXPAND_TAC "K" THEN ASM_REWRITE_TAC[IN_DIFF; IN_UNIV] THEN ASM_MESON_TAC[REAL_ARITH `~(abs x < x)`]; ASM_REWRITE_TAC[]] THEN MATCH_MP_TAC(TAUT `(a /\ b ==> c) /\ b /\ a /\ d ==> a /\ b /\ c /\ d`) THEN CONJ_TAC THENL [MESON_TAC[CONNECTED_COMPONENT_NONOVERLAP]; ALL_TAC] THEN SUBGOAL_THEN `(y:real^2) IN interval[--vector [&1; &9 / &5],vector [&1; &9 / &5]] /\ (y:real^2) IN interval[--vector [&1; &2],vector [&1; &2]]` STRIP_ASSUME_TAC THENL [REWRITE_TAC[IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN ASM_REAL_ARITH_TAC; ALL_TAC] THEN SUBGOAL_THEN `((:real^2) DIFF interval[--vector [&1; &9 / &5],vector [&1; &9 / &5]]) SUBSET connected_component K n /\ ((:real^2) DIFF interval[--vector [&1; &2],vector [&1; &2]]) SUBSET connected_component K n` STRIP_ASSUME_TAC THENL [MATCH_MP_TAC(SET_RULE `i SUBSET j /\ UNIV DIFF i SUBSET s ==> UNIV DIFF i SUBSET s /\ UNIV DIFF j SUBSET s`) THEN REWRITE_TAC[SUBSET_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN MATCH_MP_TAC CONNECTED_COMPONENT_MAXIMAL THEN SIMP_TAC[CONNECTED_COMPLEMENT_BOUNDED_CONVEX; LE_REFL; DIMINDEX_2; BOUNDED_INTERVAL; CONVEX_INTERVAL] THEN EXPAND_TAC "K" THEN ASM_REWRITE_TAC[SET_RULE `UNIV DIFF s SUBSET UNIV DIFF t <=> t SUBSET s`; UNION_SUBSET] THEN ASM_REWRITE_TAC[IN_UNIV; IN_DIFF; IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN SUBGOAL_THEN `~bounded(connected_component K (n:real^2))` ASSUME_TAC THENL [MATCH_MP_TAC COBOUNDED_IMP_UNBOUNDED THEN MATCH_MP_TAC BOUNDED_SUBSET THEN EXISTS_TAC `interval[--vector[&1; &2]:real^2,vector [&1; &2]]` THEN ONCE_REWRITE_TAC[SET_RULE `UNIV DIFF s SUBSET t <=> UNIV DIFF t SUBSET s`] THEN ASM_REWRITE_TAC[BOUNDED_INTERVAL]; ASM_REWRITE_TAC[]] THEN SUBGOAL_THEN `bounded(connected_component K (y:real^2))` ASSUME_TAC THENL [REWRITE_TAC[bounded] THEN EXISTS_TAC `&4` THEN X_GEN_TAC `z:real^2` THEN REWRITE_TAC[GSYM REAL_NOT_LT] THEN ASM_SIMP_TAC[GSYM OPEN_PATH_CONNECTED_COMPONENT] THEN REWRITE_TAC[path_component; IN] THEN DISCH_THEN(X_CHOOSE_THEN `i:real^1->real^2` STRIP_ASSUME_TAC) THEN STRIP_TAC THEN MP_TAC(ISPECL [`i:real^1->real^2`; `interval[--vector[&1; &2]:real^2,vector[&1; &2]]`] EXISTS_PATH_SUBPATH_TO_FRONTIER_CLOSED) THEN ASM_REWRITE_TAC[CLOSED_INTERVAL; NOT_IMP; SUBSET_INTER; GSYM CONJ_ASSOC] THEN CONJ_TAC THENL [REWRITE_TAC[IN_INTERVAL; VECTOR_2; DIMINDEX_2; FORALL_2; VECTOR_NEG_COMPONENT; GSYM REAL_ABS_BOUNDS] THEN STRIP_TAC THEN UNDISCH_TAC `&4 < norm(z:real^2)` THEN REWRITE_TAC[REAL_NOT_LT] THEN W(MP_TAC o PART_MATCH lhand NORM_LE_L1 o lhand o snd) THEN REWRITE_TAC[DIMINDEX_2; SUM_2] THEN ASM_REAL_ARITH_TAC; REWRITE_TAC[IN_DIFF; FRONTIER_CLOSED_INTERVAL; IN_INTERVAL; FORALL_2; DIMINDEX_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN DISCH_THEN(X_CHOOSE_THEN `j:real^1->real^2` MP_TAC) THEN REPEAT(DISCH_THEN(CONJUNCTS_THEN2 STRIP_ASSUME_TAC MP_TAC)) THEN ASM_REWRITE_TAC[REAL_LT_LE; REAL_ARITH `x:real = i$j <=> i$j = x`] THEN REWRITE_TAC[GSYM DE_MORGAN_THM; GSYM DISJ_ASSOC] THEN DISCH_TAC] THEN SUBGOAL_THEN `path_image(j:real^1->real^2) SUBSET K` ASSUME_TAC THENL [ASM_MESON_TAC[SUBSET_TRANS]; ALL_TAC] THEN SUBGOAL_THEN `~(pathfinish(j):real^2 = basis 1) /\ ~(pathfinish(j):real^2 = --basis 1)` STRIP_ASSUME_TAC THENL [REPEAT STRIP_TAC THEN UNDISCH_THEN `path_image(j:real^1->real^2) SUBSET K` (MP_TAC o SPEC `pathfinish(j):real^2` o REWRITE_RULE[SUBSET]) THEN ASM_REWRITE_TAC[PATHFINISH_IN_PATH_IMAGE] THEN EXPAND_TAC "K" THEN ASM_REWRITE_TAC[IN_DIFF; IN_UNIV; IN_UNION]; ALL_TAC] THEN SUBGOAL_THEN `?e:real^1->real^2. path e /\ pathstart e = y /\ ((pathfinish e)$2 = -- &2 \/ (pathfinish e)$2 = &2) /\ path_image e SUBSET K /\ path_image e SUBSET interval[--vector [&1; &2],vector [&1; &2]]` MP_TAC THENL [FIRST_X_ASSUM(DISJ_CASES_THEN STRIP_ASSUME_TAC) THENL [UNDISCH_TAC `~((pathfinish j):real^2 = --basis 1)` THEN SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH] THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_TAC o MATCH_MP (REAL_ARITH `~(x = -- &0) ==> x < &0 \/ &0 < x`)) THENL [EXISTS_TAC `(j:real^1->real^2) ++ linepath(pathfinish j,--vector [&1; &2])` THEN ASM_SIMP_TAC[PATH_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_IMAGE_JOIN] THEN REWRITE_TAC[VECTOR_2; VECTOR_NEG_COMPONENT] THEN REWRITE_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH] THEN ONCE_REWRITE_TAC[AC CONJ_ACI `(a /\ b) /\ c /\ d <=> (a /\ c) /\ b /\ d`] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN EXPAND_TAC "K" THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> x IN t /\ ~(x IN u)) ==> s SUBSET (UNIV DIFF u) /\ s SUBSET t`) THEN X_GEN_TAC `x:real^2` THEN DISCH_TAC THEN MP_TAC(ISPECL [`pathfinish(j:real^1->real^2)`; `--vector[&1; &2]:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; VECTOR_2; IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN DISCH_TAC THEN CONJ_TAC THENL [ASM_REAL_ARITH_TAC; ALL_TAC] THEN DISCH_TAC THEN UNDISCH_THEN `!z:real^2. z IN path_image U UNION path_image D /\ z$1 = -- &1 <=> z = --basis 1` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REAL_ARITH_TAC; EXISTS_TAC `(j:real^1->real^2) ++ linepath(pathfinish j,vector [-- &1; &2])` THEN ASM_SIMP_TAC[PATH_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_IMAGE_JOIN] THEN REWRITE_TAC[VECTOR_2; VECTOR_NEG_COMPONENT] THEN REWRITE_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH] THEN ONCE_REWRITE_TAC[AC CONJ_ACI `(a /\ b) /\ c /\ d <=> (a /\ c) /\ b /\ d`] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN EXPAND_TAC "K" THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> x IN t /\ ~(x IN u)) ==> s SUBSET (UNIV DIFF u) /\ s SUBSET t`) THEN X_GEN_TAC `x:real^2` THEN DISCH_TAC THEN MP_TAC(ISPECL [`pathfinish(j:real^1->real^2)`; `vector[-- &1; &2]:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; VECTOR_2; IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN DISCH_TAC THEN CONJ_TAC THENL [ASM_REAL_ARITH_TAC; ALL_TAC] THEN DISCH_TAC THEN UNDISCH_THEN `!z:real^2. z IN path_image U UNION path_image D /\ z$1 = -- &1 <=> z = --basis 1` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REAL_ARITH_TAC]; UNDISCH_TAC `~((pathfinish j):real^2 = basis 1)` THEN SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH] THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_TAC o MATCH_MP (REAL_ARITH `~(x = &0) ==> x < &0 \/ &0 < x`)) THENL [EXISTS_TAC `(j:real^1->real^2) ++ linepath(pathfinish j,vector [&1; -- &2])` THEN ASM_SIMP_TAC[PATH_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_IMAGE_JOIN] THEN REWRITE_TAC[VECTOR_2; VECTOR_NEG_COMPONENT] THEN REWRITE_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH] THEN ONCE_REWRITE_TAC[AC CONJ_ACI `(a /\ b) /\ c /\ d <=> (a /\ c) /\ b /\ d`] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN EXPAND_TAC "K" THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> x IN t /\ ~(x IN u)) ==> s SUBSET (UNIV DIFF u) /\ s SUBSET t`) THEN X_GEN_TAC `x:real^2` THEN DISCH_TAC THEN MP_TAC(ISPECL [`pathfinish(j:real^1->real^2)`; `vector[&1; -- &2]:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; VECTOR_2; IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN DISCH_TAC THEN CONJ_TAC THENL [ASM_REAL_ARITH_TAC; ALL_TAC] THEN DISCH_TAC THEN UNDISCH_THEN `!z:real^2. z IN path_image U UNION path_image D /\ z$1 = &1 <=> z = basis 1` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REAL_ARITH_TAC; EXISTS_TAC `(j:real^1->real^2) ++ linepath(pathfinish j,vector [&1; &2])` THEN ASM_SIMP_TAC[PATH_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATH_IMAGE_JOIN] THEN REWRITE_TAC[VECTOR_2; VECTOR_NEG_COMPONENT] THEN REWRITE_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH] THEN ONCE_REWRITE_TAC[AC CONJ_ACI `(a /\ b) /\ c /\ d <=> (a /\ c) /\ b /\ d`] THEN CONJ_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN EXPAND_TAC "K" THEN MATCH_MP_TAC(SET_RULE `(!x. x IN s ==> x IN t /\ ~(x IN u)) ==> s SUBSET (UNIV DIFF u) /\ s SUBSET t`) THEN X_GEN_TAC `x:real^2` THEN DISCH_TAC THEN MP_TAC(ISPECL [`pathfinish(j:real^1->real^2)`; `vector[&1; &2]:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_SIMP_TAC[VECTOR_NEG_COMPONENT; VECTOR_2; IN_INTERVAL; FORALL_2; DIMINDEX_2] THEN DISCH_TAC THEN CONJ_TAC THENL [ASM_REAL_ARITH_TAC; ALL_TAC] THEN DISCH_TAC THEN UNDISCH_THEN `!z:real^2. z IN path_image U UNION path_image D /\ z$1 = &1 <=> z = basis 1` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[CART_EQ; DIMINDEX_2; FORALL_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REAL_ARITH_TAC]; EXISTS_TAC `j:real^1->real^2` THEN ASM_REWRITE_TAC[]; EXISTS_TAC `j:real^1->real^2` THEN ASM_REWRITE_TAC[]]; FIRST_X_ASSUM(K ALL_TAC o check (is_disj o concl))] THEN STRIP_TAC THENL [MP_TAC(ISPECL [`reversepath(D:real^1->real^2)`; `reversepath e ++ linepath(y,p) ++ reversepath h ++ linepath(t:real^2,vector[t$1;&2])`; `--vector[&1; &2]:real^2`; `vector[&1; &2]:real^2`] FASHODA) THEN ASM_SIMP_TAC [PATH_REVERSEPATH; ARC_IMP_PATH; PATH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATH_IMAGE_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_IMAGE_REVERSEPATH] THEN SIMP_TAC[VECTOR_2; VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH; NOT_IMP; GSYM CONJ_ASSOC] THEN CONJ_TAC THENL [ASM_SIMP_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH; SEGMENT_CONVEX_HULL; SUBSET_HULL; CONVEX_INTERVAL; INSERT_SUBSET; EMPTY_SUBSET] THEN REWRITE_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_NEG_COMPONENT; VECTOR_2] THEN CONV_TAC REAL_RAT_REDUCE_CONV; DISCH_THEN(X_CHOOSE_THEN `x:real^2` (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN REWRITE_TAC[DE_MORGAN_THM; IN_UNION] THEN REPEAT CONJ_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN DISCH_TAC THEN MP_TAC(ISPECL [`y:real^2`; `p:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[] THEN UNDISCH_THEN `!z:real^2. z$2 <= (p:real^2)$2 /\ z$1 = &0 /\ z IN path_image D ==> z$2 <= (q:real^2)$2` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_REAL_ARITH_TAC; DISCH_TAC THEN UNDISCH_THEN `path_image(h:real^1->real^2) SUBSET (path_image U) DIFF {pathstart U, pathfinish U}` (MP_TAC o SPEC `x:real^2` o REWRITE_RULE[SUBSET]) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN DISCH_TAC THEN MP_TAC(ISPECL [`t:real^2`; `vector[&0;&2]:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[VECTOR_2] THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ASM_CASES_TAC `x:real^2 = t` THENL [ASM_MESON_TAC[]; UNDISCH_TAC `~(x:real^2 = t)`] THEN ASM_REWRITE_TAC[CART_EQ; DIMINDEX_2; FORALL_2] THEN UNDISCH_THEN `!z:real^2. z$1 = &0 /\ z IN path_image U UNION path_image D ==> z$2 <= (t:real^2)$2` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[IN_UNION] THEN ASM_REAL_ARITH_TAC]]; MP_TAC(ISPECL [`U:real^1->real^2`; `linepath(vector[b$1; -- &2],b) ++ reversepath l ++ linepath(q:real^2,y) ++ e`; `--vector[&1; &2]:real^2`; `vector[&1; &2]:real^2`] FASHODA) THEN ASM_SIMP_TAC [PATH_REVERSEPATH; ARC_IMP_PATH; PATH_JOIN; PATH_LINEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATH_IMAGE_JOIN; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_IMAGE_REVERSEPATH] THEN SIMP_TAC[VECTOR_2; VECTOR_NEG_COMPONENT; BASIS_COMPONENT; DIMINDEX_2; ARITH; NOT_IMP; GSYM CONJ_ASSOC] THEN CONJ_TAC THENL [ASM_SIMP_TAC[UNION_SUBSET; PATH_IMAGE_LINEPATH; SEGMENT_CONVEX_HULL; SUBSET_HULL; CONVEX_INTERVAL; INSERT_SUBSET; EMPTY_SUBSET] THEN REWRITE_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_NEG_COMPONENT; VECTOR_2] THEN CONV_TAC REAL_RAT_REDUCE_CONV; DISCH_THEN(X_CHOOSE_THEN `x:real^2` (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THEN REWRITE_TAC[DE_MORGAN_THM; IN_UNION] THEN REPEAT CONJ_TAC THENL [REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN DISCH_TAC THEN MP_TAC(ISPECL [`vector[&0; -- &2]:real^2`; `b:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[VECTOR_2] THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN ASM_CASES_TAC `x:real^2 = b` THENL [ASM_MESON_TAC[]; UNDISCH_TAC `~(x:real^2 = b)`] THEN ASM_REWRITE_TAC[CART_EQ; DIMINDEX_2; FORALL_2] THEN UNDISCH_THEN `!z:real^2. z$1 = &0 /\ z IN path_image U ==> (p:real^2)$2 <= z$2` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_REAL_ARITH_TAC; DISCH_TAC THEN UNDISCH_THEN `path_image(l:real^1->real^2) SUBSET (path_image D) DIFF {pathstart D, pathfinish D}` (MP_TAC o SPEC `x:real^2` o REWRITE_RULE[SUBSET]) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; REWRITE_TAC[PATH_IMAGE_LINEPATH] THEN DISCH_TAC THEN MP_TAC(ISPECL [`q:real^2`; `y:real^2`; `x:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[] THEN UNDISCH_THEN `!z:real^2. z$1 = &0 /\ z IN path_image U ==> (p:real^2)$2 <= z$2` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN ASM_REAL_ARITH_TAC; ASM SET_TAC[]]]]; ASM_REWRITE_TAC[]] THEN SUBGOAL_THEN `!x:real^2. x IN K /\ ~bounded(connected_component K x) ==> connected_component K x = connected_component K n` ASSUME_TAC THENL [X_GEN_TAC `z:real^2` THEN DISCH_TAC THEN MATCH_MP_TAC COBOUNDED_UNIQUE_UNBOUNDED_COMPONENT THEN ASM_REWRITE_TAC[DIMINDEX_2; LE_REFL] THEN EXPAND_TAC "K" THEN REWRITE_TAC[SET_RULE `UNIV DIFF (UNIV DIFF s) = s`] THEN ASM_SIMP_TAC[COMPACT_PATH_IMAGE; COMPACT_IMP_BOUNDED; SIMPLE_PATH_IMP_PATH]; ALL_TAC] THEN SUBGOAL_THEN `!x:real^2. x IN K ==> frontier(connected_component K x) = path_image U UNION path_image D` ASSUME_TAC THENL [X_GEN_TAC `x:real^2` THEN DISCH_TAC THEN REWRITE_TAC[frontier] THEN ASM_SIMP_TAC[OPEN_CONNECTED_COMPONENT; INTERIOR_OPEN] THEN MATCH_MP_TAC(SET_RULE `s SUBSET t /\ ~(s PSUBSET t) ==> s = t`) THEN CONJ_TAC THENL [REWRITE_TAC[SUBSET; IN_DIFF] THEN X_GEN_TAC `w:real^2` THEN ASM_CASES_TAC `(w:real^2) IN K` THENL [ALL_TAC; ASM SET_TAC[]] THEN ASM_SIMP_TAC[IN_CLOSURE_CONNECTED_COMPONENT] THEN CONV_TAC TAUT; DISCH_TAC] THEN SUBGOAL_THEN `?A:real^1->real^2. arc A /\ (closure (connected_component K x) DIFF connected_component K x) SUBSET path_image A /\ path_image A SUBSET path_image(U) UNION path_image(D)` STRIP_ASSUME_TAC THENL [SUBST1_TAC(SYM(ASSUME `path_image(U ++ D:real^1->real^2) = path_image U UNION path_image D`)) THEN MATCH_MP_TAC EXISTS_ARC_PSUBSET_SIMPLE_PATH THEN ASM_REWRITE_TAC[] THEN ASM_SIMP_TAC[CLOSED_DIFF; CLOSED_CLOSURE; OPEN_CONNECTED_COMPONENT]; ALL_TAC] THEN MP_TAC(ISPEC `A:real^1->real^2` RETRACTION_ARC) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `rr:real^2->real^2` STRIP_ASSUME_TAC) THEN ASM_CASES_TAC `bounded(connected_component K (x:real^2))` THENL [MP_TAC(ISPECL [`connected_component K (x:real^2)`; `path_image(A:real^1->real^2)`; `rr:real^2->real^2`] FRONTIER_SUBSET_RETRACTION) THEN ASM_REWRITE_TAC[NOT_IMP; GSYM CONJ_ASSOC] THEN ASM_SIMP_TAC[frontier; INTERIOR_OPEN; OPEN_CONNECTED_COMPONENT] THEN CONJ_TAC THENL [ASM_MESON_TAC[CONTINUOUS_ON_SUBSET; SUBSET_UNIV]; ALL_TAC] THEN CONJ_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[SUBSET]] THEN DISCH_THEN(MP_TAC o SPEC `x:real^2`) THEN GEN_REWRITE_TAC (LAND_CONV o LAND_CONV) [IN] THEN ASM_REWRITE_TAC[CONNECTED_COMPONENT_REFL_EQ] THEN DISCH_TAC THEN UNDISCH_TAC `(x:real^2) IN K` THEN UNDISCH_TAC `path_image(A:real^1->real^2) SUBSET path_image U UNION path_image D` THEN REWRITE_TAC[SUBSET] THEN EXPAND_TAC "K" THEN REWRITE_TAC[IN_UNIV; IN_DIFF] THEN DISCH_THEN(MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[]; MP_TAC(ISPECL [`(:real^2) DIFF (connected_component K (x:real^2))`; `path_image(A:real^1->real^2)`; `rr:real^2->real^2`] FRONTIER_SUBSET_RETRACTION) THEN ASM_REWRITE_TAC[FRONTIER_COMPLEMENT; NOT_IMP; GSYM CONJ_ASSOC] THEN UNDISCH_THEN `!x:real^2. x IN K /\ ~bounded (connected_component K x) ==> connected_component K x = connected_component K n` (MP_TAC o SPEC `x:real^2`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN SUBST_ALL_TAC THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC BOUNDED_SUBSET THEN EXISTS_TAC `interval[--vector[&1; &2]:real^2,vector [&1; &2]]` THEN ONCE_REWRITE_TAC[SET_RULE `UNIV DIFF s SUBSET t <=> UNIV DIFF t SUBSET s`] THEN ASM_REWRITE_TAC[BOUNDED_INTERVAL]; REWRITE_TAC[frontier] THEN FIRST_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `s DIFF t SUBSET u ==> t' = t ==> s DIFF t' SUBSET u`)) THEN MATCH_MP_TAC INTERIOR_OPEN THEN MATCH_MP_TAC OPEN_CONNECTED_COMPONENT THEN ASM_REWRITE_TAC[]; MATCH_MP_TAC CONTINUOUS_ON_SUBSET THEN EXISTS_TAC `(:real^2)` THEN ASM_REWRITE_TAC[SUBSET_UNIV]; MATCH_MP_TAC(SET_RULE `IMAGE f UNIV SUBSET s ==> IMAGE f t SUBSET s`) THEN ASM_REWRITE_TAC[]; UNDISCH_TAC `path_image(A:real^1->real^2) SUBSET path_image U UNION path_image D` THEN MATCH_MP_TAC(SET_RULE `!y. y IN c /\ ~(y IN p) ==> a SUBSET p ==> ~(c SUBSET a)`) THEN EXISTS_TAC `y:real^2` THEN REWRITE_TAC[IN_DIFF; IN_UNIV] THEN CONJ_TAC THENL [DISCH_THEN(MP_TAC o MATCH_MP CONNECTED_COMPONENT_EQ) THEN DISCH_THEN SUBST_ALL_TAC THEN ASM_MESON_TAC[]; UNDISCH_TAC `(y:real^2) IN K` THEN EXPAND_TAC "K" THEN REWRITE_TAC[IN_UNIV; IN_DIFF] THEN ASM_REWRITE_TAC[]]]]; ALL_TAC] THEN CONJ_TAC THENL [ALL_TAC; ASM_MESON_TAC[]] THEN MATCH_MP_TAC SUBSET_ANTISYM THEN REWRITE_TAC[UNION_SUBSET; CONNECTED_COMPONENT_SUBSET] THEN REWRITE_TAC[SUBSET; IN_UNION; IN_ELIM_THM] THEN X_GEN_TAC `x:real^2` THEN DISCH_THEN(fun th -> ASSUME_TAC th THEN MP_TAC(MATCH_MP CONNECTED_COMPONENT_REFL th)) THEN GEN_REWRITE_TAC LAND_CONV [GSYM IN] THEN ASM_CASES_TAC `connected_component K (x:real^2) = connected_component K y` THEN ASM_REWRITE_TAC[] THEN SIMP_TAC[] THEN ASM_CASES_TAC `connected_component K (x:real^2) = connected_component K n` THEN ASM_REWRITE_TAC[] THEN SIMP_TAC[] THEN DISCH_TAC THEN MATCH_MP_TAC(TAUT `F ==> p`) THEN SUBGOAL_THEN `bounded(connected_component K (x:real^2))` ASSUME_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN ABBREV_TAC `bp = (linepath(n:real^2,t) ++ h) ++ linepath(p,q) ++ (l ++ linepath(b,--n))` THEN SUBGOAL_THEN `path_image bp SUBSET ((path_image U UNION path_image D) DIFF {--basis 1:real^2,basis 1}) UNION (connected_component K n) UNION (connected_component K y)` (LABEL_TAC "*") THENL [EXPAND_TAC "bp" THEN REPEAT(MATCH_MP_TAC SUBSET_PATH_IMAGE_JOIN THEN CONJ_TAC) THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC(SET_RULE `(path_image(linepath(n,t)) DELETE t) SUBSET nn /\ t IN uu DIFF kk ==> path_image(linepath(n,t)) SUBSET ((uu UNION dd) DIFF kk) UNION nn UNION yy`) THEN ASM_REWRITE_TAC[IN_DIFF; IN_INSERT; NOT_IN_EMPTY] THEN CONJ_TAC THENL [ALL_TAC; SIMP_TAC[CART_EQ; FORALL_2; DIMINDEX_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REWRITE_TAC[] THEN CONV_TAC REAL_RAT_REDUCE_CONV] THEN MATCH_MP_TAC CONNECTED_COMPONENT_MAXIMAL THEN SIMP_TAC[CONVEX_CONNECTED; PATH_IMAGE_LINEPATH; CONVEX_SEMIOPEN_SEGMENT; IN_DELETE; ENDS_IN_SEGMENT] THEN CONJ_TAC THENL [DISCH_THEN(MP_TAC o AP_TERM `\z:real^2. z$2`) THEN ASM_REWRITE_TAC[] THEN ASM_REAL_ARITH_TAC; ALL_TAC] THEN REWRITE_TAC[SUBSET; IN_DELETE] THEN X_GEN_TAC `w:real^2` THEN STRIP_TAC THEN MP_TAC(ISPECL [`n:real^2`; `t:real^2`; `w:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN UNDISCH_TAC `~(w:real^2 = t)` THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2; IMP_IMP] THEN DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `~(w = t) /\ (&2 <= w /\ w <= t \/ t <= w /\ w <= &2) ==> abs(t) < &2 ==> t < w`)) THEN MATCH_MP_TAC(TAUT `a /\ (~c ==> ~b) ==> (a ==> b) ==> c`) THEN CONJ_TAC THENL [FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[IN_UNION]; EXPAND_TAC "K" THEN ASM_REWRITE_TAC[IN_DIFF; IN_UNIV; REAL_NOT_LT] THEN ASM_MESON_TAC[]]; MATCH_MP_TAC SUBSET_TRANS THEN EXISTS_TAC `path_image U DIFF {pathstart U:real^2, pathfinish U}` THEN ASM_REWRITE_TAC[] THEN CONV_TAC SET_RULE; MATCH_MP_TAC(SET_RULE `(p IN uu /\ q IN dd /\ ~(p IN kk) /\ ~(q IN kk)) /\ (path_image(linepath(p,q)) DIFF {p,q}) SUBSET yy ==> path_image(linepath(p,q)) SUBSET ((uu UNION dd) DIFF kk) UNION nn UNION yy`) THEN ASM_REWRITE_TAC[IN_INSERT; NOT_IN_EMPTY] THEN CONJ_TAC THENL [SIMP_TAC[CART_EQ; FORALL_2; DIMINDEX_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REWRITE_TAC[] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN MATCH_MP_TAC CONNECTED_COMPONENT_MAXIMAL THEN REWRITE_TAC[PATH_IMAGE_LINEPATH; GSYM open_segment] THEN REWRITE_TAC[CONNECTED_SEGMENT] THEN EXPAND_TAC "y" THEN REWRITE_TAC[MIDPOINT_IN_SEGMENT] THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[open_segment; SUBSET; IN_DIFF; IN_INSERT; NOT_IN_EMPTY] THEN X_GEN_TAC `w:real^2` THEN REWRITE_TAC[DE_MORGAN_THM] THEN STRIP_TAC THEN MP_TAC(ISPECL [`p:real^2`; `q:real^2`; `w:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN MAP_EVERY UNDISCH_TAC [`~(w:real^2 = p)`; `~(w:real^2 = q)`] THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2; IMP_IMP] THEN DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `(~(w = q) /\ ~(w = p)) /\ (p <= w /\ w <= q \/ q <= w /\ w <= p) ==> q < p ==> q < w /\ w < p`)) THEN ASM_REWRITE_TAC[] THEN STRIP_TAC THEN MAP_EVERY (C UNDISCH_THEN (MP_TAC o SPEC `w:real^2`)) [`!z:real^2. z$2 <= (p:real^2)$2 /\ z$1 = &0 /\ z IN path_image D ==> z$2 <= (q:real^2)$2`; `!z:real^2. z$1 = &0 /\ z IN path_image U ==> (p:real^2)$2 <= z$2`] THEN ASM_REWRITE_TAC[GSYM REAL_NOT_LT] THEN DISCH_TAC THEN EXPAND_TAC "K" THEN ASM_REWRITE_TAC[IN_DIFF; IN_UNIV; IN_UNION] THEN MATCH_MP_TAC(TAUT `~p ==> ~(~p /\ q) ==> ~q`) THEN ASM_REAL_ARITH_TAC; MATCH_MP_TAC SUBSET_TRANS THEN EXISTS_TAC `path_image D DIFF {pathstart D:real^2, pathfinish D}` THEN ASM_REWRITE_TAC[] THEN CONV_TAC SET_RULE; MATCH_MP_TAC(SET_RULE `(path_image(linepath(b,n)) DELETE b) SUBSET nn /\ b IN dd DIFF kk ==> path_image(linepath(b,n)) SUBSET ((uu UNION dd) DIFF kk) UNION nn UNION yy`) THEN ASM_REWRITE_TAC[IN_DIFF; IN_INSERT; NOT_IN_EMPTY] THEN CONJ_TAC THENL [ALL_TAC; SIMP_TAC[CART_EQ; FORALL_2; DIMINDEX_2; BASIS_COMPONENT; VECTOR_NEG_COMPONENT; ARITH; REAL_NEG_0] THEN ASM_REWRITE_TAC[] THEN CONV_TAC REAL_RAT_REDUCE_CONV] THEN SUBGOAL_THEN `connected_component K (n:real^2) = connected_component K (--n)` SUBST1_TAC THENL [CONV_TAC SYM_CONV THEN MATCH_MP_TAC CONNECTED_COMPONENT_EQ THEN MATCH_MP_TAC(REWRITE_RULE[SUBSET] (ASSUME `(:real^2) DIFF interval[--vector [&1; &9 / &5],vector [&1; &9 / &5]] SUBSET connected_component K n`)) THEN ASM_REWRITE_TAC[IN_UNIV; IN_DIFF; IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_NEG_COMPONENT; VECTOR_2] THEN CONV_TAC REAL_RAT_REDUCE_CONV; ALL_TAC] THEN MATCH_MP_TAC CONNECTED_COMPONENT_MAXIMAL THEN SIMP_TAC[CONVEX_CONNECTED; PATH_IMAGE_LINEPATH; CONVEX_SEMIOPEN_SEGMENT; IN_DELETE; ENDS_IN_SEGMENT] THEN CONJ_TAC THENL [DISCH_THEN(MP_TAC o AP_TERM `\z:real^2. z$2`) THEN ASM_REWRITE_TAC[VECTOR_NEG_COMPONENT] THEN ASM_REAL_ARITH_TAC; ALL_TAC] THEN REWRITE_TAC[SUBSET; IN_DELETE] THEN X_GEN_TAC `w:real^2` THEN STRIP_TAC THEN MP_TAC(ISPECL [`b:real^2`; `--n:real^2`; `w:real^2`] SEGMENT_VERTICAL) THEN ASM_REWRITE_TAC[VECTOR_NEG_COMPONENT; REAL_NEG_0] THEN DISCH_THEN(CONJUNCTS_THEN2 ASSUME_TAC MP_TAC) THEN UNDISCH_TAC `~(w:real^2 = b)` THEN ASM_REWRITE_TAC[CART_EQ; FORALL_2; DIMINDEX_2; IMP_IMP] THEN DISCH_THEN(MP_TAC o MATCH_MP (REAL_ARITH `~(w = b) /\ (b <= w /\ w <= -- &2 \/ -- &2 <= w /\ w <= b) ==> abs(b) < &2 ==> w < b`)) THEN MATCH_MP_TAC(TAUT `a /\ (~c ==> ~b) ==> (a ==> b) ==> c`) THEN CONJ_TAC THENL [FIRST_X_ASSUM MATCH_MP_TAC THEN ASM_REWRITE_TAC[IN_UNION]; EXPAND_TAC "K" THEN ASM_REWRITE_TAC[IN_DIFF; IN_UNIV; IN_UNION; REAL_NOT_LT] THEN ASM_CASES_TAC `(w:real^2) IN path_image D` THEN ASM_REWRITE_TAC[] THENL [ASM_MESON_TAC[]; DISCH_TAC] THEN UNDISCH_TAC `!z. z$1 = &0 /\ z IN path_image U ==> (p:real^2)$2 <= (z:real^2)$2` THEN DISCH_THEN(MP_TAC o SPEC `w:real^2`) THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(REAL_ARITH `!q. b <= q /\ q < p ==> p <= w ==> b <= w`) THEN EXISTS_TAC `(q:real^2)$2` THEN ASM_MESON_TAC[]]]; ALL_TAC] THEN SUBGOAL_THEN `path(bp:real^1->real^2)` ASSUME_TAC THENL [EXPAND_TAC "bp" THEN REPEAT(MATCH_MP_TAC PATH_JOIN_IMP THEN REPEAT CONJ_TAC THEN ASM_REWRITE_TAC[PATHSTART_LINEPATH; PATHFINISH_LINEPATH]) THEN ASM_REWRITE_TAC[PATH_LINEPATH; PATHSTART_JOIN; PATHFINISH_JOIN; PATHSTART_LINEPATH; PATHFINISH_LINEPATH]; ALL_TAC] THEN SUBGOAL_THEN `(?d1. &0 < d1 /\ cball(--basis 1,d1) SUBSET ((:real^2) DIFF path_image bp)) /\ (?d2. &0 < d2 /\ cball(basis 1,d2) SUBSET ((:real^2) DIFF path_image bp))` STRIP_ASSUME_TAC THENL [CONJ_TAC THEN MATCH_MP_TAC(REWRITE_RULE[RIGHT_IMP_FORALL_THM; IMP_IMP] (fst(EQ_IMP_RULE(SPEC_ALL OPEN_CONTAINS_CBALL)))) THEN REWRITE_TAC[GSYM closed; IN_DIFF; IN_UNIV] THEN (CONJ_TAC THENL [ASM_MESON_TAC[COMPACT_IMP_CLOSED; COMPACT_PATH_IMAGE]; ALL_TAC]) THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `bp SUBSET s ==> ~(x IN s) ==> ~(x IN bp)`)) THEN ASM_REWRITE_TAC[IN_UNION; IN_DIFF; IN_INSERT] THEN MATCH_MP_TAC(SET_RULE `(!x. connected_component k x SUBSET k) /\ ~(a IN k) /\ ~(b IN k) ==> ~(a IN connected_component k x \/ b IN connected_component k y)`) THEN REWRITE_TAC[CONNECTED_COMPONENT_SUBSET] THEN EXPAND_TAC "K" THEN REWRITE_TAC[IN_UNIV; IN_DIFF] THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `--(basis 1) IN frontier(connected_component K (x:real^2)) /\ (basis 1) IN frontier(connected_component K x)` MP_TAC THENL [ASM_SIMP_TAC[]; ALL_TAC] THEN REWRITE_TAC[frontier] THEN ASM_SIMP_TAC[OPEN_CONNECTED_COMPONENT; INTERIOR_OPEN] THEN REWRITE_TAC[IN_DIFF] THEN DISCH_THEN(CONJUNCTS_THEN(CONJUNCTS_THEN2 MP_TAC ASSUME_TAC)) THEN REWRITE_TAC[CLOSURE_APPROACHABLE] THEN DISCH_THEN(MP_TAC o SPEC `d2:real`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `w2:real^2` STRIP_ASSUME_TAC) THEN DISCH_THEN(MP_TAC o SPEC `d1:real`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `w1:real^2` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `(w1:real^2) IN interval[--vector [&1; &2],vector [&1; &2]] /\ (w2:real^2) IN interval[--vector [&1; &2],vector [&1; &2]]` STRIP_ASSUME_TAC THENL [CONJ_TAC THEN MATCH_MP_TAC(SET_RULE `!n j. j SUBSET i /\ (:real^N) DIFF j SUBSET n /\ ~(w IN n) ==> w IN i`) THEN EXISTS_TAC `connected_component K (n:real^2)` THEN EXISTS_TAC `interval[--vector [&1; &9 / &5]:real^2,vector [&1; &9 / &5]]` THEN ASM_REWRITE_TAC[SUBSET_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN ASM_MESON_TAC[CONNECTED_COMPONENT_EQ]; ALL_TAC] THEN SUBGOAL_THEN `?br. path br /\ path_image br SUBSET connected_component K x /\ pathstart br = w1 /\ pathfinish br = (w2:real^2)` STRIP_ASSUME_TAC THENL [MP_TAC(ISPEC `connected_component K (x:real^2)` path_connected) THEN ASM_SIMP_TAC[PATH_CONNECTED_EQ_CONNECTED; OPEN_CONNECTED_COMPONENT] THEN REWRITE_TAC[CONNECTED_CONNECTED_COMPONENT] THEN DISCH_THEN MATCH_MP_TAC THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `pathstart bp = n:real^2 /\ pathfinish bp = (--n:real^2)` STRIP_ASSUME_TAC THENL [EXPAND_TAC "bp" THEN REWRITE_TAC[PATHSTART_JOIN; PATHFINISH_JOIN] THEN REWRITE_TAC[PATHSTART_LINEPATH; PATHFINISH_LINEPATH]; ALL_TAC] THEN MP_TAC(ISPECL [`linepath(--basis 1:real^2,w1) ++ br ++ linepath(w2,basis 1)`; `reversepath bp:real^1->real^2`; `--vector[&1; &2]:real^2`; `vector[&1; &2]:real^2`] FASHODA) THEN ASM_SIMP_TAC[PATH_JOIN; PATH_LINEPATH; PATH_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATHSTART_LINEPATH; PATHFINISH_LINEPATH; PATHSTART_JOIN; PATHFINISH_JOIN; PATH_IMAGE_JOIN; PATH_IMAGE_REVERSEPATH] THEN SUBST1_TAC(SYM(ASSUME `&2 % basis 2:real^2 = n`)) THEN SIMP_TAC[BASIS_COMPONENT; VECTOR_2; VECTOR_NEG_COMPONENT; DIMINDEX_2; ARITH; VECTOR_MUL_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN REWRITE_TAC[NOT_IMP; UNION_SUBSET; GSYM CONJ_ASSOC] THEN REWRITE_TAC[PATH_IMAGE_LINEPATH; SEGMENT_CONVEX_HULL] THEN SIMP_TAC[SUBSET_HULL; CONVEX_INTERVAL] THEN ASM_REWRITE_TAC[SET_RULE `{a,b} SUBSET s <=> a IN s /\ b IN s`] THEN SIMP_TAC[IN_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_2; BASIS_COMPONENT; ARITH; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC SUBSET_TRANS THEN EXISTS_TAC `connected_component K (x:real^2)` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `!n j. j SUBSET i /\ (:real^N) DIFF j SUBSET n /\ c INTER n = {} ==> c SUBSET i`) THEN EXISTS_TAC `connected_component K (n:real^2)` THEN EXISTS_TAC `interval[--vector [&1; &9 / &5]:real^2,vector [&1; &9 / &5]]` THEN ASM_REWRITE_TAC[SUBSET_INTERVAL; DIMINDEX_2; FORALL_2; VECTOR_2; VECTOR_NEG_COMPONENT] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN ASM_REWRITE_TAC[CONNECTED_COMPONENT_NONOVERLAP]; EXPAND_TAC "bp" THEN REPEAT(MATCH_MP_TAC SUBSET_PATH_IMAGE_JOIN THEN CONJ_TAC) THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[PATH_IMAGE_LINEPATH; SEGMENT_CONVEX_HULL] THEN SIMP_TAC[SUBSET_HULL; CONVEX_INTERVAL] THEN ASM_REWRITE_TAC[SET_RULE `{a,b} SUBSET s <=> a IN s /\ b IN s`] THEN SUBST1_TAC(SYM(ASSUME `&2 % basis 2:real^2 = n`)) THEN SIMP_TAC[BASIS_COMPONENT; VECTOR_2; VECTOR_NEG_COMPONENT; DIMINDEX_2; ARITH; VECTOR_MUL_COMPONENT; IN_INTERVAL; FORALL_2] THEN CONV_TAC REAL_RAT_REDUCE_CONV THEN ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `w:real^2` (CONJUNCTS_THEN2 MP_TAC ASSUME_TAC)) THEN REWRITE_TAC[IN_UNION; DE_MORGAN_THM] THEN REPEAT CONJ_TAC THENL [UNDISCH_TAC `cball(--basis 1,d1) SUBSET (:real^2) DIFF path_image bp` THEN REWRITE_TAC[SUBSET; IN_DIFF; IN_UNIV] THEN DISCH_THEN(MP_TAC o SPEC `w:real^2`) THEN ASM_REWRITE_TAC[CONTRAPOS_THM] THEN SPEC_TAC(`w:real^2`,`u:real^2`) THEN REWRITE_TAC[GSYM SUBSET] THEN SIMP_TAC[SUBSET_HULL; CONVEX_CBALL] THEN REWRITE_TAC[SET_RULE `{a,b} SUBSET s <=> a IN s /\ b IN s`] THEN ASM_SIMP_TAC[CENTRE_IN_CBALL; REAL_LT_IMP_LE] THEN REWRITE_TAC[IN_CBALL] THEN ASM_MESON_TAC[REAL_LT_IMP_LE; DIST_SYM]; ALL_TAC; UNDISCH_TAC `cball(basis 1,d2) SUBSET (:real^2) DIFF path_image bp` THEN REWRITE_TAC[SUBSET; IN_DIFF; IN_UNIV] THEN DISCH_THEN(MP_TAC o SPEC `w:real^2`) THEN ASM_REWRITE_TAC[CONTRAPOS_THM] THEN SPEC_TAC(`w:real^2`,`u:real^2`) THEN REWRITE_TAC[GSYM SUBSET] THEN SIMP_TAC[SUBSET_HULL; CONVEX_CBALL] THEN REWRITE_TAC[SET_RULE `{a,b} SUBSET s <=> a IN s /\ b IN s`] THEN ASM_SIMP_TAC[CENTRE_IN_CBALL; REAL_LT_IMP_LE] THEN REWRITE_TAC[IN_CBALL] THEN ASM_MESON_TAC[REAL_LT_IMP_LE; DIST_SYM]] THEN FIRST_X_ASSUM(MP_TAC o C MATCH_MP (ASSUME `(w:real^2) IN path_image bp`) o GEN_REWRITE_RULE I [SUBSET]) THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `br SUBSET k ==> s INTER k = {} ==> w IN s ==> ~(w IN br)`)) THEN MATCH_MP_TAC(SET_RULE `xx SUBSET (UNIV DIFF du) /\ nn INTER xx = {} /\ yy INTER xx = {} ==> ((du DIFF ee) UNION nn UNION yy) INTER xx = {}`) THEN ASM_REWRITE_TAC[CONNECTED_COMPONENT_NONOVERLAP] THEN MATCH_MP_TAC SUBSET_TRANS THEN EXISTS_TAC `K:real^2->bool` THEN REWRITE_TAC[CONNECTED_COMPONENT_SUBSET] THEN EXPAND_TAC "K" THEN MATCH_MP_TAC(SET_RULE `s = t ==> s SUBSET t`) THEN AP_TERM_TAC THEN ASM_REWRITE_TAC[]);;
let JORDAN_DISCONNECTED = 
prove (`!c. simple_path c /\ pathfinish c = pathstart c ==> ~connected((:real^2) DIFF path_image c)`,
GEN_TAC THEN DISCH_TAC THEN REWRITE_TAC[connected] THEN FIRST_ASSUM(MP_TAC o MATCH_MP JORDAN_CURVE_THEOREM) THEN REPEAT(MATCH_MP_TAC MONO_EXISTS THEN GEN_TAC) THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]);;
let JORDAN_INSIDE_OUTSIDE = 
prove (`!c:real^1->real^2. simple_path c /\ pathfinish c = pathstart c ==> ~(inside(path_image c) = {}) /\ open(inside(path_image c)) /\ connected(inside(path_image c)) /\ ~(outside(path_image c) = {}) /\ open(outside(path_image c)) /\ connected(outside(path_image c)) /\ bounded(inside(path_image c)) /\ ~bounded(outside(path_image c)) /\ inside(path_image c) INTER outside(path_image c) = {} /\ inside(path_image c) UNION outside(path_image c) = (:real^2) DIFF path_image c /\ frontier(inside(path_image c)) = path_image c /\ frontier(outside(path_image c)) = path_image c`,
GEN_TAC THEN DISCH_TAC THEN FIRST_ASSUM(MP_TAC o MATCH_MP JORDAN_CURVE_THEOREM) THEN REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`ins:real^2->bool`; `out:real^2->bool`] THEN STRIP_TAC THEN SUBGOAL_THEN `inside(path_image c) :real^2->bool = ins /\ outside(path_image c):real^2->bool = out ` (fun th -> ASM_REWRITE_TAC[th]) THEN MATCH_MP_TAC INSIDE_OUTSIDE_UNIQUE THEN ASM_SIMP_TAC[JORDAN_DISCONNECTED]);;
(* ------------------------------------------------------------------------- *) (* Splitting the inside of a closed curve into two with a cut across it. *) (* The hardest part, that there is no third component inside the original *) (* curve, is taken from Whyburn's, "Topological Analysis" (1.4 on p31). *) (* ------------------------------------------------------------------------- *)
let SPLIT_INSIDE_SIMPLE_CLOSED_CURVE = 
prove (`!c1 c2 c a b:real^2. ~(a = b) /\ simple_path c1 /\ pathstart c1 = a /\ pathfinish c1 = b /\ simple_path c2 /\ pathstart c2 = a /\ pathfinish c2 = b /\ simple_path c /\ pathstart c = a /\ pathfinish c = b /\ path_image c1 INTER path_image c2 = {a,b} /\ path_image c1 INTER path_image c = {a,b} /\ path_image c2 INTER path_image c = {a,b} /\ ~(path_image c INTER inside(path_image c1 UNION path_image c2) = {}) ==> inside(path_image c1 UNION path_image c) INTER inside(path_image c2 UNION path_image c) = {} /\ inside(path_image c1 UNION path_image c) UNION inside(path_image c2 UNION path_image c) UNION (path_image c DIFF {a,b}) = inside(path_image c1 UNION path_image c2)`,
REPEAT GEN_TAC THEN STRIP_TAC THEN MAP_EVERY (MP_TAC o C ISPEC JORDAN_INSIDE_OUTSIDE) [`(c1 ++ reversepath c2):real^1->real^2`; `(c1 ++ reversepath c):real^1->real^2`; `(c2 ++ reversepath c):real^1->real^2`] THEN ASM_SIMP_TAC[PATHSTART_JOIN; PATHFINISH_JOIN; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; SIMPLE_PATH_JOIN_LOOP; SIMPLE_PATH_IMP_ARC; PATH_IMAGE_JOIN; SIMPLE_PATH_IMP_PATH; PATH_IMAGE_REVERSEPATH; SIMPLE_PATH_REVERSEPATH; ARC_REVERSEPATH; SUBSET_REFL] THEN REPLICATE_TAC 3 STRIP_TAC THEN SUBGOAL_THEN `path_image(c:real^1->real^2) INTER outside(path_image c1 UNION path_image c2) = {}` ASSUME_TAC THENL [MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN SUBGOAL_THEN `connected(path_image(c:real^1->real^2) DIFF {pathstart c,pathfinish c})` MP_TAC THENL [ASM_SIMP_TAC[CONNECTED_SIMPLE_PATH_ENDLESS]; ALL_TAC] THEN ASM_REWRITE_TAC[connected] THEN MAP_EVERY EXISTS_TAC [`inside(path_image c1 UNION path_image c2):real^2->bool`; `outside(path_image c1 UNION path_image c2):real^2->bool`] THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `outside(path_image c1 UNION path_image c2) SUBSET outside(path_image c1 UNION path_image (c:real^1->real^2)) /\ outside(path_image c1 UNION path_image c2) SUBSET outside(path_image c2 UNION path_image c)` STRIP_ASSUME_TAC THENL [CONJ_TAC THENL [ALL_TAC; GEN_REWRITE_TAC (LAND_CONV o ONCE_DEPTH_CONV) [UNION_COMM]] THEN MATCH_MP_TAC OUTSIDE_UNION_OUTSIDE_UNION THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[UNION_COMM] THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `path_image(c1:real^1->real^2) INTER inside(path_image c2 UNION path_image c) = {}` ASSUME_TAC THENL [MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN SUBGOAL_THEN `frontier(outside(path_image c1 UNION path_image c2)):real^2->bool = frontier(outside(path_image c2 UNION path_image c))` MP_TAC THENL [AP_TERM_TAC THEN MATCH_MP_TAC SUBSET_ANTISYM THEN ASM_REWRITE_TAC[] THEN GEN_REWRITE_TAC (RAND_CONV o RAND_CONV) [UNION_COMM] THEN MATCH_MP_TAC OUTSIDE_UNION_OUTSIDE_UNION THEN MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN SUBGOAL_THEN `connected(path_image(c1:real^1->real^2) DIFF {pathstart c1,pathfinish c1})` MP_TAC THENL [ASM_SIMP_TAC[CONNECTED_SIMPLE_PATH_ENDLESS]; ALL_TAC] THEN ASM_REWRITE_TAC[connected] THEN MAP_EVERY EXISTS_TAC [`inside(path_image c2 UNION path_image c):real^2->bool`; `outside(path_image c2 UNION path_image c):real^2->bool`] THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; MP_TAC(ISPEC `c:real^1->real^2` NONEMPTY_SIMPLE_PATH_ENDLESS) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `path_image(c2:real^1->real^2) INTER inside(path_image c1 UNION path_image c) = {}` ASSUME_TAC THENL [MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN SUBGOAL_THEN `frontier(outside(path_image c1 UNION path_image c2)):real^2->bool = frontier(outside(path_image c1 UNION path_image c))` MP_TAC THENL [AP_TERM_TAC THEN MATCH_MP_TAC SUBSET_ANTISYM THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC OUTSIDE_UNION_OUTSIDE_UNION THEN MATCH_MP_TAC(TAUT `(~p ==> F) ==> p`) THEN DISCH_TAC THEN SUBGOAL_THEN `connected(path_image(c2:real^1->real^2) DIFF {pathstart c2,pathfinish c2})` MP_TAC THENL [ASM_SIMP_TAC[CONNECTED_SIMPLE_PATH_ENDLESS]; ALL_TAC] THEN ASM_REWRITE_TAC[connected] THEN MAP_EVERY EXISTS_TAC [`inside(path_image c1 UNION path_image c):real^2->bool`; `outside(path_image c1 UNION path_image c):real^2->bool`] THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; MP_TAC(ISPEC `c:real^1->real^2` NONEMPTY_SIMPLE_PATH_ENDLESS) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `inside(path_image c1 UNION path_image (c:real^1->real^2)) SUBSET inside(path_image c1 UNION path_image c2) /\ inside(path_image c2 UNION path_image (c:real^1->real^2)) SUBSET inside(path_image c1 UNION path_image c2)` STRIP_ASSUME_TAC THENL [CONJ_TAC THEN REWRITE_TAC[INSIDE_OUTSIDE] THEN REWRITE_TAC[SET_RULE `UNIV DIFF t SUBSET UNIV DIFF s <=> s SUBSET t`] THENL [ALL_TAC; GEN_REWRITE_TAC (LAND_CONV o LAND_CONV) [UNION_COMM]] THEN MATCH_MP_TAC(SET_RULE `out1 SUBSET out2 /\ c2 DIFF (c1 UNION c) SUBSET out2 ==> (c1 UNION c2) UNION out1 SUBSET (c1 UNION c) UNION out2`) THEN ASM_REWRITE_TAC[] THEN REWRITE_TAC[OUTSIDE_INSIDE] THEN ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `inside(path_image c1 UNION path_image c :real^2->bool) SUBSET outside(path_image c2 UNION path_image c) /\ inside(path_image c2 UNION path_image c) SUBSET outside(path_image c1 UNION path_image c)` STRIP_ASSUME_TAC THENL [REWRITE_TAC[SUBSET] THEN CONJ_TAC THEN X_GEN_TAC `x:real^2` THEN DISCH_TAC THENL [SUBGOAL_THEN `?z:real^2. z IN path_image c1 /\ z IN outside(path_image c2 UNION path_image c)` (CHOOSE_THEN (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THENL [REWRITE_TAC[OUTSIDE_INSIDE; IN_DIFF; IN_UNION; IN_UNIV] THEN MP_TAC(ISPEC `c1:real^1->real^2` NONEMPTY_SIMPLE_PATH_ENDLESS) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(fun th -> ASSUME_TAC th THEN MP_TAC th) THEN REWRITE_TAC[OUTSIDE; IN_ELIM_THM; CONTRAPOS_THM] THEN MATCH_MP_TAC EQ_IMP THEN AP_TERM_TAC THEN MATCH_MP_TAC CONNECTED_COMPONENT_EQ THEN REWRITE_TAC[IN] THEN MP_TAC(ASSUME `open(outside(path_image c2 UNION path_image c):real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `e:real` STRIP_ASSUME_TAC) THEN MP_TAC(ASSUME `frontier(inside(path_image c1 UNION path_image c):real^2->bool) = path_image c1 UNION path_image c`) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN REWRITE_TAC[frontier] THEN ASM_SIMP_TAC[IN_UNION; IN_DIFF; CLOSURE_APPROACHABLE; IN_ELIM_THM] THEN DISCH_THEN(MP_TAC o SPEC `e:real` o CONJUNCT1) THEN ASM_REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `w:real^2` THEN STRIP_TAC THEN MATCH_MP_TAC CONNECTED_COMPONENT_TRANS THEN EXISTS_TAC `w:real^2` THEN REWRITE_TAC[connected_component] THEN CONJ_TAC THENL [EXISTS_TAC `outside(path_image c2 UNION path_image c:real^2->bool)` THEN ASM_REWRITE_TAC[SET_RULE `s SUBSET UNIV DIFF t <=> s INTER t = {}`; OUTSIDE_NO_OVERLAP] THEN FIRST_X_ASSUM(MATCH_MP_TAC o GEN_REWRITE_RULE I [SUBSET]) THEN ASM_REWRITE_TAC[ONCE_REWRITE_RULE[DIST_SYM] IN_BALL]; EXISTS_TAC `inside(path_image c1 UNION path_image c:real^2->bool)` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `inside(c1 UNION c) INTER (c1 UNION c) = {} /\ c2 INTER inside(c1 UNION c) = {} ==> inside(c1 UNION c) SUBSET UNIV DIFF (c2 UNION c)`) THEN ASM_REWRITE_TAC[INSIDE_NO_OVERLAP]]; SUBGOAL_THEN `?z:real^2. z IN path_image c2 /\ z IN outside(path_image c1 UNION path_image c)` (CHOOSE_THEN (CONJUNCTS_THEN2 ASSUME_TAC MP_TAC)) THENL [REWRITE_TAC[OUTSIDE_INSIDE; IN_DIFF; IN_UNION; IN_UNIV] THEN MP_TAC(ISPEC `c2:real^1->real^2` NONEMPTY_SIMPLE_PATH_ENDLESS) THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(fun th -> ASSUME_TAC th THEN MP_TAC th) THEN REWRITE_TAC[OUTSIDE; IN_ELIM_THM; CONTRAPOS_THM] THEN MATCH_MP_TAC EQ_IMP THEN AP_TERM_TAC THEN MATCH_MP_TAC CONNECTED_COMPONENT_EQ THEN REWRITE_TAC[IN] THEN MP_TAC(ASSUME `open(outside(path_image c1 UNION path_image c):real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(X_CHOOSE_THEN `e:real` STRIP_ASSUME_TAC) THEN MP_TAC(ASSUME `frontier(inside(path_image c2 UNION path_image c):real^2->bool) = path_image c2 UNION path_image c`) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN REWRITE_TAC[frontier] THEN ASM_SIMP_TAC[IN_UNION; IN_DIFF; CLOSURE_APPROACHABLE; IN_ELIM_THM] THEN DISCH_THEN(MP_TAC o SPEC `e:real` o CONJUNCT1) THEN ASM_REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN X_GEN_TAC `w:real^2` THEN STRIP_TAC THEN MATCH_MP_TAC CONNECTED_COMPONENT_TRANS THEN EXISTS_TAC `w:real^2` THEN REWRITE_TAC[connected_component] THEN CONJ_TAC THENL [EXISTS_TAC `outside(path_image c1 UNION path_image c:real^2->bool)` THEN ASM_REWRITE_TAC[SET_RULE `s SUBSET UNIV DIFF t <=> s INTER t = {}`; OUTSIDE_NO_OVERLAP] THEN FIRST_X_ASSUM(MATCH_MP_TAC o GEN_REWRITE_RULE I [SUBSET]) THEN ASM_REWRITE_TAC[ONCE_REWRITE_RULE[DIST_SYM] IN_BALL]; EXISTS_TAC `inside(path_image c2 UNION path_image c:real^2->bool)` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `inside(c2 UNION c) INTER (c2 UNION c) = {} /\ c1 INTER inside(c2 UNION c) = {} ==> inside(c2 UNION c) SUBSET UNIV DIFF (c1 UNION c)`) THEN ASM_REWRITE_TAC[INSIDE_NO_OVERLAP]]]; ALL_TAC] THEN CONJ_TAC THENL [MATCH_MP_TAC(SET_RULE `!u. s SUBSET u /\ t INTER u = {} ==> s INTER t = {}`) THEN EXISTS_TAC `outside(path_image c2 UNION path_image c):real^2->bool` THEN ASM_REWRITE_TAC[INSIDE_INTER_OUTSIDE]; ALL_TAC] THEN SUBGOAL_THEN `!x:real^2. ~(x IN path_image c) /\ x IN inside(path_image c1 UNION path_image c2) /\ ~(x IN inside(path_image c1 UNION path_image c)) /\ ~(x IN inside(path_image c2 UNION path_image c)) ==> F` ASSUME_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN X_GEN_TAC `z:real^2` THEN STRIP_TAC THEN SUBGOAL_THEN `~((z:real^2) IN (path_image c1 UNION path_image c2))` MP_TAC THENL [ASM_MESON_TAC[INSIDE_NO_OVERLAP; NOT_IN_EMPTY; IN_INTER]; PURE_REWRITE_TAC[IN_UNION; DE_MORGAN_THM] THEN STRIP_TAC] THEN MAP_EVERY (MP_TAC o C ISPEC INSIDE_UNION_OUTSIDE) [`path_image c1 UNION path_image c:real^2->bool`; `path_image c2 UNION path_image c:real^2->bool`] THEN PURE_REWRITE_TAC[IMP_IMP] THEN REWRITE_TAC[EXTENSION; IN_UNION] THEN REWRITE_TAC[AND_FORALL_THM] THEN DISCH_THEN(MP_TAC o SPEC `z:real^2`) THEN ASM_REWRITE_TAC[IN_UNIV; IN_DIFF; IN_UNION] THEN STRIP_TAC THEN MP_TAC(ASSUME `~(outside(path_image c1 UNION path_image c2:real^2->bool) = {})`) THEN GEN_REWRITE_TAC LAND_CONV [GSYM MEMBER_NOT_EMPTY] THEN DISCH_THEN(X_CHOOSE_TAC `w:real^2`) THEN SUBGOAL_THEN `(w:real^2) IN outside(path_image c1 UNION path_image c) /\ w IN outside(path_image c2 UNION path_image c)` STRIP_ASSUME_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~((w:real^2) IN (path_image c1 UNION path_image c2)) /\ ~(w IN path_image c1 UNION path_image c)` MP_TAC THENL [ASM_MESON_TAC[OUTSIDE_NO_OVERLAP; NOT_IN_EMPTY; IN_INTER]; PURE_REWRITE_TAC[IN_UNION; CONJ_ACI; DE_MORGAN_THM] THEN STRIP_TAC] THEN MAP_EVERY (MP_TAC o C ISPEC INSIDE_INTER_OUTSIDE) [`path_image c1 UNION path_image c2:real^2->bool`; `path_image c1 UNION path_image c:real^2->bool`; `path_image c2 UNION path_image c:real^2->bool`] THEN PURE_REWRITE_TAC[IMP_IMP] THEN REWRITE_TAC[EXTENSION; IN_INTER; NOT_IN_EMPTY] THEN REWRITE_TAC[AND_FORALL_THM] THEN DISCH_THEN(MP_TAC o SPEC `w:real^2`) THEN ASM_REWRITE_TAC[] THEN STRIP_TAC THEN MP_TAC(ISPEC `outside(path_image c2 UNION path_image c):real^2->bool` CONNECTED_OPEN_ARC_CONNECTED) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(MP_TAC o SPECL [`z:real^2`; `w:real^2`]) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_THEN MP_TAC) THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `g1:real^1->real^2` STRIP_ASSUME_TAC) THEN MP_TAC(ISPEC `outside(path_image c1 UNION path_image c):real^2->bool` CONNECTED_OPEN_ARC_CONNECTED) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(MP_TAC o SPECL [`z:real^2`; `w:real^2`]) THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_THEN MP_TAC) THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `g2:real^1->real^2` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `?h. arc h /\ pathstart h = (z:real^2) /\ path_image h DELETE pathfinish h SUBSET inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c) /\ pathfinish h IN path_image c1 /\ ~(pathfinish h IN path_image c2)` (X_CHOOSE_THEN `gzx:real^1->real^2` STRIP_ASSUME_TAC) THENL [MP_TAC(ISPECL [`g1:real^1->real^2`; `inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c):real^2->bool`] SUBPATH_TO_FRONTIER) THEN ASM_SIMP_TAC[ARC_IMP_PATH; IN_INTER; INTERIOR_INTER; INTERIOR_OPEN] THEN DISCH_THEN(X_CHOOSE_THEN `u:real^1` STRIP_ASSUME_TAC) THEN ABBREV_TAC `h:real^1->real^2 = subpath (vec 0) u g1` THEN EXISTS_TAC `h:real^1->real^2` THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHSTART_SUBPATH; PATHFINISH_SUBPATH] THEN MATCH_MP_TAC(TAUT `b /\ (c ==> a) /\ c ==> a /\ b /\ c`) THEN CONJ_TAC THENL [ASM_MESON_TAC[pathstart]; ALL_TAC] THEN CONJ_TAC THENL [DISCH_TAC THEN MATCH_MP_TAC ARC_SIMPLE_PATH_SUBPATH THEN ASM_SIMP_TAC[ENDS_IN_UNIT_INTERVAL; ARC_IMP_SIMPLE_PATH] THEN DISCH_THEN(SUBST_ALL_TAC o SYM) THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` INSIDE_NO_OVERLAP) THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN EXISTS_TAC `pathstart(g1:real^1->real^2)` THEN ASM_REWRITE_TAC[IN_INTER] THEN REWRITE_TAC[IN_UNION] THEN DISJ1_TAC THEN ASM_MESON_TAC[pathstart]; ALL_TAC] THEN UNDISCH_TAC `(pathfinish h:real^2) IN frontier (inside (path_image c1 UNION path_image c2) INTER outside (path_image c1 UNION path_image c) INTER outside (path_image c2 UNION path_image c))` THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHFINISH_SUBPATH] THEN DISCH_THEN(MP_TAC o MATCH_MP (REWRITE_RULE[SUBSET] FRONTIER_INTER_SUBSET)) THEN ASM_REWRITE_TAC[] THEN MP_TAC(ASSUME `(path_image g1:real^2->bool) SUBSET outside(path_image c2 UNION path_image c)`) THEN DISCH_THEN(MP_TAC o SPEC `(g1:real^1->real^2) u` o REWRITE_RULE[SUBSET]) THEN GEN_REWRITE_TAC (LAND_CONV o LAND_CONV o ONCE_DEPTH_CONV) [path_image] THEN REWRITE_TAC[IN_IMAGE] THEN ANTS_TAC THENL [ASM_MESON_TAC[]; DISCH_TAC] THEN MP_TAC(ISPEC `path_image c2 UNION path_image c:real^2->bool` OUTSIDE_NO_OVERLAP) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `(g1:real^1->real^2) u`) THEN ASM_REWRITE_TAC[IN_INTER; IN_UNION; NOT_IN_EMPTY; DE_MORGAN_THM] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_THEN2 ACCEPT_TAC MP_TAC) THEN DISCH_THEN(MP_TAC o MATCH_MP (REWRITE_RULE[SUBSET] FRONTIER_INTER_SUBSET)) THEN ASM_REWRITE_TAC[] THEN ASM_REWRITE_TAC[IN_UNION]; ABBREV_TAC `x:real^2 = pathfinish gzx`] THEN SUBGOAL_THEN `?h. arc h /\ pathstart h = (w:real^2) /\ path_image h DELETE pathfinish h SUBSET outside(path_image c1 UNION path_image c2) /\ pathfinish h IN path_image c1 /\ ~(pathfinish h IN path_image c2)` (X_CHOOSE_THEN `gwx:real^1->real^2` STRIP_ASSUME_TAC) THENL [MP_TAC(ISPECL [`reversepath g1:real^1->real^2`; `outside(path_image c1 UNION path_image c2):real^2->bool`] SUBPATH_TO_FRONTIER) THEN ASM_SIMP_TAC[ARC_IMP_PATH; IN_INTER; INTERIOR_INTER; INTERIOR_OPEN; PATH_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH] THEN ANTS_TAC THENL [ASM_MESON_TAC[INSIDE_INTER_OUTSIDE; IN_INTER; NOT_IN_EMPTY]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `u:real^1` STRIP_ASSUME_TAC) THEN ABBREV_TAC `h:real^1->real^2 = subpath (vec 0) u (reversepath g1)` THEN EXISTS_TAC `h:real^1->real^2` THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHSTART_SUBPATH; PATHFINISH_SUBPATH] THEN MATCH_MP_TAC(TAUT `b /\ (c ==> a) /\ c ==> a /\ b /\ c`) THEN CONJ_TAC THENL [ASM_MESON_TAC[pathstart; PATHSTART_REVERSEPATH]; ALL_TAC] THEN CONJ_TAC THENL [DISCH_TAC THEN MATCH_MP_TAC ARC_SIMPLE_PATH_SUBPATH THEN ASM_SIMP_TAC[ENDS_IN_UNIT_INTERVAL; ARC_IMP_SIMPLE_PATH; SIMPLE_PATH_REVERSEPATH] THEN DISCH_THEN(SUBST_ALL_TAC o SYM) THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` OUTSIDE_NO_OVERLAP) THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN EXISTS_TAC `pathstart(reversepath g1:real^1->real^2)` THEN ASM_REWRITE_TAC[IN_INTER] THEN CONJ_TAC THENL [ASM_REWRITE_TAC[PATHSTART_REVERSEPATH]; REWRITE_TAC[IN_UNION] THEN DISJ1_TAC THEN ASM_MESON_TAC[pathstart]]; ALL_TAC] THEN UNDISCH_TAC `(pathfinish h:real^2) IN path_image c1 UNION path_image c2` THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHFINISH_SUBPATH] THEN MP_TAC(ASSUME `(path_image g1:real^2->bool) SUBSET outside(path_image c2 UNION path_image c)`) THEN DISCH_THEN(MP_TAC o SPEC `reversepath (g1:real^1->real^2) u` o REWRITE_RULE[SUBSET]) THEN ANTS_TAC THENL [ONCE_REWRITE_TAC[GSYM PATH_IMAGE_REVERSEPATH] THEN REWRITE_TAC[path_image; IN_IMAGE] THEN ASM_MESON_TAC[]; DISCH_TAC] THEN MP_TAC(ISPEC `path_image c2 UNION path_image c:real^2->bool` OUTSIDE_NO_OVERLAP) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `reversepath (g1:real^1->real^2) u`) THEN ASM_REWRITE_TAC[IN_INTER; IN_UNION; NOT_IN_EMPTY; DE_MORGAN_THM] THEN CONV_TAC TAUT; ABBREV_TAC `x':real^2 = pathfinish gwx`] THEN SUBGOAL_THEN `?h. arc h /\ pathstart h = (z:real^2) /\ path_image h DELETE pathfinish h SUBSET inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c) /\ pathfinish h IN path_image c2 /\ ~(pathfinish h IN path_image c1)` (X_CHOOSE_THEN `gzy:real^1->real^2` STRIP_ASSUME_TAC) THENL [MP_TAC(ISPECL [`g2:real^1->real^2`; `inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c):real^2->bool`] SUBPATH_TO_FRONTIER) THEN ASM_SIMP_TAC[ARC_IMP_PATH; IN_INTER; INTERIOR_INTER; INTERIOR_OPEN] THEN DISCH_THEN(X_CHOOSE_THEN `u:real^1` STRIP_ASSUME_TAC) THEN ABBREV_TAC `h:real^1->real^2 = subpath (vec 0) u g2` THEN EXISTS_TAC `h:real^1->real^2` THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHSTART_SUBPATH; PATHFINISH_SUBPATH] THEN MATCH_MP_TAC(TAUT `b /\ (c ==> a) /\ c ==> a /\ b /\ c`) THEN CONJ_TAC THENL [ASM_MESON_TAC[pathstart]; ALL_TAC] THEN CONJ_TAC THENL [DISCH_TAC THEN MATCH_MP_TAC ARC_SIMPLE_PATH_SUBPATH THEN ASM_SIMP_TAC[ENDS_IN_UNIT_INTERVAL; ARC_IMP_SIMPLE_PATH] THEN DISCH_THEN(SUBST_ALL_TAC o SYM) THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` INSIDE_NO_OVERLAP) THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN EXISTS_TAC `pathstart(g2:real^1->real^2)` THEN ASM_REWRITE_TAC[IN_INTER] THEN REWRITE_TAC[IN_UNION] THEN DISJ1_TAC THEN ASM_MESON_TAC[pathstart]; ALL_TAC] THEN UNDISCH_TAC `(pathfinish h:real^2) IN frontier (inside (path_image c1 UNION path_image c2) INTER outside (path_image c1 UNION path_image c) INTER outside (path_image c2 UNION path_image c))` THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHFINISH_SUBPATH] THEN DISCH_THEN(MP_TAC o MATCH_MP (REWRITE_RULE[SUBSET] FRONTIER_INTER_SUBSET)) THEN ASM_REWRITE_TAC[] THEN MP_TAC(ASSUME `(path_image g2:real^2->bool) SUBSET outside(path_image c1 UNION path_image c)`) THEN DISCH_THEN(MP_TAC o SPEC `(g2:real^1->real^2) u` o REWRITE_RULE[SUBSET]) THEN GEN_REWRITE_TAC (LAND_CONV o LAND_CONV o ONCE_DEPTH_CONV) [path_image] THEN REWRITE_TAC[IN_IMAGE] THEN ANTS_TAC THENL [ASM_MESON_TAC[]; DISCH_TAC] THEN MP_TAC(ISPEC `path_image c1 UNION path_image c:real^2->bool` OUTSIDE_NO_OVERLAP) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `(g2:real^1->real^2) u`) THEN ASM_REWRITE_TAC[IN_INTER; IN_UNION; NOT_IN_EMPTY; DE_MORGAN_THM] THEN STRIP_TAC THEN ASM_REWRITE_TAC[] THEN DISCH_THEN(DISJ_CASES_THEN2 ACCEPT_TAC MP_TAC) THEN DISCH_THEN(MP_TAC o MATCH_MP (REWRITE_RULE[SUBSET] FRONTIER_INTER_SUBSET)) THEN ASM_REWRITE_TAC[] THEN ASM_REWRITE_TAC[IN_UNION]; ABBREV_TAC `y:real^2 = pathfinish gzy`] THEN SUBGOAL_THEN `?h. arc h /\ pathstart h = (w:real^2) /\ path_image h DELETE pathfinish h SUBSET outside(path_image c1 UNION path_image c2) /\ pathfinish h IN path_image c2 /\ ~(pathfinish h IN path_image c1)` (X_CHOOSE_THEN `gwy:real^1->real^2` STRIP_ASSUME_TAC) THENL [MP_TAC(ISPECL [`reversepath g2:real^1->real^2`; `outside(path_image c1 UNION path_image c2):real^2->bool`] SUBPATH_TO_FRONTIER) THEN ASM_SIMP_TAC[ARC_IMP_PATH; IN_INTER; INTERIOR_INTER; INTERIOR_OPEN; PATH_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH] THEN ANTS_TAC THENL [ASM_MESON_TAC[INSIDE_INTER_OUTSIDE; IN_INTER; NOT_IN_EMPTY]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `u:real^1` STRIP_ASSUME_TAC) THEN ABBREV_TAC `h:real^1->real^2 = subpath (vec 0) u (reversepath g2)` THEN EXISTS_TAC `h:real^1->real^2` THEN ASM_REWRITE_TAC[] THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHSTART_SUBPATH; PATHFINISH_SUBPATH] THEN MATCH_MP_TAC(TAUT `b /\ (c ==> a) /\ c ==> a /\ b /\ c`) THEN CONJ_TAC THENL [ASM_MESON_TAC[pathstart; PATHSTART_REVERSEPATH]; ALL_TAC] THEN CONJ_TAC THENL [DISCH_TAC THEN MATCH_MP_TAC ARC_SIMPLE_PATH_SUBPATH THEN ASM_SIMP_TAC[ENDS_IN_UNIT_INTERVAL; ARC_IMP_SIMPLE_PATH; SIMPLE_PATH_REVERSEPATH] THEN DISCH_THEN(SUBST_ALL_TAC o SYM) THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` OUTSIDE_NO_OVERLAP) THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY] THEN EXISTS_TAC `pathstart(reversepath g2:real^1->real^2)` THEN ASM_REWRITE_TAC[IN_INTER] THEN CONJ_TAC THENL [ASM_REWRITE_TAC[PATHSTART_REVERSEPATH]; REWRITE_TAC[IN_UNION] THEN DISJ2_TAC THEN ASM_MESON_TAC[pathstart]]; ALL_TAC] THEN UNDISCH_TAC `(pathfinish h:real^2) IN path_image c1 UNION path_image c2` THEN EXPAND_TAC "h" THEN REWRITE_TAC[PATHFINISH_SUBPATH] THEN MP_TAC(ASSUME `(path_image g2:real^2->bool) SUBSET outside(path_image c1 UNION path_image c)`) THEN DISCH_THEN(MP_TAC o SPEC `reversepath (g2:real^1->real^2) u` o REWRITE_RULE[SUBSET]) THEN ANTS_TAC THENL [ONCE_REWRITE_TAC[GSYM PATH_IMAGE_REVERSEPATH] THEN REWRITE_TAC[path_image; IN_IMAGE] THEN ASM_MESON_TAC[]; DISCH_TAC] THEN MP_TAC(ISPEC `path_image c1 UNION path_image c:real^2->bool` OUTSIDE_NO_OVERLAP) THEN GEN_REWRITE_TAC LAND_CONV [EXTENSION] THEN DISCH_THEN(MP_TAC o SPEC `reversepath (g2:real^1->real^2) u`) THEN ASM_REWRITE_TAC[IN_INTER; IN_UNION; NOT_IN_EMPTY; DE_MORGAN_THM] THEN CONV_TAC TAUT; ABBREV_TAC `y':real^2 = pathfinish gwy`] THEN MP_TAC(ISPECL [`reversepath gzx:real^1->real^2`; `gzy:real^1->real^2`] ARC_CONNECTED_TRANS) THEN ASM_SIMP_TAC[ARC_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATH_IMAGE_REVERSEPATH; NOT_IMP] THEN CONJ_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `xy:real^1->real^2` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `(path_image xy DIFF {x:real^2,y}) SUBSET inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c)` ASSUME_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~(((path_image xy):real^2->bool) INTER inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c) = {})` ASSUME_TAC THENL [MATCH_MP_TAC(SET_RULE `!x y. ~(s DIFF {x,y} = {}) /\ s DIFF {x,y} SUBSET t ==> ~(s INTER t = {})`) THEN MAP_EVERY EXISTS_TAC [`x:real^2`; `y:real^2`] THEN ASM_REWRITE_TAC[] THEN ASM_MESON_TAC[NONEMPTY_SIMPLE_PATH_ENDLESS; ARC_IMP_SIMPLE_PATH]; ALL_TAC] THEN MP_TAC(ISPECL [`reversepath gwy:real^1->real^2`; `gwx:real^1->real^2`] ARC_CONNECTED_TRANS) THEN ASM_SIMP_TAC[ARC_REVERSEPATH; PATHSTART_REVERSEPATH; PATHFINISH_REVERSEPATH; PATH_IMAGE_REVERSEPATH; NOT_IMP] THEN CONJ_TAC THENL [ASM_MESON_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `y'x':real^1->real^2` STRIP_ASSUME_TAC) THEN SUBGOAL_THEN `(path_image y'x' DIFF {x':real^2,y'}) SUBSET outside(path_image c1 UNION path_image c2)` ASSUME_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `?xy'. arc xy' /\ pathstart xy' = x /\ pathfinish xy' = (y':real^2) /\ (path_image xy' DELETE x) SUBSET (inside (path_image c1 UNION path_image c2) INTER outside (path_image c1 UNION path_image c) INTER outside (path_image c2 UNION path_image c)) UNION (path_image c2 DIFF {a,b}) /\ ~(path_image xy' INTER inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c) = {})` STRIP_ASSUME_TAC THENL [ASM_CASES_TAC `y':real^2 = y` THENL [UNDISCH_THEN `y':real^2 = y` SUBST_ALL_TAC THEN EXISTS_TAC `xy:real^1->real^2` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `!y. p DIFF {x,y} SUBSET i /\ y IN j ==> p DELETE x SUBSET (i UNION j)`) THEN EXISTS_TAC `y:real^2` THEN ASM_REWRITE_TAC[IN_DIFF] THEN ASM SET_TAC[]; ALL_TAC] THEN MP_TAC(ISPECL [`c2:real^1->real^2`; `y:real^2`; `y':real^2`] EXISTS_SUBARC_OF_ARC_NOENDS) THEN ASM_SIMP_TAC[SIMPLE_PATH_IMP_ARC; NOT_IMP] THEN ANTS_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `yy':real^1->real^2` STRIP_ASSUME_TAC) THEN EXISTS_TAC `xy ++ yy':real^1->real^2` THEN ASM_REWRITE_TAC[PATHSTART_JOIN; PATHFINISH_JOIN] THEN CONJ_TAC THENL [MATCH_MP_TAC ARC_JOIN THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; ASM_SIMP_TAC[PATH_IMAGE_JOIN; ARC_IMP_PATH] THEN ASM_SIMP_TAC[SET_RULE `~(a INTER c = {}) ==> ~((a UNION b) INTER c = {})`] THEN MATCH_MP_TAC(SET_RULE `s SUBSET (y INSERT t) ==> s DELETE y SUBSET t`) THEN ASM SET_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `~((path_image y'x':real^2->bool) INTER outside(path_image c1 UNION path_image c2) = {})` ASSUME_TAC THENL [MP_TAC(ISPEC `y'x':real^1->real^2` NONEMPTY_SIMPLE_PATH_ENDLESS) THEN ASM_SIMP_TAC[ARC_IMP_SIMPLE_PATH] THEN ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `?y'x. arc y'x /\ pathstart y'x = y' /\ pathfinish y'x = (x:real^2) /\ ~(path_image y'x INTER outside(path_image c1 UNION path_image c2) = {}) /\ (path_image y'x DELETE y') SUBSET outside(path_image c1 UNION path_image c2) UNION (path_image c1 DIFF {a,b})` STRIP_ASSUME_TAC THENL [ASM_CASES_TAC `x':real^2 = x` THENL [UNDISCH_THEN `x':real^2 = x` SUBST_ALL_TAC THEN EXISTS_TAC `y'x':real^1->real^2` THEN ASM_REWRITE_TAC[] THEN MATCH_MP_TAC(SET_RULE `!y. p DIFF {x,y} SUBSET i /\ y IN j ==> p DELETE x SUBSET (i UNION j)`) THEN EXISTS_TAC `x:real^2` THEN ASM_REWRITE_TAC[IN_DIFF] THEN ASM SET_TAC[]; ALL_TAC] THEN MP_TAC(ISPECL [`c1:real^1->real^2`; `x':real^2`; `x:real^2`] EXISTS_SUBARC_OF_ARC_NOENDS) THEN ASM_SIMP_TAC[SIMPLE_PATH_IMP_ARC; NOT_IMP] THEN ANTS_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN DISCH_THEN(X_CHOOSE_THEN `x'x:real^1->real^2` STRIP_ASSUME_TAC) THEN EXISTS_TAC `y'x' ++ x'x:real^1->real^2` THEN ASM_REWRITE_TAC[PATHSTART_JOIN; PATHFINISH_JOIN] THEN ASM_SIMP_TAC[PATH_IMAGE_JOIN; ARC_IMP_PATH] THEN REPEAT CONJ_TAC THENL [MATCH_MP_TAC ARC_JOIN THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; ASM SET_TAC[]; ASM SET_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `?j. simple_path j /\ pathstart j = x /\ pathfinish j = (x:real^2) /\ ~(path_image j INTER inside(path_image c1 UNION path_image c2) = {}) /\ ~(path_image j INTER outside(path_image c1 UNION path_image c2) = {}) /\ path_image j SUBSET (inside(path_image c1 UNION path_image c2) INTER outside(path_image c1 UNION path_image c) INTER outside(path_image c2 UNION path_image c)) UNION outside(path_image c1 UNION path_image c2) UNION ((path_image c1 UNION path_image c2) DIFF {a,b})` STRIP_ASSUME_TAC THENL [EXISTS_TAC `xy' ++ y'x:real^1->real^2` THEN ASM_REWRITE_TAC[PATHSTART_JOIN; PATHFINISH_JOIN] THEN ASM_SIMP_TAC[PATH_IMAGE_JOIN; ARC_IMP_PATH] THEN CONJ_TAC THENL [ALL_TAC; ASM SET_TAC[]] THEN MATCH_MP_TAC SIMPLE_PATH_JOIN_LOOP THEN ASM_REWRITE_TAC[] THEN FIRST_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `xy DELETE x SUBSET s ==> !t. yx DELETE y SUBSET t /\ s INTER t = {} ==> xy INTER yx SUBSET {x,y}`)) THEN EXISTS_TAC `outside(path_image c1 UNION path_image c2) UNION path_image c1 DIFF {a:real^2, b}` THEN ASM_REWRITE_TAC[] THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` INSIDE_NO_OVERLAP) THEN MP_TAC(ISPEC `path_image c1 UNION path_image c2:real^2->bool` INSIDE_INTER_OUTSIDE) THEN ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~((a:real^2) IN path_image j)` ASSUME_TAC THENL [FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `s SUBSET t ==> ~(a IN t) ==> ~(a IN s)`)) THEN MATCH_MP_TAC(SET_RULE `~(a IN s) /\ ~(a IN t) ==> ~(a IN ((s INTER s' INTER s'') UNION t UNION (u DIFF {a,b})))`) THEN MAP_EVERY (MP_TAC o ISPEC `path_image c1 UNION path_image c2:real^2->bool`) [OUTSIDE_NO_OVERLAP; INSIDE_NO_OVERLAP] THEN REWRITE_TAC[IMP_IMP] THEN MATCH_MP_TAC MONO_AND THEN ASM SET_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `?u v. ~(u = {}) /\ open u /\ connected u /\ ~(v = {}) /\ open v /\ connected v /\ u INTER v = {} /\ u UNION v = (:real^2) DIFF path_image j /\ frontier u = path_image j /\ frontier v = path_image j /\ a IN u` STRIP_ASSUME_TAC THENL [MP_TAC(ISPEC `j:real^1->real^2` JORDAN_CURVE_THEOREM) THEN ASM_REWRITE_TAC[LEFT_IMP_EXISTS_THM] THEN MAP_EVERY X_GEN_TAC [`u:real^2->bool`; `v:real^2->bool`] THEN STRIP_TAC THEN SUBGOAL_THEN `(a:real^2) IN u \/ (a:real^2) IN v` MP_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN STRIP_TAC THENL [MAP_EVERY EXISTS_TAC [`u:real^2->bool`; `v:real^2->bool`]; MAP_EVERY EXISTS_TAC [`v:real^2->bool`; `u:real^2->bool`]] THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[INTER_COMM; UNION_COMM] THEN ASM_REWRITE_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~(v INTER inside(path_image c1 UNION path_image c2) = {}) /\ ~((v:real^2->bool) INTER outside(path_image c1 UNION path_image c2) = {})` STRIP_ASSUME_TAC THENL [CONJ_TAC THENL [UNDISCH_TAC `~((path_image j:real^2->bool) INTER inside(path_image c1 UNION path_image c2) = {})` THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN SUBST1_TAC(SYM(ASSUME `frontier v:real^2->bool = path_image j`)) THEN DISCH_THEN(X_CHOOSE_THEN `p:real^2` MP_TAC) THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC ASSUME_TAC) THEN REWRITE_TAC[frontier; IN_DIFF; CLOSURE_APPROACHABLE] THEN MP_TAC(GEN_REWRITE_RULE I [OPEN_CONTAINS_BALL] (ASSUME `open(inside(path_image c1 UNION path_image c2):real^2->bool)`)) THEN DISCH_THEN(MP_TAC o SPEC `p:real^2`) THEN ONCE_REWRITE_TAC[DIST_SYM] THEN ASM_REWRITE_TAC[SUBSET; IN_BALL] THEN ASM_MESON_TAC[]; UNDISCH_TAC `~((path_image j:real^2->bool) INTER outside(path_image c1 UNION path_image c2) = {})` THEN REWRITE_TAC[GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN SUBST1_TAC(SYM(ASSUME `frontier v:real^2->bool = path_image j`)) THEN DISCH_THEN(X_CHOOSE_THEN `p:real^2` MP_TAC) THEN DISCH_THEN(CONJUNCTS_THEN2 MP_TAC ASSUME_TAC) THEN REWRITE_TAC[frontier; IN_DIFF; CLOSURE_APPROACHABLE] THEN MP_TAC(GEN_REWRITE_RULE I [OPEN_CONTAINS_BALL] (ASSUME `open(outside(path_image c1 UNION path_image c2):real^2->bool)`)) THEN DISCH_THEN(MP_TAC o SPEC `p:real^2`) THEN ONCE_REWRITE_TAC[DIST_SYM] THEN ASM_REWRITE_TAC[SUBSET; IN_BALL] THEN ASM_MESON_TAC[]]; ALL_TAC] THEN SUBGOAL_THEN `~((v:real^2->bool) INTER (path_image c1 UNION path_image c2) = {})` MP_TAC THENL [MP_TAC(ASSUME `connected(v:real^2->bool)`) THEN REWRITE_TAC[connected; CONTRAPOS_THM] THEN DISCH_TAC THEN MAP_EVERY EXISTS_TAC [`inside(path_image c1 UNION path_image c2):real^2->bool`; `outside(path_image c1 UNION path_image c2):real^2->bool`] THEN ASM_REWRITE_TAC[] THEN ASM SET_TAC[]; GEN_REWRITE_TAC LAND_CONV [GSYM MEMBER_NOT_EMPTY]] THEN REWRITE_TAC[IN_INTER; IN_UNION] THEN DISCH_THEN(X_CHOOSE_THEN `p:real^2` (CONJUNCTS_THEN ASSUME_TAC)) THEN SUBGOAL_THEN `~(p:real^2 = a)` ASSUME_TAC THENL [ASM SET_TAC[]; ALL_TAC] THEN FIRST_X_ASSUM DISJ_CASES_TAC THENL [SUBGOAL_THEN `~((u:real^2->bool) INTER inside(path_image c1 UNION path_image c) = {})` ASSUME_TAC THENL [MP_TAC(ASSUME `open(u:real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL; SUBSET; IN_BALL] THEN DISCH_THEN(MP_TAC o SPEC `a:real^2`) THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[DIST_SYM] THEN SUBGOAL_THEN `(a:real^2) IN frontier(inside(path_image c1 UNION path_image c))` MP_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[frontier; IN_DIFF]] THEN REWRITE_TAC[CLOSURE_APPROACHABLE; GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~((v:real^2->bool) INTER inside(path_image c1 UNION path_image c) = {})` ASSUME_TAC THENL [MP_TAC(ASSUME `open(v:real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL; SUBSET; IN_BALL] THEN DISCH_THEN(MP_TAC o SPEC `p:real^2`) THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[DIST_SYM] THEN SUBGOAL_THEN `(p:real^2) IN frontier(inside(path_image c1 UNION path_image c))` MP_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[frontier; IN_DIFF]] THEN REWRITE_TAC[CLOSURE_APPROACHABLE; GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN MESON_TAC[]; ALL_TAC] THEN MP_TAC(ASSUME `connected(inside(path_image c1 UNION path_image c):real^2->bool)`) THEN REWRITE_TAC[connected] THEN MAP_EVERY EXISTS_TAC [`u:real^2->bool`; `v:real^2->bool`] THEN ASM_REWRITE_TAC[GSYM INTER_ASSOC; INTER_EMPTY] THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `j SUBSET t ==> s INTER t = {} ==> s SUBSET UNIV DIFF j`)) THEN MATCH_MP_TAC(SET_RULE `!x. s INTER t2 = {} /\ u SUBSET t2 /\ x INTER v = {} /\ s SUBSET x ==> s INTER (t1 INTER t2 INTER t3 UNION u UNION (v DIFF w)) = {}`) THEN EXISTS_TAC `inside(path_image c1 UNION path_image c2:real^2->bool)` THEN ASM_REWRITE_TAC[INSIDE_INTER_OUTSIDE; INSIDE_NO_OVERLAP]; SUBGOAL_THEN `~((u:real^2->bool) INTER inside(path_image c2 UNION path_image c) = {})` ASSUME_TAC THENL [MP_TAC(ASSUME `open(u:real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL; SUBSET; IN_BALL] THEN DISCH_THEN(MP_TAC o SPEC `a:real^2`) THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[DIST_SYM] THEN SUBGOAL_THEN `(a:real^2) IN frontier(inside(path_image c2 UNION path_image c))` MP_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[frontier; IN_DIFF]] THEN REWRITE_TAC[CLOSURE_APPROACHABLE; GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN MESON_TAC[]; ALL_TAC] THEN SUBGOAL_THEN `~((v:real^2->bool) INTER inside(path_image c2 UNION path_image c) = {})` ASSUME_TAC THENL [MP_TAC(ASSUME `open(v:real^2->bool)`) THEN REWRITE_TAC[OPEN_CONTAINS_BALL; SUBSET; IN_BALL] THEN DISCH_THEN(MP_TAC o SPEC `p:real^2`) THEN ASM_REWRITE_TAC[] THEN ONCE_REWRITE_TAC[DIST_SYM] THEN SUBGOAL_THEN `(p:real^2) IN frontier(inside(path_image c2 UNION path_image c))` MP_TAC THENL [ASM SET_TAC[]; REWRITE_TAC[frontier; IN_DIFF]] THEN REWRITE_TAC[CLOSURE_APPROACHABLE; GSYM MEMBER_NOT_EMPTY; IN_INTER] THEN MESON_TAC[]; ALL_TAC] THEN MP_TAC(ASSUME `connected(inside(path_image c2 UNION path_image c):real^2->bool)`) THEN REWRITE_TAC[connected] THEN MAP_EVERY EXISTS_TAC [`u:real^2->bool`; `v:real^2->bool`] THEN ASM_REWRITE_TAC[GSYM INTER_ASSOC; INTER_EMPTY] THEN FIRST_X_ASSUM(MATCH_MP_TAC o MATCH_MP (SET_RULE `j SUBSET t ==> s INTER t = {} ==> s SUBSET UNIV DIFF j`)) THEN MATCH_MP_TAC(SET_RULE `!x. s INTER t3 = {} /\ u SUBSET t3 /\ x INTER v = {} /\ s SUBSET x ==> s INTER (t1 INTER t2 INTER t3 UNION u UNION (v DIFF w)) = {}`) THEN EXISTS_TAC `inside(path_image c1 UNION path_image c2:real^2->bool)` THEN ASM_REWRITE_TAC[INSIDE_INTER_OUTSIDE; INSIDE_NO_OVERLAP]]);;