module Control.Concurrent.STM.BTChan
( BTChan
, newBTChan
, newBTChanIO
, writeBTChan
, readBTChan
, tryWriteBTChan
, tryReadBTChan
, unGetBTChan
, isEmptyBTChan
, sizeOfBTChan
, setMaxOfBTChan
, maxOfBTChan
) where
import Control.Concurrent.STM
import Control.Monad (when, liftM)
import Control.Applicative
data BTChan a = BTChan !Int (TChan a) (TVar Int)
newBTChanIO :: Int -> IO (BTChan a)
newBTChanIO m = BTChan m <$> newTChanIO <*> newTVarIO 0
newBTChan :: Int -> STM (BTChan a)
newBTChan m = BTChan m <$> newTChan <*> newTVar 0
writeBTChan :: BTChan a -> a -> STM ()
writeBTChan (BTChan mx c szTV) x = do
sz <- readTVar szTV
when (sz >= mx) retry
writeTVar szTV (sz + 1)
writeTChan c x
tryWriteBTChan :: BTChan a -> a -> STM Bool
tryWriteBTChan (BTChan mx c szTV) x = do
sz <- readTVar szTV
if (sz >= mx)
then return False
else do writeTVar szTV (sz + 1)
writeTChan c x
return True
readBTChan :: BTChan a -> STM a
readBTChan (BTChan _ c szTV) = do
x <- readTChan c
sz <- readTVar szTV
let !sz' = sz 1
writeTVar szTV sz'
return x
tryReadBTChan :: BTChan a -> STM (Maybe a)
tryReadBTChan bt = do
e <- isEmptyBTChan bt
if e then return Nothing else liftM Just (readBTChan bt)
unGetBTChan :: BTChan a -> a -> STM ()
unGetBTChan (BTChan m c sTV) a = do
s <- readTVar sTV
when (s >= m) retry
let !s' = s+1
writeTVar sTV s'
unGetTChan c a
isEmptyBTChan :: BTChan a -> STM Bool
isEmptyBTChan (BTChan _ c _) = isEmptyTChan c
sizeOfBTChan :: BTChan a -> STM Int
sizeOfBTChan (BTChan _ _ sTV) = readTVar sTV
setMaxOfBTChan :: BTChan a -> Int -> BTChan a
setMaxOfBTChan (BTChan _ c s) m = BTChan m c s
maxOfBTChan :: BTChan a -> Int
maxOfBTChan (BTChan m _ _) = m