Mctt.Core.Semantic.PER.Definitions

From Coq Require Import Lia PeanoNat Relation_Definitions RelationClasses.
From Equations Require Import Equations.

From Mctt Require Import LibTactics.
From Mctt.Core Require Import Base.
From Mctt.Core.Semantic Require Export Domain Evaluation Readback.
Import Domain_Notations.

Notation "'Dom' a ≈ b ∈ R" := ((R a b : Prop) : Prop) (in custom judg at level 90, a custom domain, b custom domain, R constr).
Notation "'DF' a ≈ b ∈ R ↘ R'" := ((R R' a b : Prop) : Prop) (in custom judg at level 90, a custom domain, b custom domain, R constr, R' constr).
Notation "'Exp' a ≈ b ∈ R" := (R a b : (Prop : Type)) (in custom judg at level 90, a custom exp, b custom exp, R constr).
Notation "'EF' a ≈ b ∈ R ↘ R'" := (R R' a b : (Prop : Type)) (in custom judg at level 90, a custom exp, b custom exp, R constr, R' constr).
Precedences of the next notations follow the ones in the standard library. However, we do not use the ones in the standard library so that we can change the relation if necessary in the future.
Notation "R ~> R'" := (subrelation R R') (at level 70, right associativity).
Notation "R <~> R'" := (relation_equivalence R R') (at level 95, no associativity).

Generalizable All Variables.

Helper Bundles

Related modulo evaluation
Variant rel_mod_eval (R : relation domain -> domain -> domain -> Prop) A ρ A' ρ' R' : Prop := mk_rel_mod_eval : forall a a', {{ A ρ a }} -> {{ A' ρ' a' }} -> {{ DF a a' R R' }} -> rel_mod_eval R A ρ A' ρ' R'.
#[global]
Arguments mk_rel_mod_eval {_ _ _ _ _ _}.
#[export]
Hint Constructors rel_mod_eval : mctt.

Related modulo application
Variant rel_mod_app f a f' a' (R : relation domain) : Prop := mk_rel_mod_app : forall fa f'a', {{ $| f & a |↘ fa }} -> {{ $| f' & a' |↘ f'a' }} -> {{ Dom fa f'a' R }} -> rel_mod_app f a f' a' R.
#[global]
Arguments mk_rel_mod_app {_ _ _ _ _}.
#[export]
Hint Constructors rel_mod_app : mctt.

(Some Elements of) PER Lattice


Definition per_bot : relation domain_ne := fun m m' => (forall s, exists L, {{ Rne m in s L }} /\ {{ Rne m' in s L }}).
#[global]
Arguments per_bot /.
#[export]
Hint Transparent per_bot : mctt.
#[export]
Hint Unfold per_bot : mctt.

Definition per_top : relation domain_nf := fun m m' => (forall s, exists L, {{ Rnf m in s L }} /\ {{ Rnf m' in s L }}).
#[global]
Arguments per_top /.
#[export]
Hint Transparent per_top : mctt.
#[export]
Hint Unfold per_top : mctt.

Definition per_top_typ : relation domain := fun a a' => (forall s, exists C, {{ Rtyp a in s C }} /\ {{ Rtyp a' in s C }}).
#[global]
Arguments per_top_typ /.
#[export]
Hint Transparent per_top_typ : mctt.
#[export]
Hint Unfold per_top_typ : mctt.

