Portability | non-portable (GHC Extensions) |
---|---|
Stability | experimental |
Maintainer | Patrick Bahr <paba@diku.dk> |
This module defines the infrastructure necessary to use compositional data types for mutually recursive data types. Examples of usage are provided below.
- module Data.Comp.Multi.Term
- module Data.Comp.Multi.Algebra
- module Data.Comp.Multi.Functor
- module Data.Comp.Multi.Sum
- module Data.Comp.Multi.Product
Examples
Pure Computations
The example below illustrates how to use generalised compositional data types to implement a small expression language, with a sub language of values, and an evaluation function mapping expressions to values.
The following language extensions are
needed in order to run the example: TemplateHaskell
, TypeOperators
,
MultiParamTypeClasses
, FlexibleInstances
, FlexibleContexts
,
UndecidableInstances
, and GADTs
. Moreover, in order to derive instances for
GADTs, version 7 of GHC is needed.
import Data.Comp.Multi import Data.Comp.Multi.Show () import Data.Comp.Derive -- Signature for values and operators data Value e l where Const :: Int -> Value e Int Pair :: e s -> e t -> Value e (s,t) data Op e l where Add, Mult :: e Int -> e Int -> Op e Int Fst :: e (s,t) -> Op e s Snd :: e (s,t) -> Op e t -- Signature for the simple expression language type Sig = Op :+: Value -- Derive boilerplate code using Template Haskell (GHC 7 needed) $(derive [instanceHFunctor, instanceHShowF, smartHConstructors] [''Value, ''Op]) -- Term evaluation algebra class Eval f v where evalAlg :: Alg f (HTerm v) instance (Eval f v, Eval g v) => Eval (f :+: g) v where evalAlg (Inl x) = evalAlg x evalAlg (Inr x) = evalAlg x -- Lift the evaluation algebra to a catamorphism eval :: (HFunctor f, Eval f v) => Term f :-> Term v eval = cata evalAlg instance (Value :<: v) => Eval Value v where evalAlg = inject instance (Value :<: v) => Eval Op v where evalAlg (Add x y) = iConst $ (projC x) + (projC y) evalAlg (Mult x y) = iConst $ (projC x) * (projC y) evalAlg (Fst x) = fst $ projP x evalAlg (Snd x) = snd $ projP x projC :: (Value :<: v) => Term v Int -> Int projC v = case project v of Just (Const n) -> n projP :: (Value :<: v) => Term v (s,t) -> (Term v s, Term v t) projP v = case project v of Just (Pair x y) -> (x,y) -- Example: evalEx = iConst 2 evalEx :: Term Value Int evalEx = eval (iFst $ iPair (iConst 2) (iConst 1) :: Term Sig Int)
Monadic Computations
The example below illustrates how to use generalised compositional data types to implement a small expression language, with a sub language of values, and a monadic evaluation function mapping expressions to values.
The following language
extensions are needed in order to run the example: TemplateHaskell
,
TypeOperators
, MultiParamTypeClasses
, FlexibleInstances
,
FlexibleContexts
, UndecidableInstances
, and GADTs
. Moreover, in order to
derive instances for GADTs, version 7 of GHC is needed.
import Data.Comp.Multi import Data.Comp.Multi.Show () import Data.Comp.Derive import Control.Monad (liftM) -- Signature for values and operators data Value e l where Const :: Int -> Value e Int Pair :: e s -> e t -> Value e (s,t) data Op e l where Add, Mult :: e Int -> e Int -> Op e Int Fst :: e (s,t) -> Op e s Snd :: e (s,t) -> Op e t -- Signature for the simple expression language type Sig = Op :+: Value -- Derive boilerplate code using Template Haskell (GHC 7 needed) $(derive [instanceHFunctor, instanceHTraversable, instanceHFoldable, instanceHEqF, instanceHShowF, smartHConstructors] [''Value, ''Op]) -- Monadic term evaluation algebra class EvalM f v where evalAlgM :: AlgM Maybe f (Term v) instance (EvalM f v, EvalM g v) => EvalM (f :+: g) v where evalAlgM (Inl x) = evalAlgM x evalAlgM (Inr x) = evalAlgM x evalM :: (HTraversable f, EvalM f v) => Term f l -> Maybe (Term v l) evalM = cataM evalAlgM instance (Value :<: v) => EvalM Value v where evalAlgM = return . inject instance (Value :<: v) => EvalM Op v where evalAlgM (Add x y) = do n1 <- projC x n2 <- projC y return $ iConst $ n1 + n2 evalAlgM (Mult x y) = do n1 <- projC x n2 <- projC y return $ iConst $ n1 * n2 evalAlgM (Fst v) = liftM fst $ projP v evalAlgM (Snd v) = liftM snd $ projP v projC :: (Value :<: v) => Term v Int -> Maybe Int projC v = case project v of Just (Const n) -> return n; _ -> Nothing projP :: (Value :<: v) => Term v (a,b) -> Maybe (Term v a, Term v b) projP v = case project v of Just (Pair x y) -> return (x,y); _ -> Nothing -- Example: evalMEx = Just (iConst 5) evalMEx :: Maybe (Term Value Int) evalMEx = evalM ((iConst 1) `iAdd` (iConst 2 `iMult` iConst 2) :: Term Sig Int)
Composing Term Homomorphisms and Algebras
The example below illustrates how to compose a term homomorphism and an algebra, exemplified via a desugaring term homomorphism and an evaluation algebra.
The following language extensions are needed in order to run the example:
TemplateHaskell
, TypeOperators
, MultiParamTypeClasses
,
FlexibleInstances
, FlexibleContexts
, UndecidableInstances
, and GADTs
.
Moreover, in order to derive instances for GADTs, version 7 of GHC is needed.
import Data.Comp.Multi import Data.Comp.Multi.Show () import Data.Comp.Derive -- Signature for values, operators, and syntactic sugar data Value e l where Const :: Int -> Value e Int Pair :: e s -> e t -> Value e (s,t) data Op e l where Add, Mult :: e Int -> e Int -> Op e Int Fst :: e (s,t) -> Op e s Snd :: e (s,t) -> Op e t data Sugar e l where Neg :: e Int -> Sugar e Int Swap :: e (s,t) -> Sugar e (t,s) -- Source position information (line number, column number) data Pos = Pos Int Int deriving Show -- Signature for the simple expression language type Sig = Op :+: Value type SigP = Op :&: Pos :+: Value :&: Pos -- Signature for the simple expression language, extended with syntactic sugar type Sig' = Sugar :+: Op :+: Value type SigP' = Sugar :&: Pos :+: Op :&: Pos :+: Value :&: Pos -- Derive boilerplate code using Template Haskell (GHC 7 needed) $(derive [instanceHFunctor, instanceHTraversable, instanceHFoldable, instanceHEqF, instanceHShowF, smartHConstructors] [''Value, ''Op, ''Sugar]) -- Term homomorphism for desugaring of terms class (HFunctor f, HFunctor g) => Desugar f g where desugHom :: TermHom f g desugHom = desugHom' . hfmap Hole desugHom' :: Alg f (Context g a) desugHom' x = appCxt (desugHom x) instance (Desugar f h, Desugar g h) => Desugar (f :+: g) h where desugHom (Inl x) = desugHom x desugHom (Inr x) = desugHom x desugHom' (Inl x) = desugHom' x desugHom' (Inr x) = desugHom' x instance (Value :<: v, HFunctor v) => Desugar Value v where desugHom = simpCxt . inj instance (Op :<: v, HFunctor v) => Desugar Op v where desugHom = simpCxt . inj instance (Op :<: v, Value :<: v, HFunctor v) => Desugar Sugar v where desugHom' (Neg x) = iConst (-1) `iMult` x desugHom' (Swap x) = iSnd x `iPair` iFst x -- Term evaluation algebra class Eval f v where evalAlg :: Alg f (Term v) instance (Eval f v, Eval g v) => Eval (f :+: g) v where evalAlg (Inl x) = evalAlg x evalAlg (Inr x) = evalAlg x instance (Value :<: v) => Eval Value v where evalAlg = inject instance (Value :<: v) => Eval Op v where evalAlg (Add x y) = iConst $ (projC x) + (projC y) evalAlg (Mult x y) = iConst $ (projC x) * (projC y) evalAlg (Fst x) = fst $ projP x evalAlg (Snd x) = snd $ projP x projC :: (Value :<: v) => Term v Int -> Int projC v = case project v of Just (Const n) -> n projP :: (Value :<: v) => HTerm v (s,t) -> (HTerm v s, HTerm v t) projP v = case project v of Just (Pair x y) -> (x,y) -- Compose the evaluation algebra and the desugaring homomorphism to an -- algebra eval :: Term Sig' :-> Term Value eval = cata (evalAlg `compAlg` (desugHom :: TermHom Sig' Sig)) -- Example: evalEx = iPair (iConst 2) (iConst 1) evalEx :: Term Value (Int,Int) evalEx = eval $ iSwap $ iPair (iConst 1) (iConst 2)
Lifting Term Homomorphisms to Products
The example below illustrates how to lift a term homomorphism to products, exemplified via a desugaring term homomorphism lifted to terms annotated with source position information.
The following language extensions are needed in order to run the example:
TemplateHaskell
, TypeOperators
, MultiParamTypeClasses
,
FlexibleInstances
, FlexibleContexts
, UndecidableInstances
, and GADTs
.
Moreover, in order to derive instances for GADTs, version 7 of GHC is needed.
import Data.Comp.Multi import Data.Comp.Multi.Show () import Data.Comp.Derive -- Signature for values, operators, and syntactic sugar data Value e l where Const :: Int -> Value e Int Pair :: e s -> e t -> Value e (s,t) data Op e l where Add, Mult :: e Int -> e Int -> Op e Int Fst :: e (s,t) -> Op e s Snd :: e (s,t) -> Op e t data Sugar e l where Neg :: e Int -> Sugar e Int Swap :: e (s,t) -> Sugar e (t,s) -- Source position information (line number, column number) data Pos = Pos Int Int deriving Show -- Signature for the simple expression language type Sig = Op :+: Value type SigP = Op :&: Pos :+: Value :&: Pos -- Signature for the simple expression language, extended with syntactic sugar type Sig' = Sugar :+: Op :+: Value type SigP' = Sugar :&: Pos :+: Op :&: Pos :+: Value :&: Pos -- Derive boilerplate code using Template Haskell (GHC 7 needed) $(derive [instanceHFunctor, instanceHTraversable, instanceHFoldable, instanceHEqF, instanceHShowF, smartHConstructors] [''Value, ''Op, ''Sugar]) -- Term homomorphism for desugaring of terms class (HFunctor f, HFunctor g) => Desugar f g where desugHom :: TermHom f g desugHom = desugHom' . hfmap Hole desugHom' :: Alg f (Context g a) desugHom' x = appCxt (desugHom x) instance (Desugar f h, Desugar g h) => Desugar (f :+: g) h where desugHom (Inl x) = desugHom x desugHom (Inr x) = desugHom x desugHom' (Inl x) = desugHom' x desugHom' (Inr x) = desugHom' x instance (Value :<: v, HFunctor v) => Desugar Value v where desugHom = simpCxt . inj instance (Op :<: v, HFunctor v) => Desugar Op v where desugHom = simpCxt . inj instance (Op :<: v, Value :<: v, HFunctor v) => Desugar Sugar v where desugHom' (Neg x) = iConst (-1) `iMult` x desugHom' (Swap x) = iSnd x `iPair` iFst x -- Lift the desugaring term homomorphism to a catamorphism desug :: Term Sig' :-> Term Sig desug = appTermHom desugHom -- Example: desugEx = iPair (iConst 2) (iConst 1) desugEx :: Term Sig (Int,Int) desugEx = desug $ iSwap $ iPair (iConst 1) (iConst 2) -- Lift desugaring to terms annotated with source positions desugP :: Term SigP' :-> Term SigP desugP = appTermHom (productTermHom desugHom) iSwapP :: (DistProd f p f', Sugar :<: f) => p -> Term f' (a,b) -> Term f' (b,a) iSwapP p x = Term (injectP p $ inj $ Swap x) iConstP :: (DistProd f p f', Value :<: f) => p -> Int -> Term f' Int iConstP p x = Term (injectP p $ inj $ Const x) iPairP :: (DistProd f p f', Value :<: f) => p -> Term f' a -> Term f' b -> Term f' (a,b) iPairP p x y = Term (injectP p $ inj $ Pair x y) iFstP :: (DistProd f p f', Op :<: f) => p -> Term f' (a,b) -> Term f' a iFstP p x = Term (injectP p $ inj $ Fst x) iSndP :: (DistProd f p f', Op :<: f) => p -> Term f' (a,b) -> Term f' b iSndP p x = Term (injectP p $ inj $ Snd x) -- Example: desugPEx = iPairP (Pos 1 0) -- (iSndP (Pos 1 0) (iPairP (Pos 1 1) -- (iConstP (Pos 1 2) 1) -- (iConstP (Pos 1 3) 2))) -- (iFstP (Pos 1 0) (iPairP (Pos 1 1) -- (iConstP (Pos 1 2) 1) -- (iConstP (Pos 1 3) 2))) desugPEx :: Term SigP (Int,Int) desugPEx = desugP $ iSwapP (Pos 1 0) (iPairP (Pos 1 1) (iConstP (Pos 1 2) 1) (iConstP (Pos 1 3) 2))
module Data.Comp.Multi.Term
module Data.Comp.Multi.Algebra
module Data.Comp.Multi.Functor
module Data.Comp.Multi.Sum
module Data.Comp.Multi.Product