{- Data/Singletons.hs (c) Richard Eisenberg 2013 eir@cis.upenn.edu This is the public interface file to the singletons library. Please see the accompanying README file for more information. Haddock is not currently compatible with the features used here, so the documentation is all in the README file and /Dependently typed programming with singletons/, available at -} {-# LANGUAGE TypeFamilies, GADTs, KindSignatures, TemplateHaskell, DataKinds, PolyKinds, TypeOperators, MultiParamTypeClasses, FlexibleContexts, RankNTypes, UndecidableInstances, FlexibleInstances, ScopedTypeVariables, CPP #-} {-# OPTIONS_GHC -fwarn-incomplete-patterns -fno-warn-unused-binds #-} -- We make unused bindings for (||), (&&), and not. module Data.Singletons ( OfKind(..), Sing(..), SingI(..), SingE(..), SingRep, KindOf, Demote, Any, (:==), (:==:), SingInstance(..), SingKind(singInstance), sTrue, sFalse, SBool, sNothing, sJust, SMaybe, sLeft, sRight, SEither, sTuple0, sTuple2, sTuple3, sTuple4, sTuple5, sTuple6, sTuple7, STuple0, STuple2, STuple3, STuple4, STuple5, STuple6, STuple7, Not, sNot, (:&&), (%:&&), (:||), (%:||), (:&&:), (:||:), (:/=), (:/=:), SEq((%==%), (%/=%)), (%:==), (%:/=), If, sIf, sNil, sCons, SList, (:++), (%:++), Head, Tail, cases, bugInGHC, genSingletons, singletons, genPromotions, promote, promoteEqInstances, promoteEqInstance, singEqInstance, singEqInstances ) where import Prelude hiding ((++)) import Data.Singletons.Singletons import Data.Singletons.Promote import Data.Singletons.Exports import Language.Haskell.TH import Data.Singletons.Util import GHC.Exts (Any) -- provide a few useful singletons... $(genSingletons [''Bool, ''Maybe, ''Either, ''[]]) $(genSingletons [''(), ''(,), ''(,,), ''(,,,), ''(,,,,), ''(,,,,,), ''(,,,,,,)]) -- ... with some functions over Booleans $(singletons [d| not :: Bool -> Bool not False = True not True = False (&&) :: Bool -> Bool -> Bool False && _ = False True && a = a (||) :: Bool -> Bool -> Bool False || a = a True || _ = True |]) type family (a :: k) :==: (b :: k) :: Bool type a :== b = a :==: b -- :== and :==: are synonyms type a :/=: b = Not (a :==: b) type a :/= b = a :/=: b -- the singleton analogue of @Eq@ class (kparam ~ KindParam) => SEq (kparam :: OfKind k) where (%==%) :: forall (a :: k) (b :: k). Sing a -> Sing b -> Sing (a :==: b) (%/=%) :: forall (a :: k) (b :: k). Sing a -> Sing b -> Sing (a :/=: b) a %/=% b = sNot (a %==% b) (%:==) :: forall (a :: k) (b :: k). SEq (KindParam :: OfKind k) => Sing a -> Sing b -> Sing (a :==: b) (%:==) = (%==%) (%:/=) :: forall (a :: k) (b :: k). SEq (KindParam :: OfKind k) => Sing a -> Sing b -> Sing (a :/=: b) (%:/=) = (%/=%) $(singEqInstances [''Bool, ''Maybe, ''Either, ''[]]) $(singEqInstances [''(), ''(,), ''(,,), ''(,,,), ''(,,,,), ''(,,,,,), ''(,,,,,,)]) -- singleton conditional sIf :: Sing a -> Sing b -> Sing c -> Sing (If a b c) sIf STrue b _ = b sIf SFalse _ c = c -- symmetric syntax synonyms type a :&&: b = a :&& b type a :||: b = a :|| b $(singletons [d| (++) :: [a] -> [a] -> [a] [] ++ a = a (h:t) ++ a = h:(t ++ a) |]) -- allows for automatic checking of all constructors in a GADT for instance -- inference cases :: Name -> Q Exp -> Q Exp -> Q Exp cases tyName expq bodyq = do info <- reifyWithWarning tyName case info of TyConI (DataD _ _ _ ctors _) -> buildCases ctors TyConI (NewtypeD _ _ _ ctor _) -> buildCases [ctor] _ -> fail $ "Using <> with something other than a type constructor: " ++ (show tyName) where buildCases :: [Con] -> Q Exp buildCases ctors = caseE expq (map ((flip (flip match (normalB bodyq)) []) . conToPat) ctors) conToPat :: Con -> Q Pat conToPat = ctor1Case (\name tys -> conP name (replicate (length tys) wildP)) -- useful when suppressing GHC's warnings about incomplete pattern matches bugInGHC :: forall a. a bugInGHC = error "Bug encountered in GHC -- this should never happen"