{-# LANGUAGE DeriveDataTypeable #-}
module Control.Concurrent.STM.TSem
( TSem
, newTSem
, waitTSem
, signalTSem
, signalTSemN
) where
import Control.Concurrent.STM
import Control.Monad
import Data.Typeable
import Numeric.Natural
newtype TSem = TSem (TVar Integer)
deriving (TSem -> TSem -> Bool
(TSem -> TSem -> Bool) -> (TSem -> TSem -> Bool) -> Eq TSem
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: TSem -> TSem -> Bool
== :: TSem -> TSem -> Bool
$c/= :: TSem -> TSem -> Bool
/= :: TSem -> TSem -> Bool
Eq, Typeable)
newTSem :: Integer -> STM TSem
newTSem :: Integer -> STM TSem
newTSem Integer
i = (TVar Integer -> TSem) -> STM (TVar Integer) -> STM TSem
forall a b. (a -> b) -> STM a -> STM b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap TVar Integer -> TSem
TSem (Integer -> STM (TVar Integer)
forall a. a -> STM (TVar a)
newTVar (Integer -> STM (TVar Integer)) -> Integer -> STM (TVar Integer)
forall a b. (a -> b) -> a -> b
$! Integer
i)
waitTSem :: TSem -> STM ()
waitTSem :: TSem -> STM ()
waitTSem (TSem TVar Integer
t) = do
Integer
i <- TVar Integer -> STM Integer
forall a. TVar a -> STM a
readTVar TVar Integer
t
Bool -> STM () -> STM ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Integer
i Integer -> Integer -> Bool
forall a. Ord a => a -> a -> Bool
<= Integer
0) STM ()
forall a. STM a
retry
TVar Integer -> Integer -> STM ()
forall a. TVar a -> a -> STM ()
writeTVar TVar Integer
t (Integer -> STM ()) -> Integer -> STM ()
forall a b. (a -> b) -> a -> b
$! (Integer
iInteger -> Integer -> Integer
forall a. Num a => a -> a -> a
-Integer
1)
signalTSem :: TSem -> STM ()
signalTSem :: TSem -> STM ()
signalTSem (TSem TVar Integer
t) = do
Integer
i <- TVar Integer -> STM Integer
forall a. TVar a -> STM a
readTVar TVar Integer
t
TVar Integer -> Integer -> STM ()
forall a. TVar a -> a -> STM ()
writeTVar TVar Integer
t (Integer -> STM ()) -> Integer -> STM ()
forall a b. (a -> b) -> a -> b
$! Integer
iInteger -> Integer -> Integer
forall a. Num a => a -> a -> a
+Integer
1
signalTSemN :: Natural -> TSem -> STM ()
signalTSemN :: Natural -> TSem -> STM ()
signalTSemN Natural
0 TSem
_ = () -> STM ()
forall a. a -> STM a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
signalTSemN Natural
1 TSem
s = TSem -> STM ()
signalTSem TSem
s
signalTSemN Natural
n (TSem TVar Integer
t) = do
Integer
i <- TVar Integer -> STM Integer
forall a. TVar a -> STM a
readTVar TVar Integer
t
TVar Integer -> Integer -> STM ()
forall a. TVar a -> a -> STM ()
writeTVar TVar Integer
t (Integer -> STM ()) -> Integer -> STM ()
forall a b. (a -> b) -> a -> b
$! Integer
iInteger -> Integer -> Integer
forall a. Num a => a -> a -> a
+(Natural -> Integer
forall a. Integral a => a -> Integer
toInteger Natural
n)