2015-04-07 01:01:08 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2015 Floris van Doorn. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Authors: Floris van Doorn
|
|
|
|
|
|
|
|
|
|
Declaration of the circle
|
|
|
|
|
-/
|
|
|
|
|
|
2015-05-14 02:01:48 +00:00
|
|
|
|
import .sphere types.bool types.eq types.int.hott types.arrow types.equiv algebra.fundamental_group algebra.hott
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-05-07 02:48:11 +00:00
|
|
|
|
open eq suspension bool sphere_index is_equiv equiv equiv.ops is_trunc
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-05-07 20:35:14 +00:00
|
|
|
|
definition circle : Type₀ := sphere 1
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
|
|
|
|
namespace circle
|
2015-05-14 02:01:48 +00:00
|
|
|
|
notation `S¹` := circle
|
2015-04-07 01:01:08 +00:00
|
|
|
|
definition base1 : circle := !north
|
|
|
|
|
definition base2 : circle := !south
|
2015-04-28 02:05:59 +00:00
|
|
|
|
definition seg1 : base1 = base2 := merid !north
|
|
|
|
|
definition seg2 : base1 = base2 := merid !south
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
|
|
|
|
definition base : circle := base1
|
2015-04-28 02:05:59 +00:00
|
|
|
|
definition loop : base = base := seg1 ⬝ seg2⁻¹
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
|
|
|
|
definition rec2 {P : circle → Type} (Pb1 : P base1) (Pb2 : P base2)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
(Ps1 : seg1 ▸ Pb1 = Pb2) (Ps2 : seg2 ▸ Pb1 = Pb2) (x : circle) : P x :=
|
2015-04-07 01:01:08 +00:00
|
|
|
|
begin
|
|
|
|
|
fapply (suspension.rec_on x),
|
|
|
|
|
{ exact Pb1},
|
|
|
|
|
{ exact Pb2},
|
2015-04-28 02:05:59 +00:00
|
|
|
|
{ esimp, intro b, fapply (suspension.rec_on b),
|
|
|
|
|
{ exact Ps1},
|
|
|
|
|
{ exact Ps2},
|
|
|
|
|
{ intro x, cases x}},
|
2015-04-07 01:01:08 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
definition rec2_on [reducible] {P : circle → Type} (x : circle) (Pb1 : P base1) (Pb2 : P base2)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
(Ps1 : seg1 ▸ Pb1 = Pb2) (Ps2 : seg2 ▸ Pb1 = Pb2) : P x :=
|
2015-04-07 01:01:08 +00:00
|
|
|
|
circle.rec2 Pb1 Pb2 Ps1 Ps2 x
|
|
|
|
|
|
2015-04-27 21:34:55 +00:00
|
|
|
|
theorem rec2_seg1 {P : circle → Type} (Pb1 : P base1) (Pb2 : P base2)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
(Ps1 : seg1 ▸ Pb1 = Pb2) (Ps2 : seg2 ▸ Pb1 = Pb2)
|
2015-04-27 21:34:55 +00:00
|
|
|
|
: apd (rec2 Pb1 Pb2 Ps1 Ps2) seg1 = Ps1 :=
|
2015-04-28 01:30:20 +00:00
|
|
|
|
!rec_merid
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-04-27 21:34:55 +00:00
|
|
|
|
theorem rec2_seg2 {P : circle → Type} (Pb1 : P base1) (Pb2 : P base2)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
(Ps1 : seg1 ▸ Pb1 = Pb2) (Ps2 : seg2 ▸ Pb1 = Pb2)
|
2015-04-27 21:34:55 +00:00
|
|
|
|
: apd (rec2 Pb1 Pb2 Ps1 Ps2) seg2 = Ps2 :=
|
2015-04-28 02:05:59 +00:00
|
|
|
|
!rec_merid
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-05-07 20:35:14 +00:00
|
|
|
|
definition elim2 {P : Type} (Pb1 Pb2 : P) (Ps1 Ps2 : Pb1 = Pb2) (x : circle) : P :=
|
2015-04-07 01:01:08 +00:00
|
|
|
|
rec2 Pb1 Pb2 (!tr_constant ⬝ Ps1) (!tr_constant ⬝ Ps2) x
|
|
|
|
|
|
|
|
|
|
definition elim2_on [reducible] {P : Type} (x : circle) (Pb1 Pb2 : P)
|
2015-04-28 02:05:59 +00:00
|
|
|
|
(Ps1 : Pb1 = Pb2) (Ps2 : Pb1 = Pb2) : P :=
|
2015-04-07 01:01:08 +00:00
|
|
|
|
elim2 Pb1 Pb2 Ps1 Ps2 x
|
|
|
|
|
|
2015-04-28 02:05:59 +00:00
|
|
|
|
theorem elim2_seg1 {P : Type} (Pb1 Pb2 : P) (Ps1 : Pb1 = Pb2) (Ps2 : Pb1 = Pb2)
|
2015-04-27 21:34:55 +00:00
|
|
|
|
: ap (elim2 Pb1 Pb2 Ps1 Ps2) seg1 = Ps1 :=
|
|
|
|
|
begin
|
|
|
|
|
apply (@cancel_left _ _ _ _ (tr_constant seg1 (elim2 Pb1 Pb2 Ps1 Ps2 base1))),
|
|
|
|
|
rewrite [-apd_eq_tr_constant_con_ap,↑elim2,rec2_seg1],
|
|
|
|
|
end
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-04-28 02:05:59 +00:00
|
|
|
|
theorem elim2_seg2 {P : Type} (Pb1 Pb2 : P) (Ps1 : Pb1 = Pb2) (Ps2 : Pb1 = Pb2)
|
2015-04-27 21:34:55 +00:00
|
|
|
|
: ap (elim2 Pb1 Pb2 Ps1 Ps2) seg2 = Ps2 :=
|
|
|
|
|
begin
|
2015-04-28 02:05:59 +00:00
|
|
|
|
apply (@cancel_left _ _ _ _ (tr_constant seg2 (elim2 Pb1 Pb2 Ps1 Ps2 base1))),
|
2015-04-27 21:34:55 +00:00
|
|
|
|
rewrite [-apd_eq_tr_constant_con_ap,↑elim2,rec2_seg2],
|
|
|
|
|
end
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-05-07 20:35:14 +00:00
|
|
|
|
definition elim2_type (Pb1 Pb2 : Type) (Ps1 Ps2 : Pb1 ≃ Pb2) (x : circle) : Type :=
|
|
|
|
|
elim2 Pb1 Pb2 (ua Ps1) (ua Ps2) x
|
|
|
|
|
|
|
|
|
|
definition elim2_type_on [reducible] (x : circle) (Pb1 Pb2 : Type) (Ps1 Ps2 : Pb1 ≃ Pb2)
|
|
|
|
|
: Type :=
|
|
|
|
|
elim2_type Pb1 Pb2 Ps1 Ps2 x
|
|
|
|
|
|
|
|
|
|
theorem elim2_type_seg1 (Pb1 Pb2 : Type) (Ps1 Ps2 : Pb1 ≃ Pb2)
|
|
|
|
|
: transport (elim2_type Pb1 Pb2 Ps1 Ps2) seg1 = Ps1 :=
|
|
|
|
|
by rewrite [tr_eq_cast_ap_fn,↑elim2_type,elim2_seg1];apply cast_ua_fn
|
|
|
|
|
|
|
|
|
|
theorem elim2_type_seg2 (Pb1 Pb2 : Type) (Ps1 Ps2 : Pb1 ≃ Pb2)
|
|
|
|
|
: transport (elim2_type Pb1 Pb2 Ps1 Ps2) seg2 = Ps2 :=
|
|
|
|
|
by rewrite [tr_eq_cast_ap_fn,↑elim2_type,elim2_seg2];apply cast_ua_fn
|
|
|
|
|
|
2015-05-01 03:23:12 +00:00
|
|
|
|
protected definition rec {P : circle → Type} (Pbase : P base) (Ploop : loop ▸ Pbase = Pbase)
|
2015-04-07 01:01:08 +00:00
|
|
|
|
(x : circle) : P x :=
|
|
|
|
|
begin
|
|
|
|
|
fapply (rec2_on x),
|
|
|
|
|
{ exact Pbase},
|
|
|
|
|
{ exact (transport P seg1 Pbase)},
|
|
|
|
|
{ apply idp},
|
2015-04-28 21:31:26 +00:00
|
|
|
|
{ apply tr_eq_of_eq_inv_tr, exact (Ploop⁻¹ ⬝ !con_tr)},
|
2015-04-07 01:01:08 +00:00
|
|
|
|
end
|
2015-04-28 20:49:11 +00:00
|
|
|
|
--rewrite -tr_con, exact Ploop⁻¹
|
2015-04-27 21:34:55 +00:00
|
|
|
|
|
2015-04-07 01:01:08 +00:00
|
|
|
|
protected definition rec_on [reducible] {P : circle → Type} (x : circle) (Pbase : P base)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
(Ploop : loop ▸ Pbase = Pbase) : P x :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
circle.rec Pbase Ploop x
|
2015-04-28 20:49:11 +00:00
|
|
|
|
|
|
|
|
|
theorem rec_loop_helper {A : Type} (P : A → Type)
|
2015-05-01 03:23:12 +00:00
|
|
|
|
{x y : A} {p : x = y} {u : P x} {v : P y} (q : u = p⁻¹ ▸ v) :
|
2015-04-28 21:31:26 +00:00
|
|
|
|
eq_inv_tr_of_tr_eq (tr_eq_of_eq_inv_tr q) = q :=
|
2015-04-28 20:49:11 +00:00
|
|
|
|
by cases p; exact idp
|
|
|
|
|
|
2015-04-29 00:48:39 +00:00
|
|
|
|
definition con_refl {A : Type} {x y : A} (p : x = y) : p ⬝ refl _ = p :=
|
|
|
|
|
eq.rec_on p idp
|
|
|
|
|
|
2015-05-01 03:23:12 +00:00
|
|
|
|
theorem rec_loop {P : circle → Type} (Pbase : P base) (Ploop : loop ▸ Pbase = Pbase) :
|
2015-05-19 05:35:18 +00:00
|
|
|
|
apd (circle.rec Pbase Ploop) loop = Ploop :=
|
2015-04-28 20:49:11 +00:00
|
|
|
|
begin
|
2015-05-19 05:35:18 +00:00
|
|
|
|
rewrite [↑loop,apd_con,↑circle.rec,↑circle.rec2_on,↑base,rec2_seg1,apd_inv,rec2_seg2,↑ap], --con_idp should work here
|
2015-04-28 20:49:11 +00:00
|
|
|
|
apply concat, apply (ap (λx, x ⬝ _)), apply con_idp, esimp,
|
2015-04-28 21:31:26 +00:00
|
|
|
|
rewrite [rec_loop_helper,inv_con_inv_left],
|
2015-04-28 20:49:11 +00:00
|
|
|
|
apply con_inv_cancel_left
|
|
|
|
|
end
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
|
|
|
|
protected definition elim {P : Type} (Pbase : P) (Ploop : Pbase = Pbase)
|
|
|
|
|
(x : circle) : P :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
circle.rec Pbase (tr_constant loop Pbase ⬝ Ploop) x
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
|
|
|
|
protected definition elim_on [reducible] {P : Type} (x : circle) (Pbase : P)
|
|
|
|
|
(Ploop : Pbase = Pbase) : P :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
circle.elim Pbase Ploop x
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-04-27 21:34:55 +00:00
|
|
|
|
theorem elim_loop {P : Type} (Pbase : P) (Ploop : Pbase = Pbase) :
|
2015-05-19 05:35:18 +00:00
|
|
|
|
ap (circle.elim Pbase Ploop) loop = Ploop :=
|
2015-04-27 21:34:55 +00:00
|
|
|
|
begin
|
2015-05-19 05:35:18 +00:00
|
|
|
|
apply (@cancel_left _ _ _ _ (tr_constant loop (circle.elim Pbase Ploop base))),
|
|
|
|
|
rewrite [-apd_eq_tr_constant_con_ap,↑circle.elim,rec_loop],
|
2015-04-27 21:34:55 +00:00
|
|
|
|
end
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-04-19 21:56:24 +00:00
|
|
|
|
protected definition elim_type (Pbase : Type) (Ploop : Pbase ≃ Pbase)
|
|
|
|
|
(x : circle) : Type :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
circle.elim Pbase (ua Ploop) x
|
2015-04-19 21:56:24 +00:00
|
|
|
|
|
|
|
|
|
protected definition elim_type_on [reducible] (x : circle) (Pbase : Type)
|
|
|
|
|
(Ploop : Pbase ≃ Pbase) : Type :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
circle.elim_type Pbase Ploop x
|
2015-04-19 21:56:24 +00:00
|
|
|
|
|
2015-04-27 21:34:55 +00:00
|
|
|
|
theorem elim_type_loop (Pbase : Type) (Ploop : Pbase ≃ Pbase) :
|
2015-05-19 05:35:18 +00:00
|
|
|
|
transport (circle.elim_type Pbase Ploop) loop = Ploop :=
|
|
|
|
|
by rewrite [tr_eq_cast_ap_fn,↑circle.elim_type,circle.elim_loop];apply cast_ua_fn
|
2015-04-07 01:01:08 +00:00
|
|
|
|
|
2015-05-07 02:48:11 +00:00
|
|
|
|
theorem elim_type_loop_inv (Pbase : Type) (Ploop : Pbase ≃ Pbase) :
|
2015-05-19 05:35:18 +00:00
|
|
|
|
transport (circle.elim_type Pbase Ploop) loop⁻¹ = to_inv Ploop :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
by rewrite [tr_inv_fn,↑to_inv]; apply inv_eq_inv; apply elim_type_loop
|
2015-05-07 20:35:14 +00:00
|
|
|
|
end circle
|
|
|
|
|
|
|
|
|
|
attribute circle.base circle.base1 circle.base2 [constructor]
|
|
|
|
|
attribute circle.rec circle.elim [unfold-c 4]
|
|
|
|
|
attribute circle.elim_type [unfold-c 3]
|
|
|
|
|
attribute circle.rec_on circle.elim_on [unfold-c 2]
|
|
|
|
|
attribute circle.elim_type_on [unfold-c 1]
|
|
|
|
|
attribute circle.rec2 circle.elim2 [unfold-c 6]
|
|
|
|
|
attribute circle.elim2_type [unfold-c 5]
|
|
|
|
|
attribute circle.rec2_on circle.elim2_on [unfold-c 2]
|
|
|
|
|
attribute circle.elim2_type [unfold-c 1]
|
2015-05-07 02:48:11 +00:00
|
|
|
|
|
2015-05-07 20:35:14 +00:00
|
|
|
|
namespace circle
|
2015-05-14 02:01:48 +00:00
|
|
|
|
definition pointed_circle [instance] [constructor] : pointed circle :=
|
|
|
|
|
pointed.mk base
|
|
|
|
|
|
2015-05-01 00:45:31 +00:00
|
|
|
|
definition loop_neq_idp : loop ≠ idp :=
|
|
|
|
|
assume H : loop = idp,
|
|
|
|
|
have H2 : Π{A : Type₁} {a : A} (p : a = a), p = idp,
|
|
|
|
|
from λA a p, calc
|
|
|
|
|
p = ap (circle.elim a p) loop : elim_loop
|
|
|
|
|
... = ap (circle.elim a p) (refl base) : by rewrite H,
|
|
|
|
|
absurd !H2 eq_bnot_ne_idp
|
|
|
|
|
|
|
|
|
|
definition nonidp (x : circle) : x = x :=
|
|
|
|
|
circle.rec_on x loop
|
|
|
|
|
(calc
|
2015-05-01 03:23:12 +00:00
|
|
|
|
loop ▸ loop = loop⁻¹ ⬝ loop ⬝ loop : transport_eq_lr
|
2015-05-01 00:45:31 +00:00
|
|
|
|
... = loop : by rewrite [con.left_inv, idp_con])
|
|
|
|
|
|
|
|
|
|
definition nonidp_neq_idp : nonidp ≠ (λx, idp) :=
|
|
|
|
|
assume H : nonidp = λx, idp,
|
|
|
|
|
have H2 : loop = idp, from apd10 H base,
|
|
|
|
|
absurd H2 loop_neq_idp
|
|
|
|
|
|
2015-05-07 02:48:11 +00:00
|
|
|
|
open int
|
|
|
|
|
|
2015-05-07 22:25:35 +00:00
|
|
|
|
protected definition code (x : circle) : Type₀ :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
circle.elim_type_on x ℤ equiv_succ
|
|
|
|
|
|
2015-05-19 05:35:18 +00:00
|
|
|
|
definition transport_code_loop (a : ℤ) : transport circle.code loop a = succ a :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
ap10 !elim_type_loop a
|
|
|
|
|
|
|
|
|
|
definition transport_code_loop_inv (a : ℤ)
|
2015-05-19 05:35:18 +00:00
|
|
|
|
: transport circle.code loop⁻¹ a = pred a :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
ap10 !elim_type_loop_inv a
|
|
|
|
|
|
2015-05-19 05:35:18 +00:00
|
|
|
|
protected definition encode {x : circle} (p : base = x) : circle.code x :=
|
|
|
|
|
transport circle.code p (of_num 0) -- why is the explicit coercion needed here?
|
2015-05-07 02:48:11 +00:00
|
|
|
|
|
2015-05-19 05:35:18 +00:00
|
|
|
|
protected definition decode {x : circle} : circle.code x → base = x :=
|
2015-05-14 02:01:48 +00:00
|
|
|
|
begin
|
|
|
|
|
refine circle.rec_on x _ _,
|
|
|
|
|
{ exact power loop},
|
|
|
|
|
{ apply eq_of_homotopy, intro a,
|
|
|
|
|
refine !arrow.arrow_transport ⬝ !transport_eq_r ⬝ _,
|
|
|
|
|
rewrite [transport_code_loop_inv,power_con,succ_pred]}
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
--remove this theorem after #484
|
2015-05-19 05:35:18 +00:00
|
|
|
|
theorem encode_decode {x : circle} : Π(a : circle.code x), circle.encode (circle.decode a) = a :=
|
2015-05-14 02:01:48 +00:00
|
|
|
|
begin
|
2015-05-19 05:35:18 +00:00
|
|
|
|
unfold circle.decode, refine circle.rec_on x _ _,
|
2015-05-14 02:01:48 +00:00
|
|
|
|
{ intro a, esimp [base,base1], --simplify after #587
|
|
|
|
|
apply rec_nat_on a,
|
|
|
|
|
{ exact idp},
|
|
|
|
|
{ intros n p,
|
2015-05-19 05:35:18 +00:00
|
|
|
|
apply transport (λ(y : base = base), transport circle.code y _ = _), apply power_con,
|
|
|
|
|
rewrite [▸*,con_tr, transport_code_loop, ↑[circle.encode,circle.code] at p, p]},
|
2015-05-14 02:01:48 +00:00
|
|
|
|
{ intros n p,
|
2015-05-19 05:35:18 +00:00
|
|
|
|
apply transport (λ(y : base = base), transport circle.code y _ = _),
|
2015-05-14 02:01:48 +00:00
|
|
|
|
{ exact !power_con_inv ⬝ ap (power loop) !neg_succ⁻¹},
|
2015-05-19 05:35:18 +00:00
|
|
|
|
rewrite [▸*,@con_tr _ circle.code,transport_code_loop_inv, ↑[circle.encode] at p, p, -neg_succ]}},
|
|
|
|
|
{ apply eq_of_homotopy, intro a, apply @is_hset.elim, esimp [circle.code,base,base1], exact _}
|
2015-05-14 02:01:48 +00:00
|
|
|
|
--simplify after #587
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
2015-05-19 05:35:18 +00:00
|
|
|
|
definition circle_eq_equiv (x : circle) : (base = x) ≃ circle.code x :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
begin
|
|
|
|
|
fapply equiv.MK,
|
2015-05-19 05:35:18 +00:00
|
|
|
|
{ exact circle.encode},
|
|
|
|
|
{ exact circle.decode},
|
|
|
|
|
{ exact circle.encode_decode},
|
2015-05-07 02:48:11 +00:00
|
|
|
|
{ intro p, cases p, exact idp},
|
|
|
|
|
end
|
|
|
|
|
|
2015-05-14 02:01:48 +00:00
|
|
|
|
definition base_eq_base_equiv : base = base ≃ ℤ :=
|
2015-05-07 02:48:11 +00:00
|
|
|
|
circle_eq_equiv base
|
|
|
|
|
|
2015-05-14 02:01:48 +00:00
|
|
|
|
definition decode_add (a b : ℤ) :
|
|
|
|
|
base_eq_base_equiv⁻¹ a ⬝ base_eq_base_equiv⁻¹ b = base_eq_base_equiv⁻¹ (a + b) :=
|
|
|
|
|
!power_con_power
|
|
|
|
|
|
2015-05-19 05:35:18 +00:00
|
|
|
|
definition encode_con (p q : base = base) : circle.encode (p ⬝ q) = circle.encode p + circle.encode q :=
|
2015-05-14 02:01:48 +00:00
|
|
|
|
preserve_binary_of_inv_preserve base_eq_base_equiv concat add decode_add p q
|
|
|
|
|
|
|
|
|
|
--the carrier of π₁(S¹) is the set-truncation of base = base.
|
|
|
|
|
open core algebra trunc equiv.ops
|
|
|
|
|
definition fg_carrier_equiv_int : π₁(S¹) ≃ ℤ :=
|
|
|
|
|
trunc_equiv_trunc 0 base_eq_base_equiv ⬝e !equiv_trunc⁻¹ᵉ
|
|
|
|
|
|
|
|
|
|
definition fundamental_group_of_circle : π₁(S¹) = group_integers :=
|
|
|
|
|
begin
|
|
|
|
|
apply (Group_eq fg_carrier_equiv_int),
|
|
|
|
|
intros g h,
|
|
|
|
|
apply trunc.rec_on g, intro g', apply trunc.rec_on h, intro h',
|
|
|
|
|
-- esimp at *,
|
|
|
|
|
-- esimp [fg_carrier_equiv_int,equiv.trans,equiv.symm,equiv_trunc,trunc_equiv_trunc,
|
|
|
|
|
-- base_eq_base_equiv,circle_eq_equiv,is_equiv_tr,semigroup.to_has_mul,monoid.to_semigroup,
|
|
|
|
|
-- group.to_monoid,fundamental_group.mul],
|
|
|
|
|
apply encode_con,
|
|
|
|
|
end
|
2015-05-07 02:48:11 +00:00
|
|
|
|
|
2015-04-07 01:01:08 +00:00
|
|
|
|
end circle
|