mirror of
https://github.com/edwinb/Idris2-boot.git
synced 2024-12-29 15:52:29 +03:00
Coverage checker improvement
Check that none of the generated missing cases match explicitly given clauses, which might happen if there's some overlaps in the clauses or if there's some matching on implicits. Ignore erased argument positions when checking the match.
This commit is contained in:
parent
946881d56c
commit
102fb7fdb6
@ -308,3 +308,83 @@ getNonCoveringRefs fc n
|
||||
IsCovering => pure False
|
||||
_ => pure True
|
||||
_ => pure False
|
||||
|
||||
-- Does the second term match against the first?
|
||||
-- 'Erased' matches against anything, we assume that's a Rig0 argument that
|
||||
-- we don't care about
|
||||
match : Term vs -> Term vs -> Bool
|
||||
match (Local _ _ i _) _ = True
|
||||
match (Ref _ Bound n) _ = True
|
||||
match (Ref _ _ n) (Ref _ _ n') = n == n'
|
||||
match (App _ f a) (App _ f' a') = match f f' && match a a'
|
||||
match (As _ _ _ p) (As _ _ _ p') = match p p'
|
||||
match (As _ _ _ p) p' = match p p'
|
||||
match (TDelayed _ _ t) (TDelayed _ _ t') = match t t'
|
||||
match (TDelay _ _ _ t) (TDelay _ _ _ t') = match t t'
|
||||
match (TForce _ _ t) (TForce _ _ t') = match t t'
|
||||
match (PrimVal _ c) (PrimVal _ c') = c == c'
|
||||
match (Erased _ _) _ = True
|
||||
match _ (Erased _ _) = True
|
||||
match (TType _) (TType _) = True
|
||||
match _ _ = False
|
||||
|
||||
-- Erase according to argument position
|
||||
eraseApps : {auto c : Ref Ctxt Defs} ->
|
||||
Term vs -> Core (Term vs)
|
||||
eraseApps {vs} tm
|
||||
= case getFnArgs tm of
|
||||
(Ref fc Bound n, args) =>
|
||||
do args' <- traverse eraseApps args
|
||||
pure (apply fc (Ref fc Bound n) args')
|
||||
(Ref fc nt n, args) =>
|
||||
do defs <- get Ctxt
|
||||
mgdef <- lookupCtxtExact n (gamma defs)
|
||||
let eargs = maybe [] eraseArgs mgdef
|
||||
args' <- traverse eraseApps (dropPos fc 0 eargs args)
|
||||
pure (apply fc (Ref fc nt n) args')
|
||||
(tm, args) =>
|
||||
do args' <- traverse eraseApps args
|
||||
pure (apply (getLoc tm) tm args')
|
||||
where
|
||||
dropPos : FC -> Nat -> List Nat -> List (Term vs) -> List (Term vs)
|
||||
dropPos fc i ns [] = []
|
||||
dropPos fc i ns (x :: xs)
|
||||
= if i `elem` ns
|
||||
then Erased fc False :: dropPos fc (S i) ns xs
|
||||
else x :: dropPos fc (S i) ns xs
|
||||
|
||||
-- if tm would be matched by trylhs, then it's not an impossible case
|
||||
-- because we've already got it. Ignore anything in erased position.
|
||||
clauseMatches : {auto c : Ref Ctxt Defs} ->
|
||||
Env Term vars -> Term vars ->
|
||||
ClosedTerm -> Core Bool
|
||||
clauseMatches env tm trylhs
|
||||
= let lhs = !(eraseApps (close (getLoc tm) env tm)) in
|
||||
pure $ match !(toResolvedNames lhs) !(toResolvedNames trylhs)
|
||||
where
|
||||
mkSubstEnv : FC -> Int -> Env Term vars -> SubstEnv vars []
|
||||
mkSubstEnv fc i [] = Nil
|
||||
mkSubstEnv fc i (v :: vs)
|
||||
= Ref fc Bound (MN "cov" i) :: mkSubstEnv fc (i + 1) vs
|
||||
|
||||
close : FC -> Env Term vars -> Term vars -> ClosedTerm
|
||||
close {vars} fc env tm
|
||||
= substs (mkSubstEnv fc 0 env)
|
||||
(rewrite appendNilRightNeutral vars in tm)
|
||||
|
||||
export
|
||||
checkMatched : {auto c : Ref Ctxt Defs} ->
|
||||
List Clause -> ClosedTerm -> Core (Maybe ClosedTerm)
|
||||
checkMatched cs ulhs
|
||||
= tryClauses cs !(eraseApps ulhs)
|
||||
where
|
||||
tryClauses : List Clause -> ClosedTerm -> Core (Maybe ClosedTerm)
|
||||
tryClauses [] ulhs
|
||||
= do logTermNF 10 "Nothing matches" [] ulhs
|
||||
pure $ Just ulhs
|
||||
tryClauses (MkClause env lhs _ :: cs) ulhs
|
||||
= if !(clauseMatches env lhs ulhs)
|
||||
then do logTermNF 10 "Yes" env lhs
|
||||
pure Nothing -- something matches, discared it
|
||||
else do logTermNF 10 "No match" env lhs
|
||||
tryClauses cs ulhs
|
||||
|
@ -505,7 +505,6 @@ checkClause {vars} mult hashit n opts nest env (WithClause fc lhs_in wval_raw cs
|
||||
= do newlhs <- getNewLHS ploc drop nest wname wargnames lhs patlhs
|
||||
pure (ImpossibleClause ploc newlhs)
|
||||
|
||||
|
||||
nameListEq : (xs : List Name) -> (ys : List Name) -> Maybe (xs = ys)
|
||||
nameListEq [] [] = Just Refl
|
||||
nameListEq (x :: xs) (y :: ys) with (nameEq x y)
|
||||
@ -701,11 +700,14 @@ processDef opts nest env fc n_in cs_in
|
||||
pure ("Initially missing in " ++
|
||||
show !(getFullName (Resolved n)) ++ ":\n" ++
|
||||
showSep "\n" (map show mc)))
|
||||
-- Filter out the ones which are impossible
|
||||
missImp <- traverse (checkImpossible n mult) missCase
|
||||
let miss = mapMaybe id missImp
|
||||
-- Filter out the ones which are actually matched (perhaps having
|
||||
-- come up due to some overlapping patterns)
|
||||
missMatch <- traverse (checkMatched covcs) (mapMaybe id missImp)
|
||||
let miss = mapMaybe id missMatch
|
||||
if isNil miss
|
||||
then do [] <- getNonCoveringRefs fc (Resolved n)
|
||||
| ns => toFullNames (NonCoveringCall ns)
|
||||
pure IsCovering
|
||||
else pure (MissingCases miss)
|
||||
|
||||
|
@ -1 +1,10 @@
|
||||
____ __ _ ___
|
||||
/ _/___/ /____(_)____ |__ \
|
||||
/ // __ / ___/ / ___/ __/ / Version 0.0.0-7c8651a70
|
||||
_/ // /_/ / / / (__ ) / __/ https://www.idris-lang.org
|
||||
/___/\__,_/_/ /_/____/ /____/
|
||||
|
||||
Welcome to Idris 2. Enjoy yourself!
|
||||
1/1: Building Vending (Vending.idr)
|
||||
Main> Main.runMachine: All cases covered
|
||||
Main> Bye for now!
|
||||
|
2
tests/idris2/reg008/input
Normal file
2
tests/idris2/reg008/input
Normal file
@ -0,0 +1,2 @@
|
||||
:missing runMachine
|
||||
:q
|
@ -1,3 +1,3 @@
|
||||
$1 --check Vending.idr --debug-elab-check
|
||||
$1 Vending.idr --debug-elab-check < input
|
||||
|
||||
rm -rf build
|
||||
|
Loading…
Reference in New Issue
Block a user