{-# LANGUAGE QualifiedDo #-}
module Data.Packed.TH.Read (readFName, genRead) where
import Data.Packed.Reader hiding (return)
import qualified Data.Packed.Reader as R
import Data.Packed.TH.Case (caseFName)
import Data.Packed.TH.Flag (PackingFlag (..))
import Data.Packed.TH.Utils
import Data.Packed.Unpackable
import Language.Haskell.TH
readFName :: Name -> Name
readFName :: Name -> Name
readFName Name
tyName = String -> Name
mkName (String -> Name) -> String -> Name
forall a b. (a -> b) -> a -> b
$ String
"read" String -> String -> String
forall a. [a] -> [a] -> [a]
++ Name -> String
sanitizeConName Name
tyName
genRead ::
[PackingFlag] ->
Name ->
Q [Dec]
genRead :: [PackingFlag] -> Name -> Q [Dec]
genRead [PackingFlag]
flags Name
tyName = do
let fName :: Name
fName = Name -> Name
readFName Name
tyName
(resolvedType, typeVariables) <- Name -> Q (Type, [Name])
resolveAppliedType Name
tyName
lambdas <- genReadLambdas flags tyName
funExpr <-
foldl
(\Q Exp
rest Exp
arg -> [|$Q Exp
rest $(Exp -> Q Exp
forall a. a -> Q a
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
arg)|])
(varE $ caseFName tyName)
lambdas
let fun = Name -> [Clause] -> Dec
FunD Name
fName [[Pat] -> Body -> [Dec] -> Clause
Clause [] (Exp -> Body
NormalB Exp
funExpr) []]
signature <- genReadSignature tyName resolvedType typeVariables
return [signature, fun]
genReadLambdas :: [PackingFlag] -> Name -> Q [Exp]
genReadLambdas :: [PackingFlag] -> Name -> Q [Exp]
genReadLambdas [PackingFlag]
flags Name
tyName = do
(TyConI (DataD _ _ _ _ cs _)) <- Name -> Q Info
reify Name
tyName
genReadLambda flags `mapM` cs
genReadLambda :: [PackingFlag] -> Con -> Q Exp
genReadLambda :: [PackingFlag] -> Con -> Q Exp
genReadLambda [PackingFlag]
flags Con
con = do
let appliedConstructor :: Exp
appliedConstructor =
(Exp -> Name -> Exp) -> Exp -> [Name] -> Exp
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl
(\Exp
rest Name
arg -> Exp -> Exp -> Exp
AppE Exp
rest (Exp -> Exp) -> Exp -> Exp
forall a b. (a -> b) -> a -> b
$ Name -> Exp
VarE Name
arg)
(Name -> Exp
ConE Name
conName)
([Name] -> Exp) -> [Name] -> Exp
forall a b. (a -> b) -> a -> b
$ (\Int
i -> String -> Name
mkName (String -> Name) -> String -> Name
forall a b. (a -> b) -> a -> b
$ String
"arg" String -> String -> String
forall a. [a] -> [a] -> [a]
++ Int -> String
forall a. Show a => a -> String
show Int
i)
(Int -> Name) -> [Int] -> [Name]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [Int
0 .. ([BangType] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [BangType]
conParamTypes Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)]
Exp -> Q Exp
buildBindingExpression Exp
appliedConstructor
where
(Name
conName, [BangType]
conParamTypes) = Con -> (Name, [BangType])
getNameAndBangTypesFromCon Con
con
buildBindingExpression :: Exp -> Q Exp
buildBindingExpression :: Exp -> Q Exp
buildBindingExpression Exp
appliedConstructor =
((Type, Int, Bool) -> Q Exp -> Q Exp)
-> Q Exp -> [(Type, Int, Bool)] -> Q Exp
forall a b. (a -> b -> b) -> b -> [a] -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr
( \(Type
_, Int
idx, Bool
needsFS) Q Exp
ret ->
let skipExpr :: Q Exp
skipExpr = [|skip R.>> $Q Exp
readerExpr|]
readerExpr :: Q Exp
readerExpr = [|reader R.>>= \($(Name -> Q Pat
forall (m :: * -> *). Quote m => Name -> m Pat
varP (Name -> Q Pat) -> Name -> Q Pat
forall a b. (a -> b) -> a -> b
$ String -> Name
mkName (String -> Name) -> String -> Name
forall a b. (a -> b) -> a -> b
$ String
"arg" String -> String -> String
forall a. [a] -> [a] -> [a]
++ Int -> String
forall a. Show a => a -> String
show Int
idx)) -> $Q Exp
ret|]
in if Bool
needsFS
then Q Exp
skipExpr
else Q Exp
readerExpr
)
[|R.return ($(Q Exp -> Q Exp
forall (m :: * -> *). Quote m => m Exp -> m Exp
parensE (Exp -> Q Exp
forall a. a -> Q a
forall (m :: * -> *) a. Monad m => a -> m a
return Exp
appliedConstructor)))|]
(Con -> [PackingFlag] -> [(Type, Int, Bool)]
getConFieldsIdxAndNeedsFS Con
con [PackingFlag]
flags)
genReadSignature :: Name -> Type -> [Name] -> Q Dec
genReadSignature :: Name -> Type -> [Name] -> Q Dec
genReadSignature Name
tyName Type
resolvedType [Name]
typeVariables = do
restTypeName <- String -> Q Name
forall (m :: * -> *). Quote m => String -> m Name
newName String
"r"
let readerType = [t|PackedReader '[$(Type -> Q Type
forall a. a -> Q a
forall (m :: * -> *) a. Monad m => a -> m a
return Type
resolvedType)] $(Name -> Q Type
forall (m :: * -> *). Quote m => Name -> m Type
varT Name
restTypeName) ($(Type -> Q Type
forall a. a -> Q a
forall (m :: * -> *) a. Monad m => a -> m a
return Type
resolvedType))|]
constraints = (Name -> Q Type) -> [Name] -> Q Cxt
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
forall (m :: * -> *) a b. Monad m => (a -> m b) -> [a] -> m [b]
mapM (\Name
tyVarName -> [t|Unpackable $(Name -> Q Type
forall (m :: * -> *). Quote m => Name -> m Type
varT Name
tyVarName)|]) [Name]
typeVariables
signature = Q Type
readerType
sigD (readFName tyName) $ forallT [] constraints signature