Safe Haskell | Safe-Inferred |
---|---|
Language | Haskell2010 |
ki
is a lightweight structured concurrency library.
For a variant of this API generalized to
MonadUnliftIO
,
see ki-unlifted
.
Remember to link your program with -threaded
to use the threaded runtime!
Synopsis
- data Scope
- data Thread a
- scoped :: (Scope -> IO a) -> IO a
- fork :: Scope -> IO a -> IO (Thread a)
- forkTry :: forall e a. Exception e => Scope -> IO a -> IO (Thread (Either e a))
- await :: Thread a -> STM a
- awaitAll :: Scope -> STM ()
- fork_ :: Scope -> IO Void -> IO ()
- forkWith :: Scope -> ThreadOptions -> IO a -> IO (Thread a)
- forkWith_ :: Scope -> ThreadOptions -> IO Void -> IO ()
- forkTryWith :: forall e a. Exception e => Scope -> ThreadOptions -> IO a -> IO (Thread (Either e a))
- data ThreadOptions = ThreadOptions {}
- defaultThreadOptions :: ThreadOptions
- data ThreadAffinity
- = Unbound
- | Capability Int
- | OsThread
- data ByteCount
- kilobytes :: Natural -> ByteCount
- megabytes :: Natural -> ByteCount
Introduction
Structured concurrency is a paradigm of concurrent programming in which a lexical scope delimits the lifetime of each thread. Threads therefore form a "call tree" hierarchy in which no child can outlive its parent.
Exceptions are propagated promptly from child to parent and vice-versa:
- If an exception is raised in a child thread, the child raises the same exception in its parent, then terminates.
- If an exception is raised in a parent thread, the parent first raises an exception in all of its living children, waits for them to terminate, then re-raises the original exception.
All together, this library:
- Guarantees the absence of "ghost threads" (i.e. threads that accidentally continue to run alongside the main thread after the function that spawned them returns).
- Performs prompt, bidirectional exception propagation when an exception is raised anywhere in the call tree.
- Provides a safe and flexible API that can be used directly, or with which higher-level concurrency patterns can be built on top, such as worker queues, pub-sub pipelines, and supervision trees.
For a longer introduction to structured concurrency, including an educative analogy to structured programming, please read Nathaniel J. Smith's blog post, "Notes on structured concurrency, or: Go statement considered harmful".
👉 Quick start examples
Perform two actions concurrently, and wait for both of them to complete.
concurrently :: IO a -> IO b -> IO (a, b) concurrently action1 action2 = Ki.
scoped
\scope -> do thread1 <- Ki.fork
scope action1 result2 <- action2 result1 <- atomically (Ki.await
thread1) pure (result1, result2)Perform two actions concurrently, and when the first action terminates, stop executing the other.
race :: IO a -> IO a -> IO a race action1 action2 = Ki.
scoped
\scope -> do resultVar <- newEmptyMVar _ <- Ki.fork
scope (action1 >>= tryPutMVar resultVar) _ <- Ki.fork
scope (action2 >>= tryPutMVar resultVar) takeMVar resultVar
Core API
A scope.
👉 Details
- A scope delimits the lifetime of all threads created within it.
- A scope is only valid during the callback provided to
scoped
. - The thread that creates a scope is considered the parent of all threads created within it.
- All threads created within a scope can be awaited together (see
awaitAll
). - All threads created within a scope are terminated when the scope closes.
A thread.
👉 Details
- A thread's lifetime is delimited by the scope in which it was created.
- The thread that creates a scope is considered the parent of all threads created within it.
- If an exception is raised in a child thread, the child either propagates the exception to its parent (see
fork
), or returns the exception as a value (seeforkTry
). - All threads created within a scope are terminated when the scope closes.
scoped :: (Scope -> IO a) -> IO a Source #
Open a scope, perform an IO action with it, then close the scope.
👉 Details
- The thread that creates a scope is considered the parent of all threads created within it.
- A scope is only valid during the callback provided to
scoped
. When a scope closes (i.e. just before
scoped
returns):- The parent thread raises an exception in all of its living children.
- The parent thread blocks until those threads terminate.
fork :: Scope -> IO a -> IO (Thread a) Source #
Create a child thread to execute an action within a scope.
Note: The child thread does not mask asynchronous exceptions, regardless of the parent thread's masking state. To
create a child thread with a different initial masking state, use forkWith
.
forkTry :: forall e a. Exception e => Scope -> IO a -> IO (Thread (Either e a)) Source #
Like fork
, but the child thread does not propagate exceptions that are both:
- Synchronous (i.e. not an instance of
SomeAsyncException
). - An instance of
e
.
Extended API
forkWith :: Scope -> ThreadOptions -> IO a -> IO (Thread a) Source #
Variant of fork
that takes an additional options argument.
forkWith_ :: Scope -> ThreadOptions -> IO Void -> IO () Source #
Variant of forkWith
for threads that never return.
forkTryWith :: forall e a. Exception e => Scope -> ThreadOptions -> IO a -> IO (Thread (Either e a)) Source #
Variant of forkTry
that takes an additional options argument.
Thread options
data ThreadOptions Source #
affinity
The affinity of a thread. A thread can be unbound, bound to a specific capability, or bound to a specific OS thread.
Default:
Unbound
allocationLimit
The maximum number of bytes a thread may allocate before it is delivered an
AllocationLimitExceeded
exception. If caught, the thread is allowed to allocate an additional 100kb (tunable with+RTS -xq
) to perform any necessary cleanup actions; if exceeded, the thread is delivered another.Default:
Nothing
(no limit)label
The label of a thread, visible in the event log (
+RTS -l
).Default:
""
(no label)maskingState
The masking state a thread is created in. To unmask, use
unsafeUnmask
.Default:
Unmasked
Instances
Show ThreadOptions Source # | |
Defined in Ki.Internal.Thread showsPrec :: Int -> ThreadOptions -> ShowS # show :: ThreadOptions -> String # showList :: [ThreadOptions] -> ShowS # | |
Eq ThreadOptions Source # | |
Defined in Ki.Internal.Thread (==) :: ThreadOptions -> ThreadOptions -> Bool # (/=) :: ThreadOptions -> ThreadOptions -> Bool # |
defaultThreadOptions :: ThreadOptions Source #
Default thread options.
ThreadOptions
{affinity
=Unbound
,allocationLimit
= Nothing ,label
= "" ,maskingState
=Unmasked
}
data ThreadAffinity Source #
What, if anything, a thread is bound to.
Unbound | Unbound. |
Capability Int | Bound to a capability. |
OsThread | Bound to an OS thread. |
Instances
Show ThreadAffinity Source # | |
Defined in Ki.Internal.Thread showsPrec :: Int -> ThreadAffinity -> ShowS # show :: ThreadAffinity -> String # showList :: [ThreadAffinity] -> ShowS # | |
Eq ThreadAffinity Source # | |
Defined in Ki.Internal.Thread (==) :: ThreadAffinity -> ThreadAffinity -> Bool # (/=) :: ThreadAffinity -> ThreadAffinity -> Bool # |