{- Data/Singletons/Type.hs (c) Richard Eisenberg 2013 eir@cis.upenn.edu This file implements promotion of types into kinds. -} module Data.Singletons.Promote.Type ( promoteType ) where import Language.Haskell.TH.Desugar import Data.Singletons.Names import Data.Singletons.Util import Language.Haskell.TH -- the only monadic thing we do here is fail. This allows the function -- to be used from the Singletons module promoteType :: Monad m => DType -> m DKind promoteType = go [] where go :: Monad m => [DKind] -> DType -> m DKind -- We don't need to worry about constraints: they are used to express -- static guarantees at runtime. But, because we don't need to do -- anything special to keep static guarantees at compile time, we don't -- need to promote them. go [] (DForallT _tvbs _cxt ty) = go [] ty go [] (DAppT (DAppT DArrowT (DForallT (_:_) _ _)) _) = fail "Cannot promote types of rank above 1." go args (DAppT t1 t2) = do k2 <- go [] t2 go (k2 : args) t1 go args (DSigT ty _) = go args ty -- just ignore signatures go [] (DVarT name) = return $ DVarK name go _ (DVarT name) = fail $ "Cannot promote an applied type variable " ++ show name ++ "." go [] (DConT name) | name == typeRepName = return DStarK | name == stringName = return $ DConK symbolName [] | nameBase name == nameBase repName = return DStarK go args (DConT name) | Just n <- unboxedTupleNameDegree_maybe name = return $ DConK (tupleTypeName n) args | otherwise = return $ DConK name args go [k1, k2] DArrowT = return $ addStar (DConK tyFunName [k1, k2]) go _ (DLitT _) = fail "Cannot promote a type-level literal" go args hd = fail $ "Illegal Haskell construct encountered:\n" ++ "headed by: " ++ show hd ++ "\n" ++ "applied to: " ++ show args