{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# OPTIONS_GHC -Wall -Werror #-}
module Data.SBV.Tools.Range (
Boundary(..), Range(..)
, ranges, rangesWith
) where
import Data.SBV
import Data.SBV.Control
import Data.SBV.Internals hiding (Range, free_)
data Boundary a = Unbounded
| Open a
| Closed a
isClosed :: Boundary a -> Bool
isClosed Unbounded = False
isClosed (Open _) = False
isClosed (Closed _) = True
data Range a = Range (Boundary a) (Boundary a)
instance Show a => Show (Range a) where
show (Range l u) = sh True l ++ "," ++ sh False u
where sh onLeft b = case b of
Unbounded | onLeft -> "(-oo"
| True -> "oo)"
Open v | onLeft -> "(" ++ show v
| True -> show v ++ ")"
Closed v | onLeft -> "[" ++ show v
| True -> show v ++ "]"
ranges :: forall a. (Ord a, Num a, SymVal a, SatModel a, Metric a, SymVal (MetricSpace a), SatModel (MetricSpace a)) => (SBV a -> SBool) -> IO [Range a]
ranges = rangesWith defaultSMTCfg
rangesWith :: forall a. (Ord a, Num a, SymVal a, SatModel a, Metric a, SymVal (MetricSpace a), SatModel (MetricSpace a)) => SMTConfig -> (SBV a -> SBool) -> IO [Range a]
rangesWith cfg prop = do mbBounds <- getInitialBounds
case mbBounds of
Nothing -> return []
Just r -> search [r] []
where getInitialBounds :: IO (Maybe (Range a))
getInitialBounds = do
let getGenVal :: GeneralizedCV -> Boundary a
getGenVal (RegularCV cv) = Closed $ getRegVal cv
getGenVal (ExtendedCV ecv) = getExtVal ecv
getExtVal :: ExtCV -> Boundary a
getExtVal (Infinite _) = Unbounded
getExtVal (Epsilon k) = Open $ getRegVal (mkConstCV k (0::Integer))
getExtVal i@Interval{} = error $ unlines [ "*** Data.SBV.ranges.getExtVal: Unexpected interval bounds!"
, "***"
, "*** Found bound: " ++ show i
, "*** Please report this as a bug!"
]
getExtVal (BoundedCV cv) = Closed $ getRegVal cv
getExtVal (AddExtCV a b) = getExtVal a `addBound` getExtVal b
getExtVal (MulExtCV a b) = getExtVal a `mulBound` getExtVal b
opBound :: (a -> a -> a) -> Boundary a -> Boundary a -> Boundary a
opBound f x y = case (fromBound x, fromBound y, isClosed x && isClosed y) of
(Just a, Just b, True) -> Closed $ a `f` b
(Just a, Just b, False) -> Open $ a `f` b
_ -> Unbounded
where fromBound Unbounded = Nothing
fromBound (Open a) = Just a
fromBound (Closed a) = Just a
addBound, mulBound :: Boundary a -> Boundary a -> Boundary a
addBound = opBound (+)
mulBound = opBound (*)
getRegVal :: CV -> a
getRegVal cv = case parseCVs [cv] of
Just (v :: MetricSpace a, []) -> case unliteral (fromMetricSpace (literal v)) of
Nothing -> error $ "Data.SBV.ranges.getRegVal: Cannot extract value from metric space equivalent: " ++ show cv
Just r -> r
_ -> error $ "Data.SBV.ranges.getRegVal: Cannot parse " ++ show cv
getBound cstr = do let objName = "boundValue"
res@(LexicographicResult m) <- optimizeWith cfg Lexicographic $ do x <- free_
constrain $ prop x
cstr objName x
case m of
Unsatisfiable{} -> return Nothing
Unknown{} -> error "Solver said Unknown!"
ProofError{} -> error (show res)
_ -> return $ getModelObjectiveValue objName m
mi <- getBound minimize
ma <- getBound maximize
case (mi, ma) of
(Just minV, Just maxV) -> return $ Just $ Range (getGenVal minV) (getGenVal maxV)
_ -> return Nothing
witness :: Range a -> Symbolic (SBV a)
witness (Range lo hi) = do x :: SBV a <- free_
let restrict v open closed = case v of
Unbounded -> sTrue
Open a -> x `open` literal a
Closed a -> x `closed` literal a
lower = restrict lo (.>) (.>=)
upper = restrict hi (.<) (.<=)
constrain $ lower .&& upper
return x
isFeasible :: Range a -> IO Bool
isFeasible r = runSMTWith cfg $ do _ <- witness r
query $ do cs <- checkSat
case cs of
Unsat -> return False
Unk -> error "Data.SBV.interval.isFeasible: Solver said unknown!"
Sat -> return True
bisect :: Range a -> IO (Maybe [Range a])
bisect r@(Range lo hi) = runSMTWith cfg $ do x <- witness r
constrain $ sNot (prop x)
query $ do cs <- checkSat
case cs of
Unsat -> return Nothing
Unk -> error "Data.SBV.interval.bisect: Solver said unknown!"
Sat -> do midV <- Open <$> getValue x
return $ Just [Range lo midV, Range midV hi]
search :: [Range a] -> [Range a] -> IO [Range a]
search [] sofar = return $ reverse sofar
search (c:cs) sofar = do feasible <- isFeasible c
if feasible
then do mbCS <- bisect c
case mbCS of
Nothing -> search cs (c:sofar)
Just xss -> search (xss ++ cs) sofar
else search cs sofar
{-# ANN rangesWith ("HLint: ignore Replace case with fromMaybe" :: String) #-}