{-# LANGUAGE CPP #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE EmptyDataDecls #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE RoleAnnotations #-}
{-# LANGUAGE Trustworthy #-}
#if MIN_VERSION_base(4,9,0)
{-# LANGUAGE TypeInType #-}
#endif
module Data.Parameterized.Nonce
(
NonceGenerator
, freshNonce
, Nonce
, indexValue
, newSTNonceGenerator
, newIONonceGenerator
, withIONonceGenerator
, withSTNonceGenerator
, withGlobalSTNonceGenerator
, GlobalNonceGenerator
, globalNonceGenerator
) where
import Control.Monad.ST
import Data.Hashable
import Data.IORef
import Data.STRef
import Data.Typeable
import Data.Word
import Unsafe.Coerce
import System.IO.Unsafe (unsafePerformIO)
import Data.Parameterized.Classes
import Data.Parameterized.Some
#if MIN_VERSION_base(4,9,0)
import Data.Kind
#endif
data NonceGenerator (m :: * -> *) (s :: *) = NonceGenerator {
#if MIN_VERSION_base(4,9,0)
freshNonce :: forall k (tp :: k) . m (Nonce s tp)
#else
freshNonce :: forall (tp :: k) . m (Nonce s tp)
#endif
}
-- | Create a new counter.
withGlobalSTNonceGenerator :: (forall t . NonceGenerator (ST t) t -> ST t r) -> r
withGlobalSTNonceGenerator f = runST $ do
r <- newSTRef (toEnum 0)
f $! NonceGenerator {
freshNonce = do
i <- readSTRef r
writeSTRef r $! succ i
return $! Nonce i
}
-- | Create a new nonce generator in the ST monad.
newSTNonceGenerator :: ST t (Some (NonceGenerator (ST t)))
newSTNonceGenerator = g <$> newSTRef (toEnum 0)
where g r = Some $!
NonceGenerator {
freshNonce = do
i <- readSTRef r
writeSTRef r $! succ i
return $! Nonce i
}
-- | Create a new nonce generator in the ST monad.
newIONonceGenerator :: IO (Some (NonceGenerator IO))
newIONonceGenerator = g <$> newIORef (toEnum 0)
where g r = Some $!
NonceGenerator {
freshNonce = do
i <- readIORef r
writeIORef r $! succ i
return $! Nonce i
}
-- | Run a ST computation with a new nonce generator in the ST monad.
withSTNonceGenerator :: (forall s . NonceGenerator (ST t) s -> (ST t) r) -> ST t r
withSTNonceGenerator f = do
Some r <- newSTNonceGenerator
f r
-- | Create a new nonce generator in the IO monad.
withIONonceGenerator :: (forall s . NonceGenerator IO s -> IO r) -> IO r
withIONonceGenerator f = do
Some r <- newIONonceGenerator
f r
-- | An index generated by the counter.
newtype Nonce (s :: *) (tp :: k) = Nonce { indexValue :: Word64 }
deriving (Eq, Ord, Hashable, Show)
-- Force the type role of Nonce to be nominal: this prevents Data.Coerce.coerce
-- from casting the types of nonces, which it would otherwise be able to do
-- because tp is a phantom type parameter. This partially helps to protect
-- the nonce abstraction.
type role Nonce nominal nominal
instance TestEquality (Nonce s) where
testEquality x y | indexValue x == indexValue y = unsafeCoerce (Just Refl)
| otherwise = Nothing
instance OrdF (Nonce s) where
compareF x y =
case compare (indexValue x) (indexValue y) of
LT -> LTF
EQ -> unsafeCoerce EQF
GT -> GTF
instance HashableF (Nonce s) where
hashWithSaltF s (Nonce x) = hashWithSalt s x
instance ShowF (Nonce s)
------------------------------------------------------------------------
-- GlobalNonceGenerator
data GlobalNonceGenerator
globalNonceIORef :: IORef Word64
globalNonceIORef = unsafePerformIO (newIORef 0)
{-# NOINLINE globalNonceIORef #-}
-- | A nonce generator that uses a globally-defined counter.
globalNonceGenerator :: NonceGenerator IO GlobalNonceGenerator
globalNonceGenerator =
NonceGenerator
{ freshNonce = Nonce <$> atomicModifyIORef' globalNonceIORef (\n -> (n+1, n))
}