{-# LANGUAGE TemplateHaskell, TypeOperators, MultiParamTypeClasses, FlexibleInstances, FlexibleContexts, UndecidableInstances, OverlappingInstances #-} -------------------------------------------------------------------------------- -- | -- Module : Examples.DesugarPos -- Copyright : (c) 2011 Patrick Bahr, Tom Hvitved -- License : BSD3 -- Maintainer : Tom Hvitved -- Stability : experimental -- Portability : non-portable (GHC Extensions) -- -- Desugaring + Propagation of Annotations -- -- The example illustrates how to lift a term homomorphism to products, -- exemplified via a desugaring term homomorphism lifted to terms annotated with -- source position information. -- -------------------------------------------------------------------------------- module Examples.DesugarPos where import Data.Comp import Data.Comp.Show () import Data.Comp.Equality () import Data.Comp.Derive import Data.Comp.Desugar -- Signature for values, operators, and syntactic sugar data Value e = Const Int | Pair e e data Op e = Add e e | Mult e e | Fst e | Snd e data Sugar e = Neg e | Swap e -- Source position information (line number, column number) data Pos = Pos Int Int deriving (Show, Eq) -- 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 $(derive [makeFunctor, makeTraversable, makeFoldable, makeEqF, makeShowF, smartConstructors, smartAConstructors] [''Value, ''Op, ''Sugar]) instance (Op :<: f, Value :<: f, Functor f) => Desugar Sugar f where desugHom' (Neg x) = iConst (-1) `iMult` x desugHom' (Swap x) = iSnd x `iPair` iFst x -- Example: desugEx = iPair (iConst 2) (iConst 1) desugEx :: Term Sig desugEx = desugar (iSwap $ iPair (iConst 1) (iConst 2) :: Term Sig') -- Lift desugaring to terms annotated with source positions desugP :: Term SigP' -> Term SigP desugP = appHom (propAnn desugHom) -- Example: desugPEx = iAPair (Pos 1 0) -- (iASnd (Pos 1 0) (iAPair (Pos 1 1) -- (iAConst (Pos 1 2) 1) -- (iAConst (Pos 1 3) 2))) -- (iAFst (Pos 1 0) (iAPair (Pos 1 1) -- (iAConst (Pos 1 2) 1) -- (iAConst (Pos 1 3) 2))) desugPEx :: Term SigP desugPEx = desugP $ iASwap (Pos 1 0) (iAPair (Pos 1 1) (iAConst (Pos 1 2) 1) (iAConst (Pos 1 3) 2))