Inductive per_nat : relation domain :=
| per_nat_zero : {{ Dom zero zero per_nat }}
| per_nat_succ :
  `{ {{ Dom m m' per_nat }} ->
     {{ Dom succ m succ m' per_nat }} }
| per_nat_neut :
  `{ {{ Dom m m' per_bot }} ->
     {{ Dom a m a' m' per_nat }} }
.
#[export]
Hint Constructors per_nat : mctt.

Variant per_eq (point_rel : relation domain) m1 m2' : relation domain :=
| per_eq_refl :
  `{ {{ Dom m1 n point_rel }} ->
     {{ Dom n n' point_rel }} ->
     {{ Dom n' m2' point_rel }} ->
     {{ Dom refl n refl n' per_eq point_rel m1 m2' }} }
| per_eq_neut :
  `{ {{ Dom n n' per_bot }} ->
     {{ Dom a n a' n' per_eq point_rel m1 m2' }} }
.
#[export]
Hint Constructors per_eq : mctt.

Variant per_ne : relation domain :=
| per_ne_neut :
  `{ {{ Dom m m' per_bot }} ->
     {{ Dom a m a' m' per_ne }} }
.
#[export]
Hint Constructors per_ne : mctt.

Universe/Element PER

Universe/Element PER Definition


Section Per_univ_elem_core_def.
  Variable
    (i : nat)
      (per_univ_rec : forall {j}, j < i -> relation domain).

  Inductive per_univ_elem_core : relation domain -> domain -> domain -> Prop :=
  | per_univ_elem_core_univ :
    `{ forall (elem_rel : relation domain)
          (lt_j_i : j < i),
          j = j' ->
          (elem_rel <~> per_univ_rec lt_j_i) ->
          {{ DF 𝕌@j 𝕌@j' per_univ_elem_core elem_rel }} }
  | per_univ_elem_core_nat :
    forall (elem_rel : relation domain),
      (elem_rel <~> per_nat) ->
      {{ DF per_univ_elem_core elem_rel }}
  | per_univ_elem_core_pi :
    `{ forall (in_rel : relation domain)
         (out_rel : forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}), relation domain)
         (elem_rel : relation domain)
         (equiv_a_a' : {{ DF a a' per_univ_elem_core in_rel }}),
          PER in_rel ->
          (forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}),
              rel_mod_eval per_univ_elem_core B d{{{ ρ c }}} B' d{{{ ρ' c' }}} (out_rel equiv_c_c')) ->
          (elem_rel <~> fun f f' => forall c c' (equiv_c_c' : {{ Dom c c' in_rel }}), rel_mod_app f c f' c' (out_rel equiv_c_c')) ->
          {{ DF Π a ρ B Π a' ρ' B' per_univ_elem_core elem_rel }} }
  | per_univ_elem_core_eq :
    `{ forall (point_rel : relation domain)
         (elem_rel : relation domain),
          {{ DF a a' per_univ_elem_core point_rel }} ->
          PER point_rel ->
          {{ Dom m1 m1' point_rel }} ->
          {{ Dom m2 m2' point_rel }} ->
          (elem_rel <~> per_eq point_rel m1 m2') ->
          {{ DF Eq a m1 m2 Eq a' m1' m2' per_univ_elem_core elem_rel }} }
  | per_univ_elem_core_neut :
    `{ forall (elem_rel : relation domain),
          {{ Dom b b' per_bot }} ->
          (elem_rel <~> per_ne) ->
          {{ DF a b a' b' per_univ_elem_core elem_rel }} }
  .

  Hypothesis
    (motive : relation domain -> domain -> domain -> Prop)
      (case_U : forall {j j' elem_rel} (lt_j_i : j < i),
          j = j' ->
          (elem_rel <~> per_univ_rec lt_j_i) ->
          motive elem_rel d{{{ 𝕌@j }}} d{{{ 𝕌@j' }}})
      (case_nat : forall {elem_rel},
          (elem_rel <~> per_nat) ->
          motive elem_rel d{{{ }}} d{{{ }}})
      (case_Pi :
        forall {a ρ B a' ρ' B' in_rel}
           (out_rel : forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}), relation domain)
           {elem_rel},
          {{ DF a a' per_univ_elem_core in_rel }} ->
          motive in_rel a a' ->
          PER in_rel ->
          (forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}),
              rel_mod_eval (fun R x y => {{ DF x y per_univ_elem_core R }} /\ motive R x y) B d{{{ ρ c }}} B' d{{{ ρ' c' }}} (out_rel equiv_c_c')) ->
          (elem_rel <~> fun f f' => forall c c' (equiv_c_c' : {{ Dom c c' in_rel }}), rel_mod_app f c f' c' (out_rel equiv_c_c')) ->
          motive elem_rel d{{{ Π a ρ B }}} d{{{ Π a' ρ' B' }}})
      (case_Eq : forall {a m1 m2 a' m1' m2' point_rel elem_rel},
          {{ DF a a' per_univ_elem_core point_rel }} ->
          motive point_rel a a' ->
          PER point_rel ->
          {{ Dom m1 m1' point_rel }} ->
          {{ Dom m2 m2' point_rel }} ->
          (elem_rel <~> per_eq point_rel m1 m2') ->
          motive elem_rel d{{{ Eq a m1 m2 }}} d{{{ Eq a' m1' m2' }}})
      (case_ne : forall {a b a' b' elem_rel},
          {{ Dom b b' per_bot }} ->
          (elem_rel <~> per_ne) ->
          motive elem_rel d{{{ a b }}} d{{{ a' b' }}}).

  #[derive(equations=no, eliminator=no)]
  Equations per_univ_elem_core_strong_ind R a b (H : {{ DF a b per_univ_elem_core R }}) : {{ DF a b motive R }} :=
  | R, a, b, (per_univ_elem_core_univ _ lt_j_i HE eq) => case_U lt_j_i HE eq;
  | R, a, b, (per_univ_elem_core_nat _ HE) => case_nat HE;
  | R, a, b, (per_univ_elem_core_pi _ out_rel _ equiv_a_a' per HT HE) =>
      case_Pi out_rel equiv_a_a' (per_univ_elem_core_strong_ind _ _ _ equiv_a_a') per
        (fun _ _ equiv_c_c' => match HT _ _ equiv_c_c' with
                              | mk_rel_mod_eval b b' evb evb' Rel =>
                                  mk_rel_mod_eval b b' evb evb' (conj _ (per_univ_elem_core_strong_ind _ _ _ Rel))
                              end)
        HE;
  | R, a, b, (per_univ_elem_core_eq _ _ HT per HE1 HE2 eq) => case_Eq HT (per_univ_elem_core_strong_ind _ _ _ HT) per HE1 HE2 eq;
  | R, a, b, (per_univ_elem_core_neut _ equiv_b_b' HE) => case_ne equiv_b_b' HE.

End Per_univ_elem_core_def.

#[export]
Hint Constructors per_univ_elem_core : mctt.

Equations per_univ_elem (i : nat) : relation domain -> domain -> domain -> Prop by wf i :=
| i => per_univ_elem_core i (fun j lt_j_i a a' => exists R', {{ DF a a' per_univ_elem j R' }}).

Definition per_univ (i : nat) : relation domain := fun a a' => exists R', {{ DF a a' per_univ_elem i R' }}.
#[global]
Arguments per_univ _ _ _ /.
#[export]
Hint Transparent per_univ : mctt.
#[export]
Hint Unfold per_univ : mctt.

Lemma per_univ_elem_core_univ' : forall j i elem_rel,
    j < i ->
    (elem_rel <~> per_univ j) ->
    {{ DF 𝕌@j 𝕌@j per_univ_elem i elem_rel }}.
Proof.
  intros.
  simp per_univ_elem.
  econstructor; mauto 3.
Qed.

#[export]
Hint Resolve per_univ_elem_core_univ' : mctt.

Universe/Element PER Induction Principle


Section Per_univ_elem_ind_def.
  Hypothesis
    (motive : nat -> relation domain -> domain -> domain -> Prop)
      (case_U : forall i {j j' elem_rel},
          j < i -> j = j' ->
          (elem_rel <~> per_univ j) ->
          (forall A B R, {{ DF A B per_univ_elem j R }} -> motive j R A B) ->
          motive i elem_rel d{{{ 𝕌@j }}} d{{{ 𝕌@j' }}})
      (case_N : forall i {elem_rel},
          (elem_rel <~> per_nat) ->
          motive i elem_rel d{{{ }}} d{{{ }}})
      (case_Pi :
        forall i {a ρ B a' ρ' B' in_rel}
           (out_rel : forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}), relation domain)
           {elem_rel},
          {{ DF a a' per_univ_elem i in_rel }} ->
          motive i in_rel a a' ->
          PER in_rel ->
          (forall {c c'} (equiv_c_c' : {{ Dom c c' in_rel }}),
              rel_mod_eval (fun R x y => {{ DF x y per_univ_elem i R }} /\ motive i R x y) B d{{{ ρ c }}} B' d{{{ ρ' c' }}} (out_rel equiv_c_c')) ->
          (elem_rel <~> fun f f' => forall c c' (equiv_c_c' : {{ Dom c c' in_rel }}), rel_mod_app f c f' c' (out_rel equiv_c_c')) ->
          motive i elem_rel d{{{ Π a ρ B }}} d{{{ Π a' ρ' B' }}})
      (case_Eq : forall i {a m1 m2 a' m1' m2' point_rel elem_rel},
          {{ DF a a' per_univ_elem i point_rel }} ->
          motive i point_rel a a' ->
          PER point_rel ->
          {{ Dom m1 m1' point_rel }} ->
          {{ Dom m2 m2' point_rel }} ->
          (elem_rel <~> per_eq point_rel m1 m2') ->
          motive i elem_rel d{{{ Eq a m1 m2 }}} d{{{ Eq a' m1' m2' }}})
      (case_ne : forall i {a b a' b' elem_rel},
          {{ Dom b b' per_bot }} ->
          (elem_rel <~> per_ne) ->
          motive i elem_rel d{{{ a b }}} d{{{ a' b' }}}).

  #[local]
  Ltac def_simp := simp per_univ_elem in *; mauto 3.

  #[derive(equations=no, eliminator=no), tactic="def_simp"]
  Equations per_univ_elem_ind' (i : nat) (R : relation domain) (a b : domain)
    (H : {{ DF a b per_univ_elem_core i (fun j lt_j_i a a' => exists R', {{ DF a a' per_univ_elem j R' }}) R }}) : {{ DF a b motive i R }} by wf i :=
  | i, R, a, b, H =>
      per_univ_elem_core_strong_ind i _ (motive i)
        (fun _ _ _ j_lt_i eq HE => case_U i j_lt_i eq HE (fun A B R' H' => per_univ_elem_ind' _ R' A B _))
        (fun _ => case_N i)
        (fun _ _ _ _ _ _ _ out_rel _ _ IHA per _ => case_Pi i out_rel _ IHA per _)
        (fun _ _ _ _ _ _ _ _ _ IHA per _ _ _ => case_Eq i _ IHA per _ _ _)
        (fun _ _ _ _ _ => case_ne i)
        R a b H.

  #[derive(equations=no, eliminator=no), tactic="def_simp"]
  Equations per_univ_elem_ind i a b R (H : per_univ_elem i a b R) : motive i a b R :=
  | i, a, b, R, H := per_univ_elem_ind' i a b R _.
End Per_univ_elem_ind_def.

Reserved Notation "'Sub' a <: b 'at' i" (in custom judg at level 90, a custom domain, b custom domain, i constr).

Universe Subtyping


Inductive per_subtyp : nat -> domain -> domain -> Prop :=
| per_subtyp_univ :
  `( i <= j ->
     j < k ->
     {{ Sub 𝕌@i <: 𝕌@j at k }} )
| per_subtyp_nat :
  `( {{ Sub <: at i }} )
| per_subtyp_pi :
  `( forall (in_rel : relation domain) elem_rel elem_rel',
        {{ DF a a' per_univ_elem i in_rel }} ->
        (forall c c' b b',
            {{ Dom c c' in_rel }} ->
            {{ B ρ c b }} ->
            {{ B' ρ' c' b' }} ->
            {{ Sub b <: b' at i }}) ->
        {{ DF Π a ρ B Π a ρ B per_univ_elem i elem_rel }} ->
        {{ DF Π a' ρ' B' Π a' ρ' B' per_univ_elem i elem_rel' }} ->
        {{ Sub Π a ρ B <: Π a' ρ' B' at i }} )
| per_subtyp_eq :
  `( forall elem_rel,
        {{ DF Eq a m1 m2 Eq a' m1' m2' per_univ_elem i elem_rel }} ->
        {{ Sub Eq a m1 m2 <: Eq a' m1' m2' at i }} )
| per_subtyp_neut :
  `( {{ Dom b b' per_bot }} ->
     {{ Sub a b <: a' b' at i }} )
where "'Sub' a <: b 'at' i" := (per_subtyp i a b) (in custom judg) : type_scope.

#[export]
 Hint Constructors per_subtyp : mctt.

Definition rel_typ i A ρ A' ρ' R' := rel_mod_eval (per_univ_elem i) A ρ A' ρ' R'.
Arguments rel_typ _ _ _ _ _ _ /.
#[export]
Hint Transparent rel_typ : mctt.
#[export]
Hint Unfold rel_typ : mctt.

Context/Environment PER


Variant cons_per_ctx_env tail_rel (head_rel : forall {ρ ρ'} (equiv_ρ_ρ' : {{ Dom ρ ρ' tail_rel }}), relation domain) : relation env :=
| mk_cons_per_ctx_env :
  `{ forall (equiv_ρ_drop_ρ'_drop : {{ Dom ρ ρ' tail_rel }}),
        {{ Dom ^(ρ 0) ^(ρ' 0) head_rel equiv_ρ_drop_ρ'_drop }} ->
        {{ Dom ρ ρ' cons_per_ctx_env tail_rel (@head_rel) }} }.
#[export]
Hint Constructors cons_per_ctx_env : mctt.

Inductive per_ctx_env : relation env -> ctx -> ctx -> Prop :=
| per_ctx_env_nil :
  `{ forall env_rel,
        (env_rel <~> fun ρ ρ' => True) ->
        {{ EF per_ctx_env env_rel }} }
| per_ctx_env_cons :
  `{ forall tail_rel
        (head_rel : forall {ρ ρ'} (equiv_ρ_ρ' : {{ Dom ρ ρ' tail_rel }}), relation domain)
        env_rel
        (equiv_Γ_Γ' : {{ EF Γ Γ' per_ctx_env tail_rel }}),
        PER tail_rel ->
        (forall {ρ ρ'} (equiv_ρ_ρ' : {{ Dom ρ ρ' tail_rel }}),
            rel_typ i A ρ A' ρ' (head_rel equiv_ρ_ρ')) ->
        (env_rel <~> cons_per_ctx_env tail_rel (@head_rel)) ->
        {{ EF Γ, A Γ', A' per_ctx_env env_rel }} }
.
#[export]
Hint Constructors per_ctx_env : mctt.

Definition per_ctx : relation ctx := fun Γ Γ' => exists R', per_ctx_env R' Γ Γ'.
Definition valid_ctx : ctx -> Prop := fun Γ => per_ctx Γ Γ.
#[export]
Hint Transparent valid_ctx : mctt.
#[export]
Hint Unfold valid_ctx : mctt.

Reserved Notation "'SubE' Γ <: Δ" (in custom judg at level 90, Γ custom exp, Δ custom exp).

Context Subtyping


Inductive per_ctx_subtyp : ctx -> ctx -> Prop :=
| per_ctx_subtyp_nil :
  {{ SubE <: }}
| per_ctx_subtyp_cons :
  `{ forall tail_rel env_rel env_rel',
        {{ SubE Γ <: Γ' }} ->
        {{ EF Γ Γ per_ctx_env tail_rel }} ->
        (forall ρ ρ' a a'
           (equiv_ρ_ρ' : {{ Dom ρ ρ' tail_rel }}),
            {{ A ρ a }} ->
            {{ A' ρ' a' }} ->
            {{ Sub a <: a' at i }}) ->
        {{ EF Γ , A Γ , A per_ctx_env env_rel }} ->
        {{ EF Γ' , A' Γ' , A' per_ctx_env env_rel' }} ->
        {{ SubE Γ, A <: Γ', A' }} }
where "'SubE' Γ <: Δ" := (per_ctx_subtyp Γ Δ) (in custom judg) : type_scope.

#[export]
Hint Constructors per_ctx_subtyp : mctt.