module Language.Hasmtlib.Type.Solver
( WithSolver(..)
, solveWith, interactiveWith
, solveMinimized, solveMinimizedDebug
, solveMaximized, solveMaximizedDebug
)
where
import Language.Hasmtlib.Type.MonadSMT
import Language.Hasmtlib.Internal.Expr
import Language.Hasmtlib.Type.SMTSort
import Language.Hasmtlib.Type.Solution
import Language.Hasmtlib.Type.Pipe
import Language.Hasmtlib.Orderable
import Language.Hasmtlib.Codec
import qualified SMTLIB.Backends as B
import qualified SMTLIB.Backends.Process as P
import Data.Default
import Control.Monad.State
class WithSolver a where
withSolver :: B.Solver -> a
instance WithSolver Pipe where
withSolver :: Solver -> Pipe
withSolver = Int -> Maybe String -> Solver -> Pipe
Pipe Int
0 Maybe String
forall a. Maybe a
Nothing
solveWith :: (Monad m, Default s, Codec a) => Solver s m -> StateT s m a -> m (Result, Maybe (Decoded a))
solveWith :: forall (m :: * -> *) s a.
(Monad m, Default s, Codec a) =>
Solver s m -> StateT s m a -> m (Result, Maybe (Decoded a))
solveWith Solver s m
solver StateT s m a
m = do
(a
a, s
problem) <- StateT s m a -> s -> m (a, s)
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT StateT s m a
m s
forall a. Default a => a
def
(Result
result, Solution
solution) <- Solver s m
solver s
problem
(Result, Maybe (Decoded a)) -> m (Result, Maybe (Decoded a))
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (Result
result, Solution -> a -> Maybe (Decoded a)
forall a. Codec a => Solution -> a -> Maybe (Decoded a)
decode Solution
solution a
a)
interactiveWith :: (MonadIO m, WithSolver s) => (B.Solver, P.Handle) -> StateT s m () -> m ()
interactiveWith :: forall (m :: * -> *) s.
(MonadIO m, WithSolver s) =>
(Solver, Handle) -> StateT s m () -> m ()
interactiveWith (Solver
solver, Handle
handle) StateT s m ()
m = do
((), s)
_ <- StateT s m () -> s -> m ((), s)
forall s (m :: * -> *) a. StateT s m a -> s -> m (a, s)
runStateT StateT s m ()
m (s -> m ((), s)) -> s -> m ((), s)
forall a b. (a -> b) -> a -> b
$ Solver -> s
forall a. WithSolver a => Solver -> a
withSolver Solver
solver
IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ Handle -> IO ()
P.close Handle
handle
solveMinimized :: (MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t, Orderable (Expr t))
=> Expr t
-> m (Result, Solution)
solveMinimized :: forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t,
Orderable (Expr t)) =>
Expr t -> m (Result, Solution)
solveMinimized = Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t) =>
Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized Maybe (Solution -> IO ())
forall a. Maybe a
Nothing Expr t -> Expr t -> Expr 'BoolSort
forall a. Orderable a => a -> a -> Expr 'BoolSort
(<?)
solveMinimizedDebug :: (MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t, Orderable (Expr t))
=> (Solution -> IO ())
-> Expr t
-> m (Result, Solution)
solveMinimizedDebug :: forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t,
Orderable (Expr t)) =>
(Solution -> IO ()) -> Expr t -> m (Result, Solution)
solveMinimizedDebug Solution -> IO ()
debug = Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t) =>
Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized ((Solution -> IO ()) -> Maybe (Solution -> IO ())
forall a. a -> Maybe a
Just Solution -> IO ()
debug) Expr t -> Expr t -> Expr 'BoolSort
forall a. Orderable a => a -> a -> Expr 'BoolSort
(<?)
solveMaximized :: (MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t, Orderable (Expr t))
=> Expr t
-> m (Result, Solution)
solveMaximized :: forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t,
Orderable (Expr t)) =>
Expr t -> m (Result, Solution)
solveMaximized = Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t) =>
Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized Maybe (Solution -> IO ())
forall a. Maybe a
Nothing Expr t -> Expr t -> Expr 'BoolSort
forall a. Orderable a => a -> a -> Expr 'BoolSort
(>?)
solveMaximizedDebug :: (MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t, Orderable (Expr t))
=> (Solution -> IO ())
-> Expr t
-> m (Result, Solution)
solveMaximizedDebug :: forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t,
Orderable (Expr t)) =>
(Solution -> IO ()) -> Expr t -> m (Result, Solution)
solveMaximizedDebug Solution -> IO ()
debug = Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t) =>
Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized ((Solution -> IO ()) -> Maybe (Solution -> IO ())
forall a. a -> Maybe a
Just Solution -> IO ()
debug) Expr t -> Expr t -> Expr 'BoolSort
forall a. Orderable a => a -> a -> Expr 'BoolSort
(<?)
solveOptimized :: (MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t)
=> Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized :: forall (m :: * -> *) (t :: SMTSort).
(MonadIncrSMT Pipe m, MonadIO m, KnownSMTSort t) =>
Maybe (Solution -> IO ())
-> (Expr t -> Expr t -> Expr 'BoolSort)
-> Expr t
-> m (Result, Solution)
solveOptimized Maybe (Solution -> IO ())
mDebug Expr t -> Expr t -> Expr 'BoolSort
op = Result -> Solution -> Expr t -> m (Result, Solution)
go Result
Unknown Solution
forall a. Monoid a => a
mempty
where
go :: Result -> Solution -> Expr t -> m (Result, Solution)
go Result
oldRes Solution
oldSol Expr t
target = do
m ()
forall s (m :: * -> *). MonadIncrSMT s m => m ()
push
(Result
res, Solution
sol) <- m (Result, Solution)
forall s (m :: * -> *).
(MonadIncrSMT s m, MonadIO m) =>
m (Result, Solution)
solve
case Result
res of
Result
Sat -> do
case Solution -> Expr t -> Maybe (Decoded (Expr t))
forall a. Codec a => Solution -> a -> Maybe (Decoded a)
decode Solution
sol Expr t
target of
Maybe (Decoded (Expr t))
Nothing -> (Result, Solution) -> m (Result, Solution)
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (Result
Sat, Solution
forall a. Monoid a => a
mempty)
Just Decoded (Expr t)
targetSol -> do
case Maybe (Solution -> IO ())
mDebug of
Maybe (Solution -> IO ())
Nothing -> () -> m ()
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
Just Solution -> IO ()
debug -> IO () -> m ()
forall a. IO a -> m a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ Solution -> IO ()
debug Solution
sol
Expr 'BoolSort -> m ()
forall s (m :: * -> *). MonadSMT s m => Expr 'BoolSort -> m ()
assert (Expr 'BoolSort -> m ()) -> Expr 'BoolSort -> m ()
forall a b. (a -> b) -> a -> b
$ Expr t
target Expr t -> Expr t -> Expr 'BoolSort
`op` Decoded (Expr t) -> Expr t
forall a. Codec a => Decoded a -> a
encode Decoded (Expr t)
targetSol
Result -> Solution -> Expr t -> m (Result, Solution)
go Result
res Solution
sol Expr t
target
Result
_ -> m ()
forall s (m :: * -> *). MonadIncrSMT s m => m ()
pop m () -> m (Result, Solution) -> m (Result, Solution)
forall a b. m a -> m b -> m b
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> (Result, Solution) -> m (Result, Solution)
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (Result
oldRes, Solution
oldSol)