2019-04-06 15:53:59 +03:00
|
|
|
module TTImp.ProcessDef
|
|
|
|
|
2019-04-20 18:54:09 +03:00
|
|
|
import Core.CaseBuilder
|
|
|
|
import Core.CaseTree
|
2019-04-06 15:53:59 +03:00
|
|
|
import Core.Context
|
|
|
|
import Core.Core
|
2019-06-02 22:03:41 +03:00
|
|
|
import Core.Coverage
|
2019-04-06 15:53:59 +03:00
|
|
|
import Core.Env
|
2019-05-31 13:42:11 +03:00
|
|
|
import Core.Hash
|
2019-05-31 07:52:54 +03:00
|
|
|
import Core.Metadata
|
2019-04-06 15:53:59 +03:00
|
|
|
import Core.Normalise
|
2019-06-05 17:59:35 +03:00
|
|
|
import Core.Termination
|
2019-04-19 18:35:06 +03:00
|
|
|
import Core.Value
|
2019-04-06 15:53:59 +03:00
|
|
|
import Core.UnifyState
|
|
|
|
|
2019-05-27 01:51:20 +03:00
|
|
|
import TTImp.BindImplicits
|
2019-04-06 15:53:59 +03:00
|
|
|
import TTImp.Elab
|
|
|
|
import TTImp.Elab.Check
|
|
|
|
import TTImp.TTImp
|
2019-06-03 01:43:21 +03:00
|
|
|
import TTImp.Unelab
|
2019-05-29 13:57:07 +03:00
|
|
|
import TTImp.Utils
|
2019-04-06 15:53:59 +03:00
|
|
|
|
2019-06-03 01:43:21 +03:00
|
|
|
import Data.NameMap
|
|
|
|
|
2019-06-01 21:05:02 +03:00
|
|
|
mutual
|
|
|
|
mismatchNF : Defs -> NF vars -> NF vars -> Core Bool
|
2019-06-05 19:28:55 +03:00
|
|
|
mismatchNF defs (NTCon _ xn xt _ xargs) (NTCon _ yn yt _ yargs)
|
|
|
|
= if xn /= yn
|
2019-06-01 21:05:02 +03:00
|
|
|
then pure True
|
2019-06-13 21:33:10 +03:00
|
|
|
else anyM (mismatch defs) (zip xargs yargs)
|
2019-06-01 21:05:02 +03:00
|
|
|
mismatchNF defs (NDCon _ _ xt _ xargs) (NDCon _ _ yt _ yargs)
|
|
|
|
= if xt /= yt
|
|
|
|
then pure True
|
2019-06-13 21:33:10 +03:00
|
|
|
else anyM (mismatch defs) (zip xargs yargs)
|
2019-06-01 21:05:02 +03:00
|
|
|
mismatchNF defs (NPrimVal _ xc) (NPrimVal _ yc) = pure (xc /= yc)
|
|
|
|
mismatchNF defs (NDelayed _ _ x) (NDelayed _ _ y) = mismatchNF defs x y
|
|
|
|
mismatchNF defs (NDelay _ _ _ x) (NDelay _ _ _ y)
|
|
|
|
= mismatchNF defs !(evalClosure defs x) !(evalClosure defs y)
|
|
|
|
mismatchNF _ _ _ = pure False
|
|
|
|
|
|
|
|
mismatch : Defs -> (Closure vars, Closure vars) -> Core Bool
|
|
|
|
mismatch defs (x, y)
|
|
|
|
= mismatchNF defs !(evalClosure defs x) !(evalClosure defs y)
|
|
|
|
|
|
|
|
-- If the terms have the same type constructor at the head, and one of
|
|
|
|
-- the argument positions has different constructors at its head, then this
|
|
|
|
-- is an impossible case, so return True
|
|
|
|
export
|
|
|
|
impossibleOK : Defs -> NF vars -> NF vars -> Core Bool
|
2019-06-05 19:28:55 +03:00
|
|
|
impossibleOK defs (NTCon _ xn xt xa xargs) (NTCon _ yn yt ya yargs)
|
|
|
|
= if xn == yn
|
2019-06-13 21:33:10 +03:00
|
|
|
then anyM (mismatch defs) (zip xargs yargs)
|
2019-06-05 19:28:55 +03:00
|
|
|
else pure False
|
|
|
|
-- If it's a data constructor, any mismatch will do
|
|
|
|
impossibleOK defs (NDCon _ _ xt _ xargs) (NDCon _ _ yt _ yargs)
|
|
|
|
= if xt /= yt
|
|
|
|
then pure True
|
2019-06-13 21:33:10 +03:00
|
|
|
else anyM (mismatch defs) (zip xargs yargs)
|
2019-06-05 19:28:55 +03:00
|
|
|
impossibleOK defs x y = pure False
|
2019-06-01 21:05:02 +03:00
|
|
|
|
2019-04-19 18:06:26 +03:00
|
|
|
-- Given a type checked LHS and its type, return the environment in which we
|
|
|
|
-- should check the RHS, the LHS and its type in that environment,
|
|
|
|
-- and a function which turns a checked RHS into a
|
|
|
|
-- pattern clause
|
2019-05-17 15:52:09 +03:00
|
|
|
-- The 'SubVars' proof contains a proof that refers to the *inner* environment,
|
|
|
|
-- so all the outer things are marked as 'DropCons'
|
|
|
|
extendEnv : Env Term vars -> SubVars inner vars ->
|
|
|
|
NestedNames vars ->
|
2019-04-19 18:06:26 +03:00
|
|
|
Term vars -> Term vars ->
|
2019-05-17 15:52:09 +03:00
|
|
|
Core (vars' **
|
|
|
|
(SubVars inner vars',
|
|
|
|
Env Term vars', NestedNames vars',
|
|
|
|
Term vars', Term vars'))
|
|
|
|
extendEnv env p nest (Bind _ n (PVar c tmty) sc) (Bind _ n' (PVTy _ _) tysc) with (nameEq n n')
|
|
|
|
extendEnv env p nest (Bind _ n (PVar c tmty) sc) (Bind _ n' (PVTy _ _) tysc) | Nothing
|
2019-04-19 18:06:26 +03:00
|
|
|
= throw (InternalError "Can't happen: names don't match in pattern type")
|
2019-05-17 15:52:09 +03:00
|
|
|
extendEnv env p nest (Bind _ n (PVar c tmty) sc) (Bind _ n (PVTy _ _) tysc) | (Just Refl)
|
|
|
|
= extendEnv (PVar c tmty :: env) (DropCons p) (weaken nest) sc tysc
|
|
|
|
extendEnv env p nest (Bind _ n (PLet c tmval tmty) sc) (Bind _ n' (PLet _ _ _) tysc) with (nameEq n n')
|
|
|
|
extendEnv env p nest (Bind _ n (PLet c tmval tmty) sc) (Bind _ n' (PLet _ _ _) tysc) | Nothing
|
2019-05-09 18:46:16 +03:00
|
|
|
= throw (InternalError "Can't happen: names don't match in pattern type")
|
|
|
|
-- PLet on the left becomes Let on the right, to give it computational force
|
2019-05-17 15:52:09 +03:00
|
|
|
extendEnv env p nest (Bind _ n (PLet c tmval tmty) sc) (Bind _ n (PLet _ _ _) tysc) | (Just Refl)
|
|
|
|
= extendEnv (Let c tmval tmty :: env) (DropCons p) (weaken nest) sc tysc
|
|
|
|
extendEnv env p nest tm ty
|
|
|
|
= pure (_ ** (p, env, nest, tm, ty))
|
2019-04-19 18:06:26 +03:00
|
|
|
|
2019-04-19 18:35:06 +03:00
|
|
|
-- Find names which are applied to a function in a Rig1/Rig0 position,
|
|
|
|
-- so that we know how they should be bound on the right hand side of the
|
|
|
|
-- pattern.
|
|
|
|
-- 'bound' counts the number of variables locally bound; these are the
|
|
|
|
-- only ones we're checking linearity of (we may be shadowing names if this
|
|
|
|
-- is a local definition, so we need to leave the earlier ones alone)
|
|
|
|
findLinear : {auto c : Ref Ctxt Defs} ->
|
|
|
|
Bool -> Nat -> RigCount -> Term vars ->
|
|
|
|
Core (List (Name, RigCount))
|
|
|
|
findLinear top bound rig (Bind fc n b sc)
|
|
|
|
= findLinear top (S bound) rig sc
|
|
|
|
findLinear top bound rig tm
|
|
|
|
= case getFnArgs tm of
|
|
|
|
(Ref _ _ n, []) => pure []
|
2019-06-13 21:33:10 +03:00
|
|
|
(Ref _ nt n, args)
|
|
|
|
=> do defs <- get Ctxt
|
2019-04-19 18:35:06 +03:00
|
|
|
Just nty <- lookupTyExact n (gamma defs)
|
|
|
|
| Nothing => pure []
|
|
|
|
findLinArg (accessible nt rig) !(nf defs [] nty) args
|
|
|
|
_ => pure []
|
|
|
|
where
|
|
|
|
accessible : NameType -> RigCount -> RigCount
|
|
|
|
accessible Func r = if top then r else Rig0
|
|
|
|
accessible _ r = r
|
|
|
|
|
|
|
|
findLinArg : RigCount -> NF [] -> List (Term vars) ->
|
|
|
|
Core (List (Name, RigCount))
|
|
|
|
findLinArg rig (NBind _ x (Pi c _ _) sc) (Local {name=a} fc _ idx prf :: as)
|
2019-05-03 04:43:47 +03:00
|
|
|
= do defs <- get Ctxt
|
|
|
|
if idx < bound
|
|
|
|
then do sc' <- sc defs (toClosure defaultOpts [] (Ref fc Bound x))
|
|
|
|
pure $ (a, rigMult c rig) ::
|
|
|
|
!(findLinArg rig sc' as)
|
|
|
|
else do sc' <- sc defs (toClosure defaultOpts [] (Ref fc Bound x))
|
|
|
|
findLinArg rig sc' as
|
2019-04-19 18:35:06 +03:00
|
|
|
findLinArg rig (NBind fc x (Pi c _ _) sc) (a :: as)
|
2019-05-03 04:43:47 +03:00
|
|
|
= do defs <- get Ctxt
|
|
|
|
pure $ !(findLinear False bound (rigMult c rig) a) ++
|
|
|
|
!(findLinArg rig !(sc defs (toClosure defaultOpts [] (Ref fc Bound x))) as)
|
2019-04-19 18:35:06 +03:00
|
|
|
findLinArg rig ty (a :: as)
|
|
|
|
= pure $ !(findLinear False bound rig a) ++ !(findLinArg rig ty as)
|
|
|
|
findLinArg _ _ [] = pure []
|
|
|
|
|
|
|
|
setLinear : List (Name, RigCount) -> Term vars -> Term vars
|
|
|
|
setLinear vs (Bind fc x (PVar c ty) sc)
|
|
|
|
= case lookup x vs of
|
|
|
|
Just c' => Bind fc x (PVar c' ty) (setLinear vs sc)
|
|
|
|
_ => Bind fc x (PVar c ty) (setLinear vs sc)
|
|
|
|
setLinear vs (Bind fc x (PVTy c ty) sc)
|
|
|
|
= case lookup x vs of
|
|
|
|
Just c' => Bind fc x (PVTy c' ty) (setLinear vs sc)
|
|
|
|
_ => Bind fc x (PVTy c ty) (setLinear vs sc)
|
|
|
|
setLinear vs tm = tm
|
|
|
|
|
|
|
|
-- Combining multiplicities on LHS:
|
|
|
|
-- Rig1 + Rig1/W not valid, since it means we have repeated use of name
|
|
|
|
-- Rig0 + RigW = RigW
|
|
|
|
-- Rig0 + Rig1 = Rig1
|
|
|
|
combineLinear : FC -> List (Name, RigCount) ->
|
|
|
|
Core (List (Name, RigCount))
|
|
|
|
combineLinear loc [] = pure []
|
|
|
|
combineLinear loc ((n, count) :: cs)
|
|
|
|
= case lookupAll n cs of
|
|
|
|
[] => pure $ (n, count) :: !(combineLinear loc cs)
|
|
|
|
counts => do count' <- combineAll count counts
|
|
|
|
pure $ (n, count') ::
|
|
|
|
!(combineLinear loc (filter notN cs))
|
|
|
|
where
|
|
|
|
notN : (Name, RigCount) -> Bool
|
|
|
|
notN (n', _) = n /= n'
|
|
|
|
|
|
|
|
lookupAll : Name -> List (Name, RigCount) -> List RigCount
|
|
|
|
lookupAll n [] = []
|
|
|
|
lookupAll n ((n', c) :: cs)
|
|
|
|
= if n == n' then c :: lookupAll n cs else lookupAll n cs
|
|
|
|
|
|
|
|
combine : RigCount -> RigCount -> Core RigCount
|
|
|
|
combine Rig1 Rig1 = throw (LinearUsed loc 2 n)
|
|
|
|
combine Rig1 RigW = throw (LinearUsed loc 2 n)
|
|
|
|
combine RigW Rig1 = throw (LinearUsed loc 2 n)
|
|
|
|
combine RigW RigW = pure RigW
|
|
|
|
combine Rig0 c = pure c
|
|
|
|
combine c Rig0 = pure c
|
|
|
|
|
|
|
|
combineAll : RigCount -> List RigCount -> Core RigCount
|
|
|
|
combineAll c [] = pure c
|
|
|
|
combineAll c (c' :: cs)
|
|
|
|
= do newc <- combine c c'
|
|
|
|
combineAll newc cs
|
|
|
|
|
2019-04-18 16:51:04 +03:00
|
|
|
export
|
2019-05-29 13:57:07 +03:00
|
|
|
checkLHS : {vars : _} ->
|
|
|
|
{auto c : Ref Ctxt Defs} ->
|
2019-05-31 07:52:54 +03:00
|
|
|
{auto m : Ref MD Metadata} ->
|
2019-05-29 13:57:07 +03:00
|
|
|
{auto u : Ref UST UState} ->
|
|
|
|
(mult : RigCount) -> (hashit : Bool) ->
|
|
|
|
Int -> List ElabOpt -> NestedNames vars -> Env Term vars ->
|
|
|
|
FC -> RawImp ->
|
|
|
|
Core (vars' ** (SubVars vars vars',
|
|
|
|
Env Term vars', NestedNames vars',
|
|
|
|
Term vars', Term vars'))
|
2019-06-09 13:58:29 +03:00
|
|
|
checkLHS {vars} mult hashit n opts nest env fc lhs_in
|
|
|
|
= do defs <- get Ctxt
|
|
|
|
lhs_raw <- lhsInCurrentNS nest lhs_in
|
2019-05-27 01:51:20 +03:00
|
|
|
autoimp <- isAutoImplicits
|
|
|
|
autoImplicits True
|
2019-06-10 01:12:11 +03:00
|
|
|
(_, lhs_bound) <- bindNames False lhs_raw
|
2019-05-27 01:51:20 +03:00
|
|
|
autoImplicits autoimp
|
2019-06-10 01:12:11 +03:00
|
|
|
lhs <- implicitsAs defs vars lhs_bound
|
2019-05-27 01:51:20 +03:00
|
|
|
|
2019-05-13 02:44:28 +03:00
|
|
|
log 5 $ "Checking " ++ show lhs
|
2019-05-17 20:47:20 +03:00
|
|
|
logEnv 5 "In env" env
|
2019-05-29 13:57:07 +03:00
|
|
|
(lhstm, _, lhstyg) <-
|
|
|
|
wrapError (InLHS fc (Resolved n)) $
|
|
|
|
elabTerm n (InLHS mult) opts nest env
|
2019-04-18 16:51:04 +03:00
|
|
|
(IBindHere fc PATTERN lhs) Nothing
|
2019-05-17 20:47:20 +03:00
|
|
|
logTerm 10 "Checked LHS term" lhstm
|
2019-04-19 18:06:26 +03:00
|
|
|
lhsty <- getTerm lhstyg
|
|
|
|
|
2019-04-19 18:35:06 +03:00
|
|
|
-- Normalise the LHS to get any functions or let bindings evaluated
|
|
|
|
-- (this might be allowed, e.g. for 'fromInteger')
|
|
|
|
defs <- get Ctxt
|
2019-05-18 13:47:57 +03:00
|
|
|
lhstm <- normaliseLHS defs (letToLam env) lhstm
|
2019-04-19 18:35:06 +03:00
|
|
|
lhsty <- normaliseHoles defs env lhsty
|
|
|
|
linvars_in <- findLinear True 0 Rig1 lhstm
|
|
|
|
log 5 $ "Linearity of names in " ++ show n ++ ": " ++
|
|
|
|
show linvars_in
|
|
|
|
|
|
|
|
linvars <- combineLinear fc linvars_in
|
|
|
|
let lhstm_lin = setLinear linvars lhstm
|
|
|
|
let lhsty_lin = setLinear linvars lhsty
|
|
|
|
|
2019-04-21 00:00:58 +03:00
|
|
|
logTerm 5 "LHS term" lhstm_lin
|
|
|
|
logTerm 5 "LHS type" lhsty_lin
|
2019-05-31 13:42:11 +03:00
|
|
|
setHoleLHS (bindEnv fc env lhstm_lin)
|
2019-04-19 18:06:26 +03:00
|
|
|
|
2019-05-29 13:57:07 +03:00
|
|
|
extendEnv env SubRefl nest lhstm_lin lhsty_lin
|
|
|
|
|
|
|
|
bindNotReq : FC -> Int -> Env Term vs -> (sub : SubVars pre vs) ->
|
|
|
|
Term vs -> Term pre
|
|
|
|
bindNotReq fc i [] SubRefl tm = embed tm
|
|
|
|
bindNotReq fc i (b :: env) SubRefl tm
|
|
|
|
= let tmptm = subst (Ref fc Bound (MN "arg" i)) tm
|
|
|
|
btm = bindNotReq fc (1 + i) env SubRefl tmptm in
|
|
|
|
refToLocal (MN "arg" i) _ btm
|
|
|
|
bindNotReq fc i (b :: env) (KeepCons p) tm
|
|
|
|
= let tmptm = subst (Ref fc Bound (MN "arg" i)) tm
|
|
|
|
btm = bindNotReq fc (1 + i) env p tmptm in
|
|
|
|
refToLocal (MN "arg" i) _ btm
|
|
|
|
bindNotReq fc i (b :: env) (DropCons p) tm
|
|
|
|
= bindNotReq fc i env p
|
|
|
|
(Bind fc _ (Pi (multiplicity b) Explicit (binderType b)) tm)
|
|
|
|
|
|
|
|
bindReq : FC -> Env Term vs -> (sub : SubVars pre vs) ->
|
|
|
|
Term pre -> Maybe ClosedTerm
|
|
|
|
bindReq fc env SubRefl tm = pure (bindEnv fc env tm)
|
|
|
|
bindReq fc (b :: env) (KeepCons p) tm
|
|
|
|
= do b' <- shrinkBinder b p
|
|
|
|
bindReq fc env p
|
|
|
|
(Bind fc _ (Pi (multiplicity b) Explicit (binderType b')) tm)
|
|
|
|
bindReq fc (b :: env) (DropCons p) tm = bindReq fc env p tm
|
|
|
|
|
|
|
|
getReq : (vs : List Name) -> SubVars pre vs -> List Name
|
|
|
|
getReq vs SubRefl = vs
|
|
|
|
getReq _ (DropCons p) = getReq _ p
|
|
|
|
getReq (v :: vs) (KeepCons p) = v :: getReq _ p
|
|
|
|
|
|
|
|
getNotReq : (vs : List Name) -> SubVars pre vs -> List Name
|
|
|
|
getNotReq vs SubRefl = []
|
|
|
|
getNotReq (v :: vs) (DropCons p) = v :: getNotReq _ p
|
|
|
|
getNotReq _ (KeepCons p) = getNotReq _ p
|
|
|
|
|
2019-06-02 23:21:07 +03:00
|
|
|
-- Return whether any of the pattern variables are in a trivially empty
|
|
|
|
-- type, where trivally empty means one of:
|
|
|
|
-- * No constructors
|
|
|
|
-- * Every constructor of the family has a return type which conflicts with
|
|
|
|
-- the given constructor's type
|
|
|
|
hasEmptyPat : Defs -> Env Term vars -> Term vars -> Core Bool
|
|
|
|
hasEmptyPat defs env (Bind fc x (PVar c ty) sc)
|
|
|
|
= pure $ !(isEmpty defs !(nf defs env ty))
|
|
|
|
|| !(hasEmptyPat defs (PVar c ty :: env) sc)
|
|
|
|
hasEmptyPat defs env _ = pure False
|
|
|
|
|
2019-05-29 13:57:07 +03:00
|
|
|
-- Check a pattern clause, returning the component of the 'Case' expression it
|
|
|
|
-- represents, or Nothing if it's an impossible clause
|
|
|
|
export
|
|
|
|
checkClause : {vars : _} ->
|
|
|
|
{auto c : Ref Ctxt Defs} ->
|
2019-05-31 07:52:54 +03:00
|
|
|
{auto m : Ref MD Metadata} ->
|
2019-05-29 13:57:07 +03:00
|
|
|
{auto u : Ref UST UState} ->
|
|
|
|
(mult : RigCount) -> (hashit : Bool) ->
|
|
|
|
Int -> List ElabOpt -> NestedNames vars -> Env Term vars ->
|
|
|
|
ImpClause -> Core (Maybe (Clause, Clause))
|
|
|
|
checkClause mult hashit n opts nest env (ImpossibleClause fc lhs)
|
2019-06-02 23:21:07 +03:00
|
|
|
= handleUnify
|
|
|
|
(do lhs_raw <- lhsInCurrentNS nest lhs
|
|
|
|
autoimp <- isAutoImplicits
|
|
|
|
autoImplicits True
|
|
|
|
(_, lhs) <- bindNames False lhs_raw
|
|
|
|
autoImplicits autoimp
|
|
|
|
|
|
|
|
log 5 $ "Checking " ++ show lhs
|
|
|
|
logEnv 5 "In env" env
|
|
|
|
(lhstm, _, lhstyg) <-
|
|
|
|
elabTerm n (InLHS mult) opts nest env
|
|
|
|
(IBindHere fc PATTERN lhs) Nothing
|
|
|
|
defs <- get Ctxt
|
|
|
|
lhs <- normaliseHoles defs env lhstm
|
|
|
|
if !(hasEmptyPat defs env lhs)
|
|
|
|
then pure Nothing
|
|
|
|
else throw (ValidCase fc env (Left lhs)))
|
|
|
|
(\err =>
|
|
|
|
case err of
|
|
|
|
ValidCase _ _ _ => throw err
|
|
|
|
WhenUnifying _ env l r err
|
|
|
|
=> do defs <- get Ctxt
|
|
|
|
logTerm 10 "Impossible" !(normalise defs env l)
|
|
|
|
logTerm 10 " ...and" !(normalise defs env r)
|
|
|
|
if !(impossibleOK defs !(nf defs env l)
|
|
|
|
!(nf defs env r))
|
|
|
|
then pure Nothing
|
|
|
|
else throw (ValidCase fc env (Right err))
|
|
|
|
_ => throw (ValidCase fc env (Right err)))
|
2019-05-29 13:57:07 +03:00
|
|
|
checkClause {vars} mult hashit n opts nest env (PatClause fc lhs_in rhs)
|
|
|
|
= do (vars' ** (sub', env', nest', lhstm', lhsty')) <-
|
|
|
|
checkLHS mult hashit n opts nest env fc lhs_in
|
2019-05-27 01:51:20 +03:00
|
|
|
let rhsMode = case mult of
|
|
|
|
Rig0 => InType
|
|
|
|
_ => InExpr
|
|
|
|
(rhstm, rhserased) <- checkTermSub n rhsMode opts nest' env' env sub' rhs (gnf env' lhsty')
|
2019-05-31 13:42:11 +03:00
|
|
|
clearHoleLHS
|
2019-04-19 18:06:26 +03:00
|
|
|
|
2019-05-11 14:19:39 +03:00
|
|
|
logTerm 5 "RHS term" rhstm
|
2019-05-31 13:42:11 +03:00
|
|
|
when hashit $
|
|
|
|
do addHash lhstm'
|
|
|
|
addHash rhstm
|
|
|
|
|
|
|
|
-- If the rhs is a hole, record the lhs in the metadata because we
|
|
|
|
-- might want to split it interactively
|
|
|
|
case rhstm of
|
|
|
|
Meta _ _ _ _ =>
|
|
|
|
addLHS (getFC lhs_in) (length env) env' lhstm'
|
|
|
|
_ => pure ()
|
|
|
|
|
|
|
|
|
2019-05-19 22:24:14 +03:00
|
|
|
pure (Just (MkClause env' lhstm' rhstm,
|
|
|
|
MkClause env' lhstm' rhserased))
|
2019-05-29 13:57:07 +03:00
|
|
|
checkClause {vars} mult hashit n opts nest env (WithClause fc lhs_in wval_raw cs)
|
|
|
|
= do (vars' ** (sub', env', nest', lhspat, reqty)) <-
|
|
|
|
checkLHS mult hashit n opts nest env fc lhs_in
|
|
|
|
let wmode
|
|
|
|
= case mult of
|
|
|
|
Rig0 => InType -- treat as used in type only
|
|
|
|
_ => InExpr
|
|
|
|
|
|
|
|
(wval, wval_erased, gwvalTy) <- wrapError (InRHS fc (Resolved n)) $
|
|
|
|
elabTermSub n wmode opts nest' env' env sub' wval_raw Nothing
|
2019-05-31 13:42:11 +03:00
|
|
|
clearHoleLHS
|
|
|
|
|
2019-05-29 13:57:07 +03:00
|
|
|
logTerm 5 "With value" wval
|
|
|
|
logTerm 5 "Required type" reqty
|
|
|
|
wvalTy <- getTerm gwvalTy
|
|
|
|
defs <- get Ctxt
|
|
|
|
wval <- normaliseHoles defs env' wval
|
|
|
|
wvalTy <- normaliseHoles defs env' wvalTy
|
|
|
|
|
|
|
|
let (wevars ** withSub) = keepOldEnv sub' (snd (findSubEnv env' wval))
|
|
|
|
logTerm 5 "With value type" wvalTy
|
|
|
|
log 5 $ "Using vars " ++ show wevars
|
|
|
|
|
|
|
|
let Just wval = shrinkTerm wval withSub
|
|
|
|
| Nothing => throw (InternalError "Impossible happened: With abstraction failure #1")
|
|
|
|
let Just wvalTy = shrinkTerm wvalTy withSub
|
|
|
|
| Nothing => throw (InternalError "Impossible happened: With abstraction failure #2")
|
2019-05-31 07:52:54 +03:00
|
|
|
-- Should the env be normalised too? If the following 'impossible'
|
|
|
|
-- error is ever thrown, that might be the cause!
|
2019-05-29 13:57:07 +03:00
|
|
|
let Just wvalEnv = shrinkEnv env' withSub
|
|
|
|
| Nothing => throw (InternalError "Impossible happened: With abstraction failure #3")
|
|
|
|
|
|
|
|
-- Abstracting over 'wval' in the scope of bNotReq in order
|
|
|
|
-- to get the 'magic with' behaviour
|
|
|
|
let wargn = MN "warg" 0
|
|
|
|
let scenv = Pi RigW Explicit wvalTy :: wvalEnv
|
|
|
|
|
|
|
|
wtyScope <- replace defs scenv !(nf defs scenv (weaken wval))
|
|
|
|
(Local fc (Just RigW) _ First)
|
|
|
|
!(nf defs scenv
|
|
|
|
(weaken (bindNotReq fc 0 env' withSub reqty)))
|
|
|
|
let bNotReq = Bind fc wargn (Pi RigW Explicit wvalTy) wtyScope
|
|
|
|
|
|
|
|
let Just wtype = bindReq fc env' withSub bNotReq
|
|
|
|
| Nothing => throw (InternalError "Impossible happened: With abstraction failure #4")
|
|
|
|
|
|
|
|
-- list of argument names - 'Just' means we need to match the name
|
|
|
|
-- in the with clauses to find out what the pattern should be.
|
|
|
|
-- 'Nothing' means it's the with pattern (so wargn)
|
|
|
|
let wargNames
|
|
|
|
= map Just (reverse (getReq _ withSub)) ++
|
|
|
|
Nothing :: reverse (map Just (getNotReq _ withSub))
|
|
|
|
|
|
|
|
logTerm 5 "With function type" wtype
|
|
|
|
log 5 $ "Argument names " ++ show wargNames
|
|
|
|
|
|
|
|
wname <- genWithName n
|
2019-06-01 17:05:04 +03:00
|
|
|
widx <- addDef wname (newDef fc wname mult vars wtype Private None)
|
2019-05-29 13:57:07 +03:00
|
|
|
let rhs_in = apply (IVar fc wname)
|
|
|
|
(map (maybe wval_raw (IVar fc)) wargNames)
|
|
|
|
|
|
|
|
(rhs, rhs_erased) <- wrapError (InRHS fc (Resolved n)) $
|
|
|
|
checkTermSub n wmode opts nest' env' env sub' rhs_in
|
|
|
|
(gnf env' reqty)
|
|
|
|
|
|
|
|
-- Generate new clauses by rewriting the matched arguments
|
|
|
|
cs' <- traverse (mkClauseWith 1 wname wargNames lhs_in) cs
|
|
|
|
|
|
|
|
-- Elaborate the new definition here
|
|
|
|
let wdef = IDef fc wname cs'
|
|
|
|
processDecl [] nest env wdef
|
|
|
|
|
|
|
|
pure (Just (MkClause env' lhspat rhs,
|
|
|
|
MkClause env' lhspat rhs_erased))
|
|
|
|
where
|
|
|
|
-- If it's 'KeepCons/SubRefl' in 'outprf', that means it was in the outer
|
|
|
|
-- environment so we need to keep it in the same place in the 'with'
|
|
|
|
-- function. Hence, turn it to KeepCons whatever
|
|
|
|
keepOldEnv : (outprf : SubVars outer vs) -> SubVars vs' vs ->
|
|
|
|
(vs'' : List Name ** SubVars vs'' vs)
|
|
|
|
keepOldEnv {vs} SubRefl p = (vs ** SubRefl)
|
|
|
|
keepOldEnv {vs} p SubRefl = (vs ** SubRefl)
|
|
|
|
keepOldEnv (DropCons p) (DropCons p')
|
|
|
|
= let (_ ** rest) = keepOldEnv p p' in
|
|
|
|
(_ ** DropCons rest)
|
|
|
|
keepOldEnv (DropCons p) (KeepCons p')
|
|
|
|
= let (_ ** rest) = keepOldEnv p p' in
|
|
|
|
(_ ** KeepCons rest)
|
|
|
|
keepOldEnv (KeepCons p) (DropCons p')
|
|
|
|
= let (_ ** rest) = keepOldEnv p p' in
|
|
|
|
(_ ** KeepCons rest)
|
|
|
|
keepOldEnv (KeepCons p) (KeepCons p')
|
|
|
|
= let (_ ** rest) = keepOldEnv p p' in
|
|
|
|
(_ ** KeepCons rest)
|
|
|
|
|
|
|
|
dropWithArgs : FC -> Nat -> RawImp ->
|
|
|
|
Core (RawImp, List RawImp)
|
|
|
|
dropWithArgs ploc Z tm = pure (tm, [])
|
|
|
|
dropWithArgs ploc (S k) (IApp _ f arg)
|
|
|
|
= do (tm, rest) <- dropWithArgs ploc k f
|
|
|
|
pure (tm, arg :: rest)
|
|
|
|
-- Shouldn't happen if parsed correctly, but there's no guarantee that
|
|
|
|
-- inputs come from parsed source so throw an error.
|
|
|
|
dropWithArgs ploc _ _ = throw (GenericMsg ploc "Badly formed 'with' clause")
|
|
|
|
|
|
|
|
-- Get the arguments for the rewritten pattern clause of a with by looking
|
|
|
|
-- up how the argument names matched
|
|
|
|
getArgMatch : RawImp -> List (String, RawImp) ->
|
|
|
|
Maybe Name -> RawImp
|
|
|
|
getArgMatch warg ms Nothing = warg
|
|
|
|
getArgMatch warg ms (Just (UN n))
|
|
|
|
= case lookup n ms of
|
|
|
|
Nothing => Implicit fc True
|
|
|
|
Just tm => tm
|
|
|
|
getArgMatch warg ms _ = Implicit fc True
|
|
|
|
|
|
|
|
getNewLHS : FC -> (drop : Nat) -> Name -> List (Maybe Name) ->
|
|
|
|
RawImp -> RawImp -> Core RawImp
|
|
|
|
getNewLHS ploc drop wname wargnames lhs_raw patlhs
|
|
|
|
= do (mlhs_raw, wrest) <- dropWithArgs ploc drop patlhs
|
|
|
|
autoimp <- isAutoImplicits
|
|
|
|
autoImplicits True
|
|
|
|
(_, lhs) <- bindNames False lhs_raw
|
|
|
|
(_, mlhs) <- bindNames False mlhs_raw
|
|
|
|
autoImplicits autoimp
|
|
|
|
|
|
|
|
let (warg :: rest) = reverse wrest
|
|
|
|
| _ => throw (GenericMsg ploc "Badly formed 'with' clause")
|
|
|
|
log 10 $ show lhs ++ " against " ++ show mlhs ++
|
|
|
|
" dropping " ++ show (warg :: rest)
|
|
|
|
ms <- getMatch lhs mlhs
|
|
|
|
log 10 $ "Matches: " ++ show ms
|
|
|
|
let newlhs = apply (IVar ploc wname)
|
|
|
|
(map (getArgMatch warg ms) wargnames ++ rest)
|
|
|
|
log 5 $ "New LHS: " ++ show newlhs
|
|
|
|
pure newlhs
|
|
|
|
|
|
|
|
-- Rewrite the clauses in the block to use an updated LHS.
|
|
|
|
-- 'drop' is the number of additional with arguments we expect (i.e.
|
|
|
|
-- the things to drop from the end before matching LHSs)
|
|
|
|
mkClauseWith : (drop : Nat) -> Name -> List (Maybe Name) ->
|
|
|
|
RawImp -> ImpClause ->
|
|
|
|
Core ImpClause
|
|
|
|
mkClauseWith drop wname wargnames lhs (PatClause ploc patlhs rhs)
|
|
|
|
= do newlhs <- getNewLHS ploc drop wname wargnames lhs patlhs
|
|
|
|
pure (PatClause ploc newlhs rhs)
|
|
|
|
mkClauseWith drop wname wargnames lhs (WithClause ploc patlhs rhs ws)
|
|
|
|
= do newlhs <- getNewLHS ploc drop wname wargnames lhs patlhs
|
|
|
|
ws' <- traverse (mkClauseWith (S drop) wname wargnames lhs) ws
|
|
|
|
pure (WithClause ploc newlhs rhs ws')
|
|
|
|
mkClauseWith drop wname wargnames lhs (ImpossibleClause ploc patlhs)
|
|
|
|
= do newlhs <- getNewLHS ploc drop wname wargnames lhs patlhs
|
|
|
|
pure (ImpossibleClause ploc newlhs)
|
|
|
|
|
2019-05-19 22:24:14 +03:00
|
|
|
|
|
|
|
nameListEq : (xs : List Name) -> (ys : List Name) -> Maybe (xs = ys)
|
|
|
|
nameListEq [] [] = Just Refl
|
|
|
|
nameListEq (x :: xs) (y :: ys) with (nameEq x y)
|
|
|
|
nameListEq (x :: xs) (x :: ys) | (Just Refl) with (nameListEq xs ys)
|
|
|
|
nameListEq (x :: xs) (x :: xs) | (Just Refl) | Just Refl= Just Refl
|
|
|
|
nameListEq (x :: xs) (x :: ys) | (Just Refl) | Nothing = Nothing
|
|
|
|
nameListEq (x :: xs) (y :: ys) | Nothing = Nothing
|
|
|
|
nameListEq _ _ = Nothing
|
2019-04-18 16:51:04 +03:00
|
|
|
|
2019-04-20 18:54:09 +03:00
|
|
|
toPats : Clause -> (vs ** (Env Term vs, Term vs, Term vs))
|
|
|
|
toPats (MkClause {vars} env lhs rhs)
|
|
|
|
= (_ ** (env, lhs, rhs))
|
|
|
|
|
2019-04-06 15:53:59 +03:00
|
|
|
export
|
|
|
|
processDef : {auto c : Ref Ctxt Defs} ->
|
2019-05-31 07:52:54 +03:00
|
|
|
{auto m : Ref MD Metadata} ->
|
2019-04-06 15:53:59 +03:00
|
|
|
{auto u : Ref UST UState} ->
|
2019-05-18 21:08:43 +03:00
|
|
|
List ElabOpt -> NestedNames vars -> Env Term vars -> FC ->
|
2019-04-18 16:51:04 +03:00
|
|
|
Name -> List ImpClause -> Core ()
|
2019-06-03 01:43:21 +03:00
|
|
|
processDef opts nest env fc n_in cs_in
|
2019-04-06 15:53:59 +03:00
|
|
|
= do n <- inCurrentNS n_in
|
|
|
|
defs <- get Ctxt
|
|
|
|
Just gdef <- lookupCtxtExact n (gamma defs)
|
|
|
|
| Nothing => throw (NoDeclaration fc n)
|
2019-04-18 16:51:04 +03:00
|
|
|
let None = definition gdef
|
|
|
|
| _ => throw (AlreadyDefined fc n)
|
|
|
|
let ty = type gdef
|
|
|
|
let hashit = visibility gdef == Public
|
|
|
|
let mult = if multiplicity gdef == Rig0
|
|
|
|
then Rig0
|
|
|
|
else Rig1
|
2019-05-11 15:00:12 +03:00
|
|
|
nidx <- resolveName n
|
2019-05-18 21:08:43 +03:00
|
|
|
cs <- traverse (checkClause mult hashit nidx opts nest env) cs_in
|
2019-05-19 22:24:14 +03:00
|
|
|
let pats = map toPats (map fst (mapMaybe id cs))
|
|
|
|
|
|
|
|
(cargs ** tree_ct) <- getPMDef fc CompileTime n ty
|
|
|
|
(map fst (mapMaybe id cs))
|
|
|
|
(rargs ** tree_rt) <- getPMDef fc CompileTime n ty
|
|
|
|
(map snd (mapMaybe id cs))
|
|
|
|
let Just Refl = nameListEq cargs rargs
|
|
|
|
| Nothing => throw (InternalError "WAT")
|
|
|
|
|
2019-04-21 00:00:58 +03:00
|
|
|
log 5 $ "Case tree for " ++ show n ++ ": " ++ show tree_ct
|
2019-06-03 01:43:21 +03:00
|
|
|
addDef n (record { definition = PMDef cargs tree_ct tree_rt pats,
|
|
|
|
refersTo = getRefs tree_ct } gdef)
|
|
|
|
|
2019-06-05 17:59:35 +03:00
|
|
|
sc <- calculateSizeChange fc n
|
|
|
|
setSizeChange fc n sc
|
|
|
|
|
2019-06-03 01:43:21 +03:00
|
|
|
cov <- checkCoverage nidx mult cs tree_ct
|
|
|
|
setCovering fc n cov
|
|
|
|
where
|
|
|
|
simplePat : Term vars -> Bool
|
|
|
|
simplePat (Local _ _ _ _) = True
|
|
|
|
simplePat (Erased _) = True
|
|
|
|
simplePat _ = False
|
|
|
|
|
|
|
|
-- Is the clause returned from 'checkClause' a catch all clause, i.e.
|
|
|
|
-- one where all the arguments are variables? If so, no need to do the
|
|
|
|
-- (potentially expensive) coverage check
|
|
|
|
catchAll : Maybe (Clause, Clause) -> Bool
|
|
|
|
catchAll Nothing = False
|
|
|
|
catchAll (Just (MkClause env lhs _, _))
|
2019-06-13 21:33:10 +03:00
|
|
|
= all simplePat (getArgs lhs)
|
2019-06-03 01:43:21 +03:00
|
|
|
|
|
|
|
-- Return 'Nothing' if the clause is impossible, otherwise return the
|
|
|
|
-- original
|
|
|
|
checkImpossible : Int -> RigCount -> ClosedTerm ->
|
|
|
|
Core (Maybe ClosedTerm)
|
|
|
|
checkImpossible n mult tm
|
|
|
|
= do itm <- unelabNoPatvars [] tm
|
|
|
|
handleUnify
|
|
|
|
(do ctxt <- get Ctxt
|
|
|
|
log 3 $ "Checking for impossibility: " ++ show itm
|
|
|
|
ok <- checkClause mult False n [] (MkNested []) []
|
|
|
|
(ImpossibleClause fc itm)
|
|
|
|
put Ctxt ctxt
|
|
|
|
maybe (pure Nothing) (\chktm => pure (Just tm)) ok)
|
|
|
|
(\err => case err of
|
|
|
|
WhenUnifying _ env l r err
|
|
|
|
=> do defs <- get Ctxt
|
|
|
|
if !(impossibleOK defs !(nf defs env l)
|
|
|
|
!(nf defs env r))
|
|
|
|
then pure Nothing
|
|
|
|
else pure (Just tm)
|
|
|
|
_ => pure (Just tm))
|
|
|
|
|
|
|
|
checkCoverage : Int -> RigCount -> List (Maybe (Clause, Clause)) ->
|
|
|
|
CaseTree vs -> Core Covering
|
|
|
|
checkCoverage n mult cs ctree
|
|
|
|
= do missCase <- if any catchAll cs
|
|
|
|
then do log 3 $ "Catch all case in " ++ show n
|
|
|
|
pure []
|
|
|
|
else getMissing fc (Resolved n) ctree
|
|
|
|
log 3 ("Initially missing in " ++ show n ++ ":\n" ++
|
|
|
|
showSep "\n" (map show missCase))
|
|
|
|
missImp <- traverse (checkImpossible n mult) missCase
|
|
|
|
let miss = mapMaybe id missImp
|
|
|
|
if isNil miss
|
|
|
|
then do [] <- getNonCoveringRefs fc (Resolved n)
|
|
|
|
| ns => pure (NonCoveringCall ns)
|
|
|
|
pure IsCovering
|
|
|
|
else pure (MissingCases miss)
|
|
|
|
|