From ab958599f11c6eb46aa02d0ebb6a0f5fc46c71a6 Mon Sep 17 00:00:00 2001 From: Ana Pantilie Date: Tue, 21 Feb 2023 16:45:41 +0200 Subject: [PATCH] Implement unoptimised version of heuristic --- kore/src/Kore/Internal/Predicate.hs | 6 ++-- kore/src/Kore/Rewrite/SMT/Evaluator.hs | 50 ++++++++++++++++++++------ kore/src/SMT/SimpleSMT.hs | 7 +++- 3 files changed, 49 insertions(+), 14 deletions(-) diff --git a/kore/src/Kore/Internal/Predicate.hs b/kore/src/Kore/Internal/Predicate.hs index ead7225a41..bbfb58df96 100644 --- a/kore/src/Kore/Internal/Predicate.hs +++ b/kore/src/Kore/Internal/Predicate.hs @@ -518,11 +518,11 @@ unparse2WithSort :: Pretty.Doc ann unparse2WithSort sort = unparse2 . fromPredicate sort --- |'PredicateFalse' is a pattern for matching 'bottom' predicates. +-- | 'PredicateFalse' is a pattern for matching 'bottom' predicates. pattern PredicateFalse :: Predicate variable pattern PredicateFalse <- (Recursive.project -> _ :< BottomF _) --- |'PredicateTrue' is a pattern for matching 'top' predicates. +-- | 'PredicateTrue' is a pattern for matching 'top' predicates. pattern PredicateTrue :: Predicate variable pattern PredicateTrue <- (Recursive.project -> _ :< TopF _) @@ -1271,7 +1271,7 @@ mapVariables adj predicate = , Pretty.pretty termPredicate ] --- |Is the predicate free of the given variables? +-- | Is the predicate free of the given variables? isFreeOf :: Ord variable => Predicate variable -> diff --git a/kore/src/Kore/Rewrite/SMT/Evaluator.hs b/kore/src/Kore/Rewrite/SMT/Evaluator.hs index 598204aba3..8292f42a8e 100644 --- a/kore/src/Kore/Rewrite/SMT/Evaluator.hs +++ b/kore/src/Kore/Rewrite/SMT/Evaluator.hs @@ -16,6 +16,7 @@ module Kore.Rewrite.SMT.Evaluator ( import Control.Error ( MaybeT, + hoistMaybe, runMaybeT, ) import Control.Lens qualified as Lens @@ -27,7 +28,6 @@ import Data.Limit import Data.Map.Strict qualified as Map import Data.Set qualified as Set import Data.Text qualified as Text -import Kore.Attribute.Pattern.FreeVariables (freeVariableNames) import Kore.Attribute.Pattern.FreeVariables qualified as FreeVariables import Kore.Attribute.Symbol qualified as Attribute ( Symbol, @@ -73,10 +73,13 @@ import Kore.Log.DecidePredicateUnknown ( ) import Kore.Rewrite.SMT.Translate import Kore.Simplify.Simplify as Simplifier +import Kore.Substitute (substitute) import Kore.TopBottom ( TopBottom, ) import Log +import Logic (LogicT) +import Logic qualified import Prelude.Kore import Pretty ( Pretty, @@ -167,7 +170,22 @@ decidePredicate onUnknown sideCondition predicates = result <- query predicates >>= whenUnknown retry debugEvaluateConditionResult result case result of - Unsat -> return False + Unsat -> do + heuristicResult <- queryWithHeuristic predicates + case heuristicResult of + Unsat -> return False + Sat -> empty + Unknown -> do + limit <- SMT.withSolver SMT.askRetryLimit + -- depending on the value of `onUnknown`, this call will either log a warning + -- or throw an error + throwDecidePredicateUnknown onUnknown limit predicates + case onUnknown of + WarnDecidePredicateUnknown _ _ -> + -- the solver may be in an inconsistent state, so we re-initialize + SMT.reinit + _ -> pure () + empty Sat -> empty Unknown -> do limit <- SMT.withSolver SMT.askRetryLimit @@ -197,15 +215,27 @@ decidePredicate onUnknown sideCondition predicates = traverse_ SMT.assert predicates' SMT.check - applyHeuristic :: Predicate variable -> [Predicate variable] - applyHeuristic p@(Predicate.PredicateNot (Predicate.PredicateExists var child)) = do - -- freeVar <- getFreeVariables child - -- return (substitute freeVar var child) - return p - applyHeuristic pred' = return pred' + queryWithHeuristic :: NonEmpty (Predicate variable) -> MaybeT Simplifier Result + queryWithHeuristic preds = do + results <- + SMT.withSolver . evalTranslator $ do + tools <- Simplifier.askMetadataTools + Morph.hoist SMT.liftSMT . Logic.observeAllT $ do + preds' <- traverse applyHeuristic preds + predicates' <- + traverse + (lift . translatePredicate sideCondition tools) + preds' + traverse_ SMT.assert predicates' + SMT.check + hoistMaybe (find SimpleSMT.isSat results) - generateAllPossibilities :: NonEmpty (Predicate variable) -> [NonEmpty (Predicate variable)] - generateAllPossibilities = traverse applyHeuristic + applyHeuristic :: Predicate variable -> LogicT m (Predicate variable) + applyHeuristic (Predicate.PredicateNot (Predicate.PredicateExists var child)) = do + freeVar <- Logic.scatter $ Predicate.freeElementVariables child + let subst = Map.singleton (inject $ variableName freeVar) (TermLike.mkElemVar var) + return (substitute subst child) + applyHeuristic _ = empty retry :: MaybeT Simplifier Result retry = do diff --git a/kore/src/SMT/SimpleSMT.hs b/kore/src/SMT/SimpleSMT.hs index f5706d466c..8d826b111b 100644 --- a/kore/src/SMT/SimpleSMT.hs +++ b/kore/src/SMT/SimpleSMT.hs @@ -61,6 +61,7 @@ module SMT.SimpleSMT ( assert, check, Result (..), + isSat, getExprs, getExpr, getConsts, @@ -260,6 +261,10 @@ data Result Unknown deriving stock (Eq, Show) +isSat :: Result -> Bool +isSat Sat = True +isSat _ = False + -- | Common values returned by SMT solvers. data Value = -- | Boolean value @@ -747,7 +752,7 @@ sexprToVal expr = | Int a <- sexprToVal x -> Int (negate a) List [Atom "/", x, y] | Int a <- sexprToVal x - , Int b <- sexprToVal y -> + , Int b <- sexprToVal y -> Real (a % b) _ -> Other expr where