{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE CPP #-}
{-# LANGUAGE DoAndIfThenElse #-}
{-# LANGUAGE NoImplicitPrelude #-}
{-# LANGUAGE RecordWildCards #-}
module GHC.Internal.Event.Windows.ManagedThreadPool
( ThreadPool(..)
, startThreadPool
, notifyRunning
, notifyWaiting
, monitorThreadPool
) where
import GHC.Internal.Control.Concurrent.MVar
import GHC.Internal.Data.Maybe
import GHC.Internal.Base
import GHC.Internal.Num ((-), (+))
import GHC.Internal.Real (fromIntegral)
import qualified GHC.Internal.Event.Array as A
import GHC.Internal.IO.Handle.Internals (debugIO)
import GHC.Internal.Conc.Sync (ThreadId(..))
import GHC.Internal.RTS.Flags
type WorkerJob = IO ()
data ThreadPool = ThreadPool
{ ThreadPool -> Maybe ThreadId
thrMainThread :: Maybe ThreadId
, ThreadPool -> Int
thrMaxThreads :: {-# UNPACK #-} !Int
, ThreadPool -> Int
thrMinThreads :: {-# UNPACK #-} !Int
, ThreadPool -> Int
thrCurThreads :: {-# UNPACK #-} !Int
, ThreadPool -> WorkerJob
thrCallBack :: WorkerJob
, ThreadPool -> MVar Int
thrActiveThreads :: MVar Int
, ThreadPool -> MVar ()
thrMonitor :: MVar ()
, ThreadPool -> Array ThreadId
thrThreadIds :: {-#UNPACK #-} !(A.Array ThreadId)
}
startThreadPool :: WorkerJob -> IO ThreadPool
startThreadPool :: WorkerJob -> IO ThreadPool
startThreadPool WorkerJob
job = do
String -> WorkerJob
debugIO String
"Starting I/O manager threadpool..."
let thrMinThreads :: Int
thrMinThreads = Int
2
let thrCurThreads :: Int
thrCurThreads = Int
0
let thrCallBack :: WorkerJob
thrCallBack = WorkerJob
job
thrMaxThreads <- (Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Word32 -> Int) -> (MiscFlags -> Word32) -> MiscFlags -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. MiscFlags -> Word32
numIoWorkerThreads) (MiscFlags -> Int) -> IO MiscFlags -> IO Int
forall a b. (a -> b) -> IO a -> IO b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
`fmap` IO MiscFlags
getMiscFlags
thrActiveThreads <- newMVar 0
thrMonitor <- newEmptyMVar
thrThreadIds <- undefined
let thrMainThread = Maybe a
forall a. Maybe a
Nothing
let !pool = ThreadPool{Int
Maybe ThreadId
WorkerJob
MVar Int
MVar ()
Array ThreadId
forall a. Maybe a
thrMainThread :: Maybe ThreadId
thrMaxThreads :: Int
thrMinThreads :: Int
thrCurThreads :: Int
thrCallBack :: WorkerJob
thrActiveThreads :: MVar Int
thrMonitor :: MVar ()
thrThreadIds :: Array ThreadId
thrMinThreads :: Int
thrCurThreads :: Int
thrCallBack :: WorkerJob
thrMaxThreads :: Int
thrActiveThreads :: MVar Int
thrMonitor :: MVar ()
thrThreadIds :: Array ThreadId
thrMainThread :: forall a. Maybe a
..}
return pool
monitorThreadPool :: MVar () -> IO ()
monitorThreadPool :: MVar () -> WorkerJob
monitorThreadPool MVar ()
monitor = do
_active <- MVar () -> WorkerJob
forall a. MVar a -> IO a
takeMVar MVar ()
monitor
return ()
notifyRunning :: Maybe ThreadPool -> IO ()
notifyRunning :: Maybe ThreadPool -> WorkerJob
notifyRunning Maybe ThreadPool
Nothing = () -> WorkerJob
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
notifyRunning (Just ThreadPool
pool) = do
MVar Int -> (Int -> IO Int) -> WorkerJob
forall a. MVar a -> (a -> IO a) -> WorkerJob
modifyMVar_ (ThreadPool -> MVar Int
thrActiveThreads ThreadPool
pool) (\Int
x -> Int -> IO Int
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return (Int -> IO Int) -> Int -> IO Int
forall a b. (a -> b) -> a -> b
$ Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
_ <- MVar () -> () -> IO Bool
forall a. MVar a -> a -> IO Bool
tryPutMVar (ThreadPool -> MVar ()
thrMonitor ThreadPool
pool) ()
return ()
notifyWaiting :: Maybe ThreadPool -> IO ()
notifyWaiting :: Maybe ThreadPool -> WorkerJob
notifyWaiting Maybe ThreadPool
Nothing = () -> WorkerJob
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return ()
notifyWaiting (Just ThreadPool
pool) = do
MVar Int -> (Int -> IO Int) -> WorkerJob
forall a. MVar a -> (a -> IO a) -> WorkerJob
modifyMVar_ (ThreadPool -> MVar Int
thrActiveThreads ThreadPool
pool) (\Int
x -> Int -> IO Int
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return (Int -> IO Int) -> Int -> IO Int
forall a b. (a -> b) -> a -> b
$ Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)
_ <- MVar () -> () -> IO Bool
forall a. MVar a -> a -> IO Bool
tryPutMVar (ThreadPool -> MVar ()
thrMonitor ThreadPool
pool) ()
return ()