module Ki.Internal.Thread
( Thread,
makeThread,
await,
Tid,
ThreadAffinity (..),
forkWithAffinity,
ThreadOptions (..),
defaultThreadOptions,
ThreadFailed (..),
unwrapThreadFailed,
)
where
import Control.Concurrent (ThreadId, forkOS)
import Control.Exception
( BlockedIndefinitelyOnSTM (..),
Exception (fromException, toException),
MaskingState (..),
SomeException,
asyncExceptionFromException,
asyncExceptionToException,
)
import GHC.Conc (STM)
import Ki.Internal.ByteCount
import Ki.Internal.IO (forkIO, forkOn, tryEitherSTM)
data Thread a = Thread
{ forall a. Thread a -> ThreadId
threadId :: {-# UNPACK #-} !ThreadId,
forall a. Thread a -> STM a
await_ :: !(STM a)
}
deriving stock ((forall a b. (a -> b) -> Thread a -> Thread b)
-> (forall a b. a -> Thread b -> Thread a) -> Functor Thread
forall a b. a -> Thread b -> Thread a
forall a b. (a -> b) -> Thread a -> Thread b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
$cfmap :: forall a b. (a -> b) -> Thread a -> Thread b
fmap :: forall a b. (a -> b) -> Thread a -> Thread b
$c<$ :: forall a b. a -> Thread b -> Thread a
<$ :: forall a b. a -> Thread b -> Thread a
Functor)
instance Eq (Thread a) where
Thread ThreadId
ix STM a
_ == :: Thread a -> Thread a -> Bool
== Thread ThreadId
iy STM a
_ =
ThreadId
ix ThreadId -> ThreadId -> Bool
forall a. Eq a => a -> a -> Bool
== ThreadId
iy
instance Ord (Thread a) where
compare :: Thread a -> Thread a -> Ordering
compare (Thread ThreadId
ix STM a
_) (Thread ThreadId
iy STM a
_) =
ThreadId -> ThreadId -> Ordering
forall a. Ord a => a -> a -> Ordering
compare ThreadId
ix ThreadId
iy
makeThread :: ThreadId -> STM a -> Thread a
makeThread :: forall a. ThreadId -> STM a -> Thread a
makeThread ThreadId
threadId STM a
action =
Thread
{ ThreadId
$sel:threadId:Thread :: ThreadId
threadId :: ThreadId
threadId,
$sel:await_:Thread :: STM a
await_ = (BlockedIndefinitelyOnSTM -> STM a)
-> (a -> STM a) -> STM a -> STM a
forall e b a.
Exception e =>
(e -> STM b) -> (a -> STM b) -> STM a -> STM b
tryEitherSTM (\BlockedIndefinitelyOnSTM
BlockedIndefinitelyOnSTM -> STM a
action) a -> STM a
forall a. a -> STM a
forall (f :: * -> *) a. Applicative f => a -> f a
pure STM a
action
}
type Tid =
Int
data ThreadAffinity
=
Unbound
|
Capability Int
|
OsThread
deriving stock (ThreadAffinity -> ThreadAffinity -> Bool
(ThreadAffinity -> ThreadAffinity -> Bool)
-> (ThreadAffinity -> ThreadAffinity -> Bool) -> Eq ThreadAffinity
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: ThreadAffinity -> ThreadAffinity -> Bool
== :: ThreadAffinity -> ThreadAffinity -> Bool
$c/= :: ThreadAffinity -> ThreadAffinity -> Bool
/= :: ThreadAffinity -> ThreadAffinity -> Bool
Eq, Int -> ThreadAffinity -> ShowS
[ThreadAffinity] -> ShowS
ThreadAffinity -> String
(Int -> ThreadAffinity -> ShowS)
-> (ThreadAffinity -> String)
-> ([ThreadAffinity] -> ShowS)
-> Show ThreadAffinity
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> ThreadAffinity -> ShowS
showsPrec :: Int -> ThreadAffinity -> ShowS
$cshow :: ThreadAffinity -> String
show :: ThreadAffinity -> String
$cshowList :: [ThreadAffinity] -> ShowS
showList :: [ThreadAffinity] -> ShowS
Show)
forkWithAffinity :: ThreadAffinity -> IO () -> IO ThreadId
forkWithAffinity :: ThreadAffinity -> IO () -> IO ThreadId
forkWithAffinity = \case
ThreadAffinity
Unbound -> IO () -> IO ThreadId
forkIO
Capability Int
n -> Int -> IO () -> IO ThreadId
forkOn Int
n
ThreadAffinity
OsThread -> IO () -> IO ThreadId
Control.Concurrent.forkOS
data ThreadOptions = ThreadOptions
{ ThreadOptions -> ThreadAffinity
affinity :: ThreadAffinity,
ThreadOptions -> Maybe ByteCount
allocationLimit :: Maybe ByteCount,
ThreadOptions -> String
label :: String,
ThreadOptions -> MaskingState
maskingState :: MaskingState
}
deriving stock (ThreadOptions -> ThreadOptions -> Bool
(ThreadOptions -> ThreadOptions -> Bool)
-> (ThreadOptions -> ThreadOptions -> Bool) -> Eq ThreadOptions
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: ThreadOptions -> ThreadOptions -> Bool
== :: ThreadOptions -> ThreadOptions -> Bool
$c/= :: ThreadOptions -> ThreadOptions -> Bool
/= :: ThreadOptions -> ThreadOptions -> Bool
Eq, Int -> ThreadOptions -> ShowS
[ThreadOptions] -> ShowS
ThreadOptions -> String
(Int -> ThreadOptions -> ShowS)
-> (ThreadOptions -> String)
-> ([ThreadOptions] -> ShowS)
-> Show ThreadOptions
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> ThreadOptions -> ShowS
showsPrec :: Int -> ThreadOptions -> ShowS
$cshow :: ThreadOptions -> String
show :: ThreadOptions -> String
$cshowList :: [ThreadOptions] -> ShowS
showList :: [ThreadOptions] -> ShowS
Show)
defaultThreadOptions :: ThreadOptions
defaultThreadOptions :: ThreadOptions
defaultThreadOptions =
ThreadOptions
{ $sel:affinity:ThreadOptions :: ThreadAffinity
affinity = ThreadAffinity
Unbound,
$sel:allocationLimit:ThreadOptions :: Maybe ByteCount
allocationLimit = Maybe ByteCount
forall a. Maybe a
Nothing,
$sel:label:ThreadOptions :: String
label = String
"",
$sel:maskingState:ThreadOptions :: MaskingState
maskingState = MaskingState
Unmasked
}
data ThreadFailed = ThreadFailed
{ ThreadFailed -> Int
childId :: {-# UNPACK #-} !Tid,
ThreadFailed -> SomeException
exception :: !SomeException
}
deriving stock (Int -> ThreadFailed -> ShowS
[ThreadFailed] -> ShowS
ThreadFailed -> String
(Int -> ThreadFailed -> ShowS)
-> (ThreadFailed -> String)
-> ([ThreadFailed] -> ShowS)
-> Show ThreadFailed
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> ThreadFailed -> ShowS
showsPrec :: Int -> ThreadFailed -> ShowS
$cshow :: ThreadFailed -> String
show :: ThreadFailed -> String
$cshowList :: [ThreadFailed] -> ShowS
showList :: [ThreadFailed] -> ShowS
Show)
instance Exception ThreadFailed where
toException :: ThreadFailed -> SomeException
toException = ThreadFailed -> SomeException
forall e. Exception e => e -> SomeException
asyncExceptionToException
fromException :: SomeException -> Maybe ThreadFailed
fromException = SomeException -> Maybe ThreadFailed
forall e. Exception e => SomeException -> Maybe e
asyncExceptionFromException
unwrapThreadFailed :: SomeException -> SomeException
unwrapThreadFailed :: SomeException -> SomeException
unwrapThreadFailed SomeException
e0 =
case SomeException -> Maybe ThreadFailed
forall e. Exception e => SomeException -> Maybe e
fromException SomeException
e0 of
Just (ThreadFailed Int
_ SomeException
e1) -> SomeException
e1
Maybe ThreadFailed
Nothing -> SomeException
e0
await :: Thread a -> STM a
await :: forall a. Thread a -> STM a
await =
Thread a -> STM a
forall a. Thread a -> STM a
await_