{-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE Trustworthy #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE ViewPatterns #-} {-# OPTIONS -Wall #-} {-| Module: Data.Vector.Unboxed.Deriving Copyright: © Liyang HU 2012 License: BSD3 Maintainer: vector-th-unbox@liyang.hu Stability: experimental Portability: non-portable Writing @Unbox@ instances for new data types is tedious and formulaic. More often than not, there is a straightforward mapping of the new type onto some existing one already imbued with an @Unbox@ instance. The example from the @vector@ package represents @Complex a@ as pairs @(a, a)@. (See .) Using 'derivingUnbox', we can define the same instances much more succinctly: >derivingUnbox "Complex" > [d| instance (Unbox a) => Unbox' (Complex a) (a, a) |] > [| \ (r :+ i) -> (r, i) |] > [| \ (r, i) -> r :+ i |] Requires the @MultiParamTypeClasses@, @TemplateHaskell@ and @TypeFamilies@ @LANGUAGE@ extensions. -} module Data.Vector.Unboxed.Deriving ( Unbox', derivingUnbox ) where import Control.Arrow import Control.Applicative import Control.Monad import qualified Data.Vector.Generic as G import qualified Data.Vector.Generic.Mutable as M import Data.Vector.Unboxed.Base (MVector (..), Vector (..), Unbox) import Language.Haskell.TH -- | A dummy class providing a convenient way to pass in the source and -- representation types, along with any requisite constraints and (implicit) -- type variable introductions. class Unbox' src rep -- Create a @Pat@ bound to the given name and an @Exp@ for said binding. newPatExp :: String -> Q (Pat, Exp) newPatExp = fmap (VarP &&& VarE) . newName -- Create a wrapper for the given function with the same 'nameBase', given -- a list of argument bindings and expressions in terms of said bindings. -- A final coercion (@Exp -> Exp@) is applied to the body of the function. -- Complimentary @INLINE@ pragma included. wrap :: Name -> [(Pat, Exp)] -> (Exp -> Exp) -> [Dec] wrap fun (unzip -> (pats, exps)) coerce = [inline, method] where base = mkName (nameBase fun) inline = PragmaD (InlineP base Inline FunLike AllPhases) body = coerce $ foldl AppE (VarE fun) exps method = FunD base [Clause pats (NormalB body) []] {-| Let's consider a more complex example: suppose we want an @Unbox@ instance for @Maybe a@. We can encode this using the pair @(Bool, a)@, with the boolean indicating whether we have @Nothing@ or @Just@ something. This encoding requires a dummy value in the @Nothing@ case, necessitating an additional @Default@ (see the @data-default@ package) constraint. Thus: >derivingUnbox "Maybe" > [d| instance (Default a, Unbox a) => Unbox' (Maybe a) (Bool, a) |] > [| maybe (False, def) (\ x -> (True, x)) |] > [| \ (b, x) -> if b then Just x else Nothing |] -} derivingUnbox :: String -- ^ Unique constructor suffix for the MVector and Vector data families -> DecsQ -- ^ Quotation of the form @[d| instance /ctxt/ => Unbox' src rep |]@ -> ExpQ -- ^ Quotation of an expression of type @src -> rep@ -> ExpQ -- ^ Quotation of an expression of type @rep -> src@ -> DecsQ -- ^ Declarations to be spliced for the derived Unbox instance derivingUnbox name argsQ toRepQ fromRepQ = do let mvName = mkName ("MV_" ++ name) let vName = mkName ("V_" ++ name) toRep <- toRepQ fromRep <- fromRepQ -- fail unless argsQ quotes a single Unbox' instance [ InstanceD cxts (ConT (nameBase -> "Unbox'") `AppT` typ `AppT` rep) [] ] <- argsQ let liftE e = InfixE (Just e) (VarE 'liftM) . Just let mvCon = ConE mvName let vCon = ConE vName i <- newPatExp "idx" n <- newPatExp "len" a <- second (AppE toRep) <$> newPatExp "val" mv <- first (ConP mvName . (:[])) <$> newPatExp "mvec" mv' <- first (ConP mvName . (:[])) <$> newPatExp "mvec'" v <- first (ConP vName . (:[])) <$> newPatExp "vec" s <- VarT <$> newName "s" let newtypeMVector = NewtypeInstD [] ''MVector [s, typ] (NormalC mvName [(NotStrict, ConT ''MVector `AppT` s `AppT` rep)]) [] let instanceMVector = InstanceD cxts (ConT ''M.MVector `AppT` ConT ''MVector `AppT` typ) $ concat [ wrap 'M.basicLength [mv] id , wrap 'M.basicUnsafeSlice [i, n, mv] (AppE mvCon) , wrap 'M.basicOverlaps [mv, mv'] id , wrap 'M.basicUnsafeNew [n] (liftE mvCon) , wrap 'M.basicUnsafeReplicate [n, a] (liftE mvCon) , wrap 'M.basicUnsafeRead [mv, i] (liftE fromRep) , wrap 'M.basicUnsafeWrite [mv, i, a] id , wrap 'M.basicClear [mv] id , wrap 'M.basicSet [mv, a] id , wrap 'M.basicUnsafeCopy [mv, mv'] id , wrap 'M.basicUnsafeGrow [mv, n] (liftE mvCon) ] let newtypeVector = NewtypeInstD [] ''Vector [typ] (NormalC vName [(NotStrict, ConT ''Vector `AppT` rep)]) [] let instanceVector = InstanceD cxts (ConT ''G.Vector `AppT` ConT ''Vector `AppT` typ) $ concat [ wrap 'G.basicUnsafeFreeze [mv] (liftE vCon) , wrap 'G.basicUnsafeThaw [v] (liftE mvCon) , wrap 'G.basicLength [v] id , wrap 'G.basicUnsafeSlice [i, n, v] (AppE vCon) , wrap 'G.basicUnsafeIndexM [v, i] (liftE fromRep) , wrap 'G.basicUnsafeCopy [mv, v] id , wrap 'G.elemseq [v, a] id ] return [ InstanceD cxts (ConT ''Unbox `AppT` typ) [] , newtypeMVector, instanceMVector , newtypeVector, instanceVector ]