Fix a bug with being inconsistent with how we make substitutions.

This commit is contained in:
Iavor S. Diatchki 2014-12-16 15:18:49 -08:00
parent 494e6eca0e
commit 53cc08ece5
3 changed files with 48 additions and 22 deletions

View File

@ -26,6 +26,7 @@ import qualified Cryptol.TypeCheck.Solver.Numeric.ImportExport as Num
import qualified Cryptol.TypeCheck.Solver.CrySAT as Num import qualified Cryptol.TypeCheck.Solver.CrySAT as Num
import Cryptol.Utils.Panic(panic) import Cryptol.Utils.Panic(panic)
import Cryptol.Parser.Position(rCombs) import Cryptol.Parser.Position(rCombs)
import Cryptol.Utils.PP(pp)
import Cryptol.TypeCheck.Defaulting(tryDefaultWith) import Cryptol.TypeCheck.Defaulting(tryDefaultWith)
@ -35,6 +36,7 @@ import qualified Data.Map as Map
import Data.Maybe ( fromMaybe ) import Data.Maybe ( fromMaybe )
import Data.Set ( Set ) import Data.Set ( Set )
import qualified Data.Set as Set import qualified Data.Set as Set
import qualified SimpleSMT as SMT
-- Add additional constraints that ensure validity of type function. -- Add additional constraints that ensure validity of type function.
checkTypeFunction :: TFun -> [Type] -> [Prop] checkTypeFunction :: TFun -> [Type] -> [Prop]
@ -69,14 +71,29 @@ proveImplication lnam as ps gs =
Right su -> return su Right su -> return su
Left err -> recordError err >> return emptySubst Left err -> recordError err >> return emptySubst
debugBlock :: Num.Solver -> String -> IO a -> IO a
debugBlock s name m =
do let logger = Num.logger s
debugLog s name
SMT.logTab logger
a <- m
SMT.logUntab logger
return a
debugLog :: Num.Solver -> String -> IO ()
debugLog _ _ = return ()
proveImplication' :: LQName -> [TParam] -> [Prop] -> [Goal] -> proveImplication' :: LQName -> [TParam] -> [Prop] -> [Goal] ->
IO (Either Error Subst) IO (Either Error Subst)
proveImplication' lname as ps gs = proveImplication' lname as ps gs =
Num.withSolver $ \s -> Num.withSolver $ \s ->
do putStrLn "PROVING" debugBlock s "proveImplication'" $
mapM_ print gs do debugBlock s "assumes" $
putStrLn "--------------" mapM_ (debugLog s . show . pp) ps
debugBlock s "shows" $
mapM_ (debugLog s . show . pp . goal) gs
varMap <- Num.assumeProps s ps varMap <- Num.assumeProps s ps
@ -84,22 +101,23 @@ proveImplication' lname as ps gs =
let su = importImps varMap imps let su = importImps varMap imps
gs0 = apSubst su gs gs0 = apSubst su gs
print su debugBlock s "substitution" $ debugLog s (show (pp su))
if not possible if not possible
then return $ Left $ UnusableFunction (thing lname) ps then return $ Left $ UnusableFunction (thing lname) ps
else -- XXX: Use imps else -- XXX: Use imps
do let gs1 = filter ((`notElem` ps) . goal) gs0 do let gs1 = filter ((`notElem` ps) . goal) gs0
gs2 <- simpGoals s gs1 mb <- simpGoals s gs1
print gs2
-- XXX: Minimize the goals invovled in the conflict -- XXX: Minimize the goals invovled in the conflict
let gs3 = fromMaybe (gs1,emptySubst) gs2 let gs3 = fromMaybe (gs1,emptySubst) mb
case gs3 of case gs3 of
([],su1) -> return (Right su1) ([],su1) -> return (Right su1)
-- XXX: Do we need the su? -- XXX: Do we need the su?
(us,_) -> return $ Left (us,su2) ->
do debugBlock s "2nd su:" (debugLog s . show . pp $ su2)
return $ Left
$ UnsolvedDelcayedCt $ UnsolvedDelcayedCt
$ DelayedCt { dctSource = lname $ DelayedCt { dctSource = lname
, dctForall = as , dctForall = as
@ -132,7 +150,10 @@ simpGoals s gs0 =
case mbOk of case mbOk of
Nothing -> return Nothing Nothing -> return Nothing
Just (nonDef,def,imps) -> Just (nonDef,def,imps) ->
do let (su,extraProps) = importSplitImps varMap imps do debugBlock s "simpGoals results (defined)" $
mapM_ (debugLog s . ($ "") . SMT.showsSExpr . Num.smtpLinPart) def
let (su,extraProps) = importSplitImps varMap imps
let def1 = eliminateSimpleGEQ def let def1 = eliminateSimpleGEQ def
toGoal = toGoal =
@ -181,10 +202,7 @@ importSplitImps varMap = mk . partitionEithers . map imp . Map.toList
(Just tv, Just ty) -> (Just tv, Just ty) ->
case tv of case tv of
TVFree {} -> Left (tv,ty) TVFree {} -> Left (tv,ty)
TVBound {} -> TVBound {} -> Right (TVar tv =#= ty)
case ty of
TVar tv2@(TVFree {}) -> Left (tv2, TVar tv)
_ -> Right (TVar tv =#= ty)
_ -> panic "importImps" [ "Failed to import:", show x, show e ] _ -> panic "importImps" [ "Failed to import:", show x, show e ]

View File

@ -4,8 +4,8 @@ module Cryptol.TypeCheck.Solver.CrySAT
( withScope, withSolver ( withScope, withSolver
, assumeProps, checkDefined, simplifyProps , assumeProps, checkDefined, simplifyProps
, check , check
, Solver , Solver, logger
, SMTProp, smtpOther , SMTProp (..)
) where ) where
import qualified Cryptol.TypeCheck.AST as Cry import qualified Cryptol.TypeCheck.AST as Cry

View File

@ -13,6 +13,7 @@ import Cryptol.TypeCheck.Solver.Numeric.AST
import Cryptol.Utils.Panic ( panic ) import Cryptol.Utils.Panic ( panic )
import Control.Monad ( ap, guard ) import Control.Monad ( ap, guard )
import Data.List ( partition )
import Data.Map ( Map ) import Data.Map ( Map )
import qualified Data.Map as Map import qualified Data.Map as Map
import Data.Set ( Set ) import Data.Set ( Set )
@ -144,8 +145,15 @@ smtFinName x = "fin_" ++ show (ppName x)
-- | Given a model, compute a set of equations of the form `x = e`, -- | Given a model, compute a set of equations of the form `x = e`,
-- that are impleied by the model. -- that are impleied by the model.
cryImproveModel :: SMT.Solver -> Set Name -> Map Name Expr -> IO (Map Name Expr) cryImproveModel :: SMT.Solver -> Set Name -> Map Name Expr -> IO (Map Name Expr)
cryImproveModel solver uniVars m = go Map.empty (Map.toList m) cryImproveModel solver uniVars m = go Map.empty initialTodo
-- XXX: Hook in linRel
where where
-- Process unification variables first. That way, if we get `x = y`, we'd
-- prefer `x` to be a unificaiton variabl.e
initialTodo = uncurry (++) $ partition isUniVar $ Map.toList m
isUniVar (x,_) = x `Set.member` uniVars
go done [] = return done go done [] = return done
go done ((x,e) : rest) = go done ((x,e) : rest) =
do yesK <- cryMustEqualK solver x e do yesK <- cryMustEqualK solver x e