2023-08-24 12:42:26 -04:00
|
|
|
|
module Quox.Whnf.Interface
|
|
|
|
|
|
|
|
|
|
import public Quox.No
|
|
|
|
|
import public Quox.Syntax
|
|
|
|
|
import public Quox.Definition
|
|
|
|
|
import public Quox.Typing.Context
|
|
|
|
|
import public Quox.Typing.Error
|
|
|
|
|
import public Data.Maybe
|
|
|
|
|
import public Control.Eff
|
|
|
|
|
|
|
|
|
|
%default total
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public export
|
|
|
|
|
Whnf : List (Type -> Type)
|
2023-10-20 00:09:20 -04:00
|
|
|
|
Whnf = [Except Error, NameGen]
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public export
|
|
|
|
|
0 RedexTest : TermLike -> Type
|
2023-12-06 19:35:39 -05:00
|
|
|
|
RedexTest tm =
|
|
|
|
|
{0 d, n : Nat} -> Definitions -> WhnfContext d n -> SQty -> tm d n -> Bool
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
public export
|
|
|
|
|
interface CanWhnf (0 tm : TermLike) (0 isRedex : RedexTest tm) | tm
|
|
|
|
|
where
|
2023-10-15 10:23:38 -04:00
|
|
|
|
whnf : (defs : Definitions) -> (ctx : WhnfContext d n) -> (q : SQty) ->
|
2023-12-06 19:35:39 -05:00
|
|
|
|
tm d n -> Eff Whnf (Subset (tm d n) (No . isRedex defs ctx q))
|
2023-08-24 12:42:26 -04:00
|
|
|
|
-- having isRedex be part of the class header, and needing to be explicitly
|
|
|
|
|
-- quantified on every use since idris can't infer its type, is a little ugly.
|
|
|
|
|
-- but none of the alternatives i've thought of so far work. e.g. in some
|
|
|
|
|
-- cases idris can't tell that `isRedex` and `isRedexT` are the same thing
|
|
|
|
|
|
|
|
|
|
public export %inline
|
2023-10-15 10:23:38 -04:00
|
|
|
|
whnf0 : {0 isRedex : RedexTest tm} -> CanWhnf tm isRedex =>
|
2023-09-18 12:21:30 -04:00
|
|
|
|
Definitions -> WhnfContext d n -> SQty -> tm d n -> Eff Whnf (tm d n)
|
|
|
|
|
whnf0 defs ctx q t = fst <$> whnf defs ctx q t
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
public export
|
|
|
|
|
0 IsRedex, NotRedex : {isRedex : RedexTest tm} -> CanWhnf tm isRedex =>
|
2023-12-06 19:35:39 -05:00
|
|
|
|
Definitions -> WhnfContext d n -> SQty -> Pred (tm d n)
|
|
|
|
|
IsRedex defs ctx q = So . isRedex defs ctx q
|
|
|
|
|
NotRedex defs ctx q = No . isRedex defs ctx q
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
public export
|
|
|
|
|
0 NonRedex : (tm : TermLike) -> {isRedex : RedexTest tm} ->
|
2023-09-18 12:21:30 -04:00
|
|
|
|
CanWhnf tm isRedex => (d, n : Nat) ->
|
2023-12-06 19:35:39 -05:00
|
|
|
|
Definitions -> WhnfContext d n -> SQty -> Type
|
|
|
|
|
NonRedex tm d n defs ctx q = Subset (tm d n) (NotRedex defs ctx q)
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
public export %inline
|
|
|
|
|
nred : {0 isRedex : RedexTest tm} -> (0 _ : CanWhnf tm isRedex) =>
|
2023-12-06 19:35:39 -05:00
|
|
|
|
(t : tm d n) -> (0 nr : NotRedex defs ctx q t) =>
|
|
|
|
|
NonRedex tm d n defs ctx q
|
2023-08-24 12:42:26 -04:00
|
|
|
|
nred t = Element t nr
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
||| an expression like `(λ x ⇒ s) ∷ π.(x : A) → B`
|
|
|
|
|
public export %inline
|
|
|
|
|
isLamHead : Elim {} -> Bool
|
|
|
|
|
isLamHead (Ann (Lam {}) (Pi {}) _) = True
|
|
|
|
|
isLamHead (Coe {}) = True
|
|
|
|
|
isLamHead _ = False
|
|
|
|
|
|
|
|
|
|
||| an expression like `(δ 𝑖 ⇒ s) ∷ Eq (𝑖 ⇒ A) s t`
|
|
|
|
|
public export %inline
|
|
|
|
|
isDLamHead : Elim {} -> Bool
|
|
|
|
|
isDLamHead (Ann (DLam {}) (Eq {}) _) = True
|
|
|
|
|
isDLamHead (Coe {}) = True
|
|
|
|
|
isDLamHead _ = False
|
|
|
|
|
|
|
|
|
|
||| an expression like `(s, t) ∷ (x : A) × B`
|
|
|
|
|
public export %inline
|
|
|
|
|
isPairHead : Elim {} -> Bool
|
|
|
|
|
isPairHead (Ann (Pair {}) (Sig {}) _) = True
|
|
|
|
|
isPairHead (Coe {}) = True
|
|
|
|
|
isPairHead _ = False
|
|
|
|
|
|
|
|
|
|
||| an expression like `'a ∷ {a, b, c}`
|
|
|
|
|
public export %inline
|
|
|
|
|
isTagHead : Elim {} -> Bool
|
|
|
|
|
isTagHead (Ann (Tag {}) (Enum {}) _) = True
|
|
|
|
|
isTagHead (Coe {}) = True
|
|
|
|
|
isTagHead _ = False
|
|
|
|
|
|
|
|
|
|
||| an expression like `0 ∷ ℕ` or `suc n ∷ ℕ`
|
|
|
|
|
public export %inline
|
|
|
|
|
isNatHead : Elim {} -> Bool
|
2023-11-02 15:01:34 -04:00
|
|
|
|
isNatHead (Ann (Nat {}) (NAT {}) _) = True
|
2023-11-02 13:14:22 -04:00
|
|
|
|
isNatHead (Ann (Succ {}) (NAT {}) _) = True
|
2023-08-24 12:42:26 -04:00
|
|
|
|
isNatHead (Coe {}) = True
|
|
|
|
|
isNatHead _ = False
|
|
|
|
|
|
2023-11-02 15:01:34 -04:00
|
|
|
|
||| a natural constant, with or without an annotation
|
|
|
|
|
public export %inline
|
|
|
|
|
isNatConst : Term d n -> Bool
|
|
|
|
|
isNatConst (Nat {}) = True
|
|
|
|
|
isNatConst (E (Ann (Nat {}) _ _)) = True
|
|
|
|
|
isNatConst _ = False
|
|
|
|
|
|
2023-08-24 12:42:26 -04:00
|
|
|
|
||| an expression like `[s] ∷ [π. A]`
|
|
|
|
|
public export %inline
|
|
|
|
|
isBoxHead : Elim {} -> Bool
|
|
|
|
|
isBoxHead (Ann (Box {}) (BOX {}) _) = True
|
|
|
|
|
isBoxHead (Coe {}) = True
|
|
|
|
|
isBoxHead _ = False
|
|
|
|
|
|
|
|
|
|
||| an elimination in a term context
|
|
|
|
|
public export %inline
|
|
|
|
|
isE : Term {} -> Bool
|
|
|
|
|
isE (E {}) = True
|
|
|
|
|
isE _ = False
|
|
|
|
|
|
|
|
|
|
||| an expression like `s ∷ A`
|
|
|
|
|
public export %inline
|
|
|
|
|
isAnn : Elim {} -> Bool
|
|
|
|
|
isAnn (Ann {}) = True
|
|
|
|
|
isAnn _ = False
|
|
|
|
|
|
|
|
|
|
||| a syntactic type
|
|
|
|
|
public export %inline
|
|
|
|
|
isTyCon : Term {} -> Bool
|
2023-11-01 10:17:15 -04:00
|
|
|
|
isTyCon (TYPE {}) = True
|
|
|
|
|
isTyCon (IOState {}) = True
|
|
|
|
|
isTyCon (Pi {}) = True
|
|
|
|
|
isTyCon (Lam {}) = False
|
|
|
|
|
isTyCon (Sig {}) = True
|
|
|
|
|
isTyCon (Pair {}) = False
|
|
|
|
|
isTyCon (Enum {}) = True
|
|
|
|
|
isTyCon (Tag {}) = False
|
|
|
|
|
isTyCon (Eq {}) = True
|
|
|
|
|
isTyCon (DLam {}) = False
|
2023-11-02 13:14:22 -04:00
|
|
|
|
isTyCon (NAT {}) = True
|
2023-11-02 15:01:34 -04:00
|
|
|
|
isTyCon (Nat {}) = False
|
2023-11-01 10:17:15 -04:00
|
|
|
|
isTyCon (Succ {}) = False
|
|
|
|
|
isTyCon (STRING {}) = True
|
|
|
|
|
isTyCon (Str {}) = False
|
|
|
|
|
isTyCon (BOX {}) = True
|
|
|
|
|
isTyCon (Box {}) = False
|
2023-12-04 16:47:52 -05:00
|
|
|
|
isTyCon (Let {}) = False
|
2023-11-01 10:17:15 -04:00
|
|
|
|
isTyCon (E {}) = False
|
|
|
|
|
isTyCon (CloT {}) = False
|
|
|
|
|
isTyCon (DCloT {}) = False
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
||| a syntactic type, or a neutral
|
|
|
|
|
public export %inline
|
|
|
|
|
isTyConE : Term {} -> Bool
|
|
|
|
|
isTyConE s = isTyCon s || isE s
|
|
|
|
|
|
|
|
|
|
||| a syntactic type with an annotation `★ᵢ`
|
|
|
|
|
public export %inline
|
|
|
|
|
isAnnTyCon : Elim {} -> Bool
|
|
|
|
|
isAnnTyCon (Ann ty (TYPE {}) _) = isTyCon ty
|
|
|
|
|
isAnnTyCon _ = False
|
|
|
|
|
|
|
|
|
|
||| 0 or 1
|
|
|
|
|
public export %inline
|
|
|
|
|
isK : Dim d -> Bool
|
|
|
|
|
isK (K {}) = True
|
|
|
|
|
isK _ = False
|
|
|
|
|
|
|
|
|
|
|
2023-08-26 15:00:19 -04:00
|
|
|
|
||| if `ty` is a type constructor, and `val` is a value of that type where a
|
|
|
|
|
||| coercion can be reduced. which is the case if any of:
|
|
|
|
|
||| - `ty` is an atomic type
|
|
|
|
|
||| - `ty` has η
|
|
|
|
|
||| - `val` is a constructor form
|
|
|
|
|
public export %inline
|
2023-09-18 12:21:30 -04:00
|
|
|
|
canPushCoe : SQty -> (ty, val : Term {}) -> Bool
|
2023-11-01 10:17:15 -04:00
|
|
|
|
canPushCoe sg (TYPE {}) _ = True
|
|
|
|
|
canPushCoe sg (IOState {}) _ = True
|
|
|
|
|
canPushCoe sg (Pi {}) _ = True
|
|
|
|
|
canPushCoe sg (Lam {}) _ = False
|
|
|
|
|
canPushCoe sg (Sig {}) (Pair {}) = True
|
|
|
|
|
canPushCoe sg (Sig {}) _ = False
|
|
|
|
|
canPushCoe sg (Pair {}) _ = False
|
|
|
|
|
canPushCoe sg (Enum {}) _ = True
|
|
|
|
|
canPushCoe sg (Tag {}) _ = False
|
|
|
|
|
canPushCoe sg (Eq {}) _ = True
|
|
|
|
|
canPushCoe sg (DLam {}) _ = False
|
2023-11-02 13:14:22 -04:00
|
|
|
|
canPushCoe sg (NAT {}) _ = True
|
2023-11-02 15:01:34 -04:00
|
|
|
|
canPushCoe sg (Nat {}) _ = False
|
2023-11-01 10:17:15 -04:00
|
|
|
|
canPushCoe sg (Succ {}) _ = False
|
|
|
|
|
canPushCoe sg (STRING {}) _ = True
|
|
|
|
|
canPushCoe sg (Str {}) _ = False
|
|
|
|
|
canPushCoe sg (BOX {}) _ = True
|
|
|
|
|
canPushCoe sg (Box {}) _ = False
|
2023-12-04 16:47:52 -05:00
|
|
|
|
canPushCoe sg (Let {}) _ = False
|
2023-11-01 10:17:15 -04:00
|
|
|
|
canPushCoe sg (E {}) _ = False
|
|
|
|
|
canPushCoe sg (CloT {}) _ = False
|
|
|
|
|
canPushCoe sg (DCloT {}) _ = False
|
2023-08-26 15:00:19 -04:00
|
|
|
|
|
|
|
|
|
|
2023-08-24 12:42:26 -04:00
|
|
|
|
mutual
|
|
|
|
|
||| a reducible elimination
|
|
|
|
|
|||
|
2023-08-26 15:00:19 -04:00
|
|
|
|
||| 1. a free variable, if its definition is known
|
2023-12-06 19:35:39 -05:00
|
|
|
|
||| 2. a bound variable pointing to a `let`
|
|
|
|
|
||| 3. an elimination whose head is reducible
|
|
|
|
|
||| 4. an "active" elimination:
|
2023-08-26 15:00:19 -04:00
|
|
|
|
||| an application whose head is an annotated lambda,
|
|
|
|
|
||| a case expression whose head is an annotated constructor form, etc
|
2023-12-06 19:35:39 -05:00
|
|
|
|
||| 5. a redundant annotation, or one whose term or type is reducible
|
|
|
|
|
||| 6. a coercion `coe (𝑖 ⇒ A) @p @q s` where:
|
2023-08-26 15:00:19 -04:00
|
|
|
|
||| a. `A` is reducible or a type constructor, or
|
|
|
|
|
||| b. `𝑖` is not mentioned in `A`
|
|
|
|
|
||| ([fixme] should be A‹0/𝑖› = A‹1/𝑖›), or
|
2023-09-18 15:52:51 -04:00
|
|
|
|
||| c. `p = q`
|
2023-12-06 19:35:39 -05:00
|
|
|
|
||| 7. a composition `comp A @p @q s @r {⋯}`
|
2023-09-18 15:52:51 -04:00
|
|
|
|
||| where `p = q`, `r = 0`, or `r = 1`
|
2023-12-06 19:35:39 -05:00
|
|
|
|
||| 8. a closure
|
2023-08-24 12:42:26 -04:00
|
|
|
|
public export
|
|
|
|
|
isRedexE : RedexTest Elim
|
2023-12-06 19:35:39 -05:00
|
|
|
|
isRedexE defs ctx sg (F {x, u, _}) = isJust $ lookupElim0 x u defs
|
|
|
|
|
isRedexE _ ctx sg (B {i, _}) = isJust (ctx.tctx !! i).term
|
|
|
|
|
isRedexE defs ctx sg (App {fun, _}) =
|
|
|
|
|
isRedexE defs ctx sg fun || isLamHead fun
|
|
|
|
|
isRedexE defs ctx sg (CasePair {pair, _}) =
|
|
|
|
|
isRedexE defs ctx sg pair || isPairHead pair || isYes (sg `decEq` SZero)
|
|
|
|
|
isRedexE defs ctx sg (Fst pair _) =
|
|
|
|
|
isRedexE defs ctx sg pair || isPairHead pair
|
|
|
|
|
isRedexE defs ctx sg (Snd pair _) =
|
|
|
|
|
isRedexE defs ctx sg pair || isPairHead pair
|
|
|
|
|
isRedexE defs ctx sg (CaseEnum {tag, _}) =
|
|
|
|
|
isRedexE defs ctx sg tag || isTagHead tag
|
|
|
|
|
isRedexE defs ctx sg (CaseNat {nat, _}) =
|
|
|
|
|
isRedexE defs ctx sg nat || isNatHead nat
|
|
|
|
|
isRedexE defs ctx sg (CaseBox {box, _}) =
|
|
|
|
|
isRedexE defs ctx sg box || isBoxHead box
|
|
|
|
|
isRedexE defs ctx sg (DApp {fun, arg, _}) =
|
|
|
|
|
isRedexE defs ctx sg fun || isDLamHead fun || isK arg
|
|
|
|
|
isRedexE defs ctx sg (Ann {tm, ty, _}) =
|
|
|
|
|
isE tm || isRedexT defs ctx sg tm || isRedexT defs ctx SZero ty
|
|
|
|
|
isRedexE defs ctx sg (Coe {ty = S _ (N _), _}) = True
|
|
|
|
|
isRedexE defs ctx sg (Coe {ty = S [< i] (Y ty), p, q, val, _}) =
|
|
|
|
|
isRedexT defs (extendDim i ctx) SZero ty ||
|
|
|
|
|
canPushCoe sg ty val || isYes (p `decEqv` q)
|
|
|
|
|
isRedexE defs ctx sg (Comp {ty, p, q, r, _}) =
|
2023-08-26 15:00:19 -04:00
|
|
|
|
isYes (p `decEqv` q) || isK r
|
2023-12-06 19:35:39 -05:00
|
|
|
|
isRedexE defs ctx sg (TypeCase {ty, ret, _}) =
|
|
|
|
|
isRedexE defs ctx sg ty || isRedexT defs ctx sg ret || isAnnTyCon ty
|
|
|
|
|
isRedexE _ _ _ (CloE {}) = True
|
|
|
|
|
isRedexE _ _ _ (DCloE {}) = True
|
2023-08-24 12:42:26 -04:00
|
|
|
|
|
|
|
|
|
||| a reducible term
|
|
|
|
|
|||
|
2023-08-26 15:00:19 -04:00
|
|
|
|
||| 1. a reducible elimination, as `isRedexE`
|
|
|
|
|
||| 2. an annotated elimination
|
|
|
|
|
||| (the annotation is redundant in a checkable context)
|
|
|
|
|
||| 3. a closure
|
2023-11-02 15:01:34 -04:00
|
|
|
|
||| 4. `succ` applied to a natural constant
|
2023-12-04 16:47:52 -05:00
|
|
|
|
||| 5. a `let` expression
|
2023-08-24 12:42:26 -04:00
|
|
|
|
public export
|
|
|
|
|
isRedexT : RedexTest Term
|
2023-12-06 19:35:39 -05:00
|
|
|
|
isRedexT _ _ _ (CloT {}) = True
|
|
|
|
|
isRedexT _ _ _ (DCloT {}) = True
|
|
|
|
|
isRedexT _ _ _ (Let {}) = True
|
|
|
|
|
isRedexT defs ctx sg (E {e, _}) = isAnn e || isRedexE defs ctx sg e
|
|
|
|
|
isRedexT _ _ _ (Succ p {}) = isNatConst p
|
|
|
|
|
isRedexT _ _ _ _ = False
|