----------------------------------------------------------------------------- -- | -- Module : Data.Singletons.Deriving.Enum -- Copyright : (C) 2015 Richard Eisenberg -- License : BSD-style (see LICENSE) -- Maintainer : Richard Eisenberg (eir@cis.upenn.edu) -- Stability : experimental -- Portability : non-portable -- -- Implements deriving of Enum instances -- ---------------------------------------------------------------------------- module Data.Singletons.Deriving.Enum ( mkEnumInstance ) where import Language.Haskell.TH.Syntax import Language.Haskell.TH.Ppr import Language.Haskell.TH.Desugar import Data.Singletons.Syntax import Data.Singletons.Util import Data.Singletons.Names import Control.Monad import Data.Maybe -- monadic for failure only mkEnumInstance :: Quasi q => DType -> [DCon] -> q UInstDecl mkEnumInstance ty cons = do when (null cons || any (\(DCon tvbs cxt _ f rty) -> or [ not $ null $ tysOfConFields f , not $ null tvbs , not $ null cxt , isJust rty ]) cons) $ fail ("Can't derive Enum instance for " ++ pprint (typeToTH ty) ++ ".") n <- qNewName "n" let to_enum = UFunction [DClause [DVarPa n] (to_enum_rhs cons [0..])] to_enum_rhs [] _ = DVarE errorName `DAppE` DLitE (StringL "toEnum: bad argument") to_enum_rhs (DCon _ _ name _ _ : rest) (num:nums) = DCaseE (DVarE equalsName `DAppE` DVarE n `DAppE` DLitE (IntegerL num)) [ DMatch (DConPa trueName []) (DConE name) , DMatch (DConPa falseName []) (to_enum_rhs rest nums) ] to_enum_rhs _ _ = error "Internal error: exhausted infinite list in to_enum_rhs" from_enum = UFunction (zipWith (\i con -> DClause [DConPa (extractName con) []] (DLitE (IntegerL i))) [0..] cons) return (InstDecl { id_cxt = [] , id_name = singletonsEnumName -- need to use singletons's Enum class to get the types -- to use Nat instead of Int , id_arg_tys = [ty] , id_meths = [ (singletonsToEnumName, to_enum) , (singletonsFromEnumName, from_enum) ] })