{-# LANGUAGE CPP         #-}
{-# LANGUAGE GADTs       #-}
#if __GLASGOW_HASKELL__ >= 706
{-# LANGUAGE PolyKinds   #-}
#endif
#if __GLASGOW_HASKELL__ >= 810
{-# LANGUAGE StandaloneKindSignatures #-}
#endif
#if (__GLASGOW_HASKELL__ >= 704 && __GLASGOW_HASKELL__ < 707) || __GLASGOW_HASKELL__ >= 801
{-# LANGUAGE Safe        #-}
#elif __GLASGOW_HASKELL__ >= 702
{-# LANGUAGE Trustworthy #-}
#endif
module Data.GADT.DeepSeq (
    GNFData (..),
    ) where

import Data.Functor.Product (Product (..))
import Data.Functor.Sum     (Sum (..))
import Data.Type.Equality   ((:~:) (..))

#if MIN_VERSION_base(4,10,0)
import qualified Type.Reflection    as TR
#endif

#if __GLASGOW_HASKELL__ >= 810
import Data.Kind (Type, Constraint)
#endif

#if __GLASGOW_HASKELL__ >= 810
type GNFData :: (k -> Type) -> Constraint
#endif

class GNFData f where
    grnf :: f a -> ()

instance (GNFData a, GNFData b) => GNFData (Product a b) where
    grnf :: forall (a :: k). Product a b a -> ()
grnf (Pair a a
a b a
b) = forall k (f :: k -> *) (a :: k). GNFData f => f a -> ()
grnf a a
a seq :: forall a b. a -> b -> b
`seq` forall k (f :: k -> *) (a :: k). GNFData f => f a -> ()
grnf b a
b

instance (GNFData a, GNFData b) => GNFData (Sum a b) where
    grnf :: forall (a :: k). Sum a b a -> ()
grnf (InL a a
x) = forall k (f :: k -> *) (a :: k). GNFData f => f a -> ()
grnf a a
x
    grnf (InR b a
y) = forall k (f :: k -> *) (a :: k). GNFData f => f a -> ()
grnf b a
y

-- | @since 1.0.3
instance GNFData ((:~:) a) where
    grnf :: forall (a :: k). (a :~: a) -> ()
grnf a :~: a
Refl = ()

#if MIN_VERSION_base(4,10,0)
-- | @since 1.0.3
instance GNFData TR.TypeRep where
    grnf :: forall (a :: k). TypeRep a -> ()
grnf = forall k (a :: k). TypeRep a -> ()
TR.rnfTypeRep
#endif