module ToySolver.Data.BoolExpr
(
BoolExpr (..)
, fold
, simplify
) where
import Control.Applicative
import Control.DeepSeq
import Control.Monad
import Data.Data
import Data.Foldable hiding (fold, concat, any)
import Data.Hashable
import Data.Traversable
import ToySolver.Data.Boolean
import ToySolver.Data.Var
data BoolExpr a
= Atom a
| And [BoolExpr a]
| Or [BoolExpr a]
| Not (BoolExpr a)
| Imply (BoolExpr a) (BoolExpr a)
| Equiv (BoolExpr a) (BoolExpr a)
deriving (Eq, Ord, Show, Read, Typeable, Data)
instance Functor BoolExpr where
fmap = fmapDefault
instance Applicative BoolExpr where
pure = Atom
(<*>) = ap
instance Monad BoolExpr where
return = pure
m >>= f = fold f m
instance Foldable BoolExpr where
foldMap = foldMapDefault
instance Traversable BoolExpr where
traverse f (Atom x) = Atom <$> f x
traverse f (And xs) = And <$> sequenceA (fmap (traverse f) xs)
traverse f (Or xs) = Or <$> sequenceA (fmap (traverse f) xs)
traverse f (Not x) = Not <$> traverse f x
traverse f (Imply x y) = Imply <$> traverse f x <*> traverse f y
traverse f (Equiv x y) = Equiv <$> traverse f x <*> traverse f y
instance NFData a => NFData (BoolExpr a) where
rnf (Atom a) = rnf a
rnf (And xs) = rnf xs
rnf (Or xs) = rnf xs
rnf (Not x) = rnf x
rnf (Imply x y) = rnf x `seq` rnf y
rnf (Equiv x y) = rnf x `seq` rnf y
instance Hashable a => Hashable (BoolExpr a) where
hashWithSalt s (Atom a) = s `hashWithSalt` (0::Int) `hashWithSalt` a
hashWithSalt s (And xs) = s `hashWithSalt` (1::Int) `hashWithSalt` xs
hashWithSalt s (Or xs) = s `hashWithSalt` (2::Int) `hashWithSalt` xs
hashWithSalt s (Not x) = s `hashWithSalt` (3::Int) `hashWithSalt` x
hashWithSalt s (Imply x y) = s `hashWithSalt` (4::Int) `hashWithSalt` x `hashWithSalt` y
hashWithSalt s (Equiv x y) = s `hashWithSalt` (5::Int) `hashWithSalt` x `hashWithSalt` y
instance Complement (BoolExpr a) where
notB = Not
instance MonotoneBoolean (BoolExpr a) where
andB = And
orB = Or
instance Boolean (BoolExpr a) where
(.=>.) = Imply
(.<=>.) = Equiv
instance Variables a => Variables (BoolExpr a) where
vars = foldMap vars
fold :: Boolean b => (atom -> b) -> BoolExpr atom -> b
fold f = g
where
g (Atom a) = f a
g (Or xs) = orB (map g xs)
g (And xs) = andB (map g xs)
g (Not x) = notB (g x)
g (Imply x y) = g x .=>. g y
g (Equiv x y) = g x .<=>. g y
simplify :: BoolExpr a -> BoolExpr a
simplify = runSimplify . fold (Simplify . Atom)
newtype Simplify a = Simplify{ runSimplify :: BoolExpr a }
instance Complement (Simplify a) where
notB (Simplify (Not x)) = Simplify x
notB (Simplify x) = Simplify (Not x)
instance MonotoneBoolean (Simplify a) where
orB xs
| any isTrue ys = Simplify true
| otherwise = Simplify $ Or ys
where
ys = concat [f x | Simplify x <- xs]
f (Or zs) = zs
f z = [z]
andB xs
| any isFalse ys = Simplify false
| otherwise = Simplify $ And ys
where
ys = concat [f x | Simplify x <- xs]
f (And zs) = zs
f z = [z]
instance Boolean (Simplify a) where
Simplify x .=>. Simplify y
| isFalse x = true
| isTrue y = true
| isTrue x = Simplify y
| isFalse y = notB (Simplify x)
| otherwise = Simplify (x .=>. y)
isTrue :: BoolExpr a -> Bool
isTrue (And []) = True
isTrue _ = False
isFalse :: BoolExpr a -> Bool
isFalse (Or []) = True
isFalse _ = False