{-# LANGUAGE DeriveDataTypeable, FlexibleContexts, BangPatterns, StandaloneDeriving #-} {-# LANGUAGE TemplateHaskell, FlexibleInstances, TypeSynonymInstances #-} -- | -- Copyright : (c) 2010-2012 Simon Meier & Benedikt Schmidt -- License : GPL v3 (see LICENSE) -- -- Maintainer : Simon Meier -- Portability : GHC only -- -- Types and operations for handling sorted first-order logic module Theory.Formula ( -- * Formulas Connective(..) , Quantifier(..) , Formula(..) , LNFormula , LFormula , FormulaE , FormulaAC , quantify , openFormula , openFormulaPrefix -- , unquantify -- ** More convenient constructors , lfalse , ltrue , (.&&.) , (.||.) , (.==>.) , (.<=>.) , exists , forall -- ** General Transformations , mapAtoms , foldFormula -- ** Pretty-Printing , prettyFormulaE , prettyFormulaAC -- * Convenience exports , module Theory.Atom ) where import Prelude hiding (negate) import Data.Monoid hiding (All) import Data.Foldable (Foldable, foldMap) import Data.Traversable import Data.Generics import Data.DeriveTH import Data.Binary import Control.DeepSeq import Control.Basics import Control.Monad.Fresh import Theory.Atom import Text.PrettyPrint.Highlight import Term.LTerm import Term.Substitution ------------------------------------------------------------------------------ -- Types ------------------------------------------------------------------------------ -- | Logical connectives. data Connective = And | Or | Imp | Iff deriving( Eq, Ord, Show, Enum, Bounded, Data, Typeable ) -- | Quantifiers. data Quantifier = All | Ex deriving( Eq, Ord, Show, Enum, Bounded, Data, Typeable ) -- | First-order formulas in locally nameless representation with hints for the -- names/sorts of quantified variables. data Formula s c v = Ato (Atom (VTerm c (BVar v))) | TF !Bool | Not (Formula s c v) | Conn !Connective (Formula s c v) (Formula s c v) | Qua !Quantifier s (Formula s c v) -- Folding ---------- -- | Fold a formula. {-# INLINE foldFormula #-} foldFormula :: (Atom (VTerm c (BVar v)) -> b) -> (Bool -> b) -> (b -> b) -> (Connective -> b -> b -> b) -> (Quantifier -> s -> b -> b) -> Formula s c v -> b foldFormula fAto fTF fNot fConn fQua = go where go (Ato a) = fAto a go (TF b) = fTF b go (Not p) = fNot (go p) go (Conn c p q) = fConn c (go p) (go q) go (Qua qua x p) = fQua qua x (go p) -- | Fold a formula. {-# INLINE foldFormulaScope #-} foldFormulaScope :: (Int -> Atom (VTerm c (BVar v)) -> b) -> (Bool -> b) -> (b -> b) -> (Connective -> b -> b -> b) -> (Quantifier -> s -> b -> b) -> Formula s c v -> b foldFormulaScope fAto fTF fNot fConn fQua = go 0 where go !i (Ato a) = fAto i a go _ (TF b) = fTF b go !i (Not p) = fNot (go i p) go !i (Conn c p q) = fConn c (go i p) (go i q) go !i (Qua qua x p) = fQua qua x (go (succ i) p) -- Instances ------------ instance Functor (Formula s c) where fmap f = foldFormula (Ato . fmap (fmap (fmap (fmap f)))) TF Not Conn Qua instance Foldable (Formula s c) where foldMap f = foldFormula (foldMap (foldMap (foldMap (foldMap f)))) mempty id (const mappend) (const $ const id) instance Traversable (Formula a s) where traverse f = foldFormula (liftA Ato . traverse (traverse (traverse (traverse f)))) (pure . TF) (liftA Not) (liftA2 . Conn) ((liftA .) . Qua) -- Abbreviations ---------------- infixl 3 .&&. infixl 2 .||. infixr 1 .==>. infix 1 .<=>. -- | Logically true. ltrue :: Formula a s v ltrue = TF True -- | Logically false. lfalse :: Formula a s v lfalse = TF False (.&&.), (.||.), (.==>.), (.<=>.) :: Formula a s v -> Formula a s v -> Formula a s v (.&&.) = Conn And (.||.) = Conn Or (.==>.) = Conn Imp (.<=>.) = Conn Iff ------------------------------------------------------------------------------ -- Dealing with bound variables ------------------------------------------------------------------------------ -- | @LFormula@ are FOL formulas with sorts abused to denote both a hint for -- the name of the bound variable, as well as the variable's actual sort. type LFormula c = Formula (String, LSort) c LVar type LNFormula = Formula (String, LSort) Name LVar -- | Change the representation of atoms. mapAtoms :: (Int -> Atom (VTerm c (BVar v)) -> Atom (VTerm c1 (BVar v1))) -> Formula s c v -> Formula s c1 v1 mapAtoms f = foldFormulaScope (\i a -> Ato $ f i a) TF Not Conn Qua -- | @openFormula f@ returns @Just (v,Q,f')@ if @f = Q v. f'@ modulo -- alpha renaming and @Nothing otherwise@. @v@ is always chosen to be fresh. openFormula :: (MonadFresh m) => LFormula c -> Maybe (Quantifier, m (LVar, LFormula c)) openFormula (Qua qua (n,s) fm) = Just ( qua , do x <- freshLVar n s return $ (x, mapAtoms (\i a -> fmap (fmap (subst x i)) a) fm) ) where subst x i (Var (Bound i')) | i == i' = Var $ Free x subst _ _ l = l openFormula _ = Nothing -- | @openFormulaPrefix f@ returns @Just (vs,Q,f')@ if @f = Q v_1 .. v_k. f'@ -- modulo alpha renaming and @Nothing otherwise@. @vs@ is always chosen to be -- fresh. openFormulaPrefix :: (MonadFresh m) => LFormula c -> m ([LVar], Quantifier, LFormula c) openFormulaPrefix f0 = case openFormula f0 of Nothing -> error $ "openFormulaPrefix: no outermost quantifier" Just (q, open) -> do (x, f) <- open go q [x] f where go q xs f = case openFormula f of Just (q', open') | q' == q -> do (x', f') <- open' go q (x' : xs) f' -- no further quantifier of the same kind => return result _ -> return (reverse xs, q, f) -- Instances ------------ deriving instance Eq LNFormula deriving instance Show LNFormula deriving instance Ord LNFormula instance HasFrees LNFormula where foldFrees f = foldMap (foldFrees f) mapFrees f = traverse (mapFrees f) instance Apply LNFormula where apply subst = mapAtoms (const $ apply subst) ------------------------------------------------------------------------------ -- Formulas modulo E and modulo AC ------------------------------------------------------------------------------ type FormulaE = LFormula Name type FormulaAC = LFormula Name -- | Introduce a bound variable for a free variable. quantify :: Eq v => v -> Formula s c v -> Formula s c v quantify x = mapAtoms (\i a -> fmap (fmap (fmap (>>= (subst i)))) a) where subst i v | v == x = Bound i | otherwise = Free v -- | Create a universal quantification with a sort hint for the bound variable. forall :: Eq v => s -> v -> Formula s c v -> Formula s c v forall hint x = Qua All hint . quantify x -- | Create a existential quantification with a sort hint for the bound variable. exists :: Eq v => s -> v -> Formula s c v -> Formula s c v exists hint x = Qua Ex hint . quantify x ------------------------------------------------------------------------------ -- Pretty printing ------------------------------------------------------------------------------ -- | Pretty print a formula. prettyLFormula :: (HighlightDocument d, MonadFresh m) => (Atom (VTerm c LVar) -> d) -- ^ Function for pretty printing atoms -> LFormula c -- ^ Formula to pretty print. -> m d -- ^ Pretty printed formula. prettyLFormula ppAtom = pp where extractFree (Free v) = v extractFree (Bound i) = error $ "prettyFormula: illegal bound variable '" ++ show i ++ "'" pp (Ato a) = return $ ppAtom (fmap (fmap (fmap extractFree)) a) pp (TF True) = return $ operator_ "T" -- "⊤" pp (TF False) = return $ operator_ "F" -- "⊥" pp (Not p) = do p' <- pp p return $ operator_ "not" <> opParens p' -- text "¬" <> parens (pp a) pp (Conn op p q) = do p' <- pp p q' <- pp q return $ sep [opParens p' <-> operator_ (ppOp op), opParens q'] where ppOp And = "&" -- "∧" ppOp Or = "|" -- "∨" ppOp Imp = "==>" -- "⇒" ppOp Iff = "<=>" -- "⇔" pp fm@(Qua _ _ _) = do (vs,qua,fm') <- openFormulaPrefix fm d' <- pp fm' return $ sep [ operator_ (ppQuant qua) <> ppVars vs <> operator_ "." , nest 1 d'] where ppVars = fsep . map (text . show) ppQuant All = "All " -- "∀" ppQuant Ex = "Ex " -- "∃" prettyFormulaAC, prettyFormulaE :: HighlightDocument d => FormulaE -> d prettyFormulaE = prettyLNFormula prettyFormulaAC = prettyLNFormula -- | Pretty print a logical formula prettyLNFormula :: HighlightDocument d => LNFormula -> d prettyLNFormula fm = prettyLFormula prettyNAtom fm `evalFreshAvoiding` fm -- Derived instances -------------------- $( derive makeBinary ''Connective) $( derive makeBinary ''Quantifier) $( derive makeBinary ''Formula) $( derive makeNFData ''Connective) $( derive makeNFData ''Quantifier) $( derive makeNFData ''Formula)