|  | //! Rayon-core houses the core stable APIs of Rayon. | 
|  | //! | 
|  | //! These APIs have been mirrored in the Rayon crate and it is recommended to use these from there. | 
|  | //! | 
|  | //! [`join`] is used to take two closures and potentially run them in parallel. | 
|  | //!   - It will run in parallel if task B gets stolen before task A can finish. | 
|  | //!   - It will run sequentially if task A finishes before task B is stolen and can continue on task B. | 
|  | //! | 
|  | //! [`scope`] creates a scope in which you can run any number of parallel tasks. | 
|  | //! These tasks can spawn nested tasks and scopes, but given the nature of work stealing, the order of execution can not be guaranteed. | 
|  | //! The scope will exist until all tasks spawned within the scope have been completed. | 
|  | //! | 
|  | //! [`spawn`] add a task into the 'static' or 'global' scope, or a local scope created by the [`scope()`] function. | 
|  | //! | 
|  | //! [`ThreadPool`] can be used to create your own thread pools (using [`ThreadPoolBuilder`]) or to customize the global one. | 
|  | //! Tasks spawned within the pool (using [`install()`], [`join()`], etc.) will be added to a deque, | 
|  | //! where it becomes available for work stealing from other threads in the local threadpool. | 
|  | //! | 
|  | //! [`join`]: fn.join.html | 
|  | //! [`scope`]: fn.scope.html | 
|  | //! [`scope()`]: fn.scope.html | 
|  | //! [`spawn`]: fn.spawn.html | 
|  | //! [`ThreadPool`]: struct.threadpool.html | 
|  | //! [`install()`]: struct.ThreadPool.html#method.install | 
|  | //! [`spawn()`]: struct.ThreadPool.html#method.spawn | 
|  | //! [`join()`]: struct.ThreadPool.html#method.join | 
|  | //! [`ThreadPoolBuilder`]: struct.ThreadPoolBuilder.html | 
|  | //! | 
|  | //! # Global fallback when threading is unsupported | 
|  | //! | 
|  | //! Rayon uses `std` APIs for threading, but some targets have incomplete implementations that | 
|  | //! always return `Unsupported` errors. The WebAssembly `wasm32-unknown-unknown` and `wasm32-wasi` | 
|  | //! targets are notable examples of this. Rather than panicking on the unsupported error when | 
|  | //! creating the implicit global threadpool, Rayon configures a fallback mode instead. | 
|  | //! | 
|  | //! This fallback mode mostly functions as if it were using a single-threaded "pool", like setting | 
|  | //! `RAYON_NUM_THREADS=1`. For example, `join` will execute its two closures sequentially, since | 
|  | //! there is no other thread to share the work. However, since the pool is not running independent | 
|  | //! of the main thread, non-blocking calls like `spawn` may not execute at all, unless a lower- | 
|  | //! priority call like `broadcast` gives them an opening. The fallback mode does not try to emulate | 
|  | //! anything like thread preemption or `async` task switching, but `yield_now` or `yield_local` | 
|  | //! can also volunteer execution time. | 
|  | //! | 
|  | //! Explicit `ThreadPoolBuilder` methods always report their error without any fallback. | 
|  | //! | 
|  | //! # Restricting multiple versions | 
|  | //! | 
|  | //! In order to ensure proper coordination between threadpools, and especially | 
|  | //! to make sure there's only one global threadpool, `rayon-core` is actively | 
|  | //! restricted from building multiple versions of itself into a single target. | 
|  | //! You may see a build error like this in violation: | 
|  | //! | 
|  | //! ```text | 
|  | //! error: native library `rayon-core` is being linked to by more | 
|  | //! than one package, and can only be linked to by one package | 
|  | //! ``` | 
|  | //! | 
|  | //! While we strive to keep `rayon-core` semver-compatible, it's still | 
|  | //! possible to arrive at this situation if different crates have overly | 
|  | //! restrictive tilde or inequality requirements for `rayon-core`. The | 
|  | //! conflicting requirements will need to be resolved before the build will | 
|  | //! succeed. | 
|  |  | 
|  | #![cfg_attr(test, allow(unused_crate_dependencies))] | 
|  | #![warn(rust_2018_idioms)] | 
|  |  | 
|  | use std::any::Any; | 
|  | use std::error::Error; | 
|  | use std::marker::PhantomData; | 
|  | use std::str::FromStr; | 
|  | use std::{env, fmt, io, thread}; | 
|  |  | 
|  | #[macro_use] | 
|  | mod private; | 
|  |  | 
|  | mod broadcast; | 
|  | mod job; | 
|  | mod join; | 
|  | mod latch; | 
|  | mod registry; | 
|  | mod scope; | 
|  | mod sleep; | 
|  | mod spawn; | 
|  | mod thread_pool; | 
|  | mod unwind; | 
|  | mod worker_local; | 
|  |  | 
|  | mod compile_fail; | 
|  | mod tests; | 
|  |  | 
|  | pub mod tlv; | 
|  |  | 
|  | pub use worker_local::WorkerLocal; | 
|  |  | 
|  | pub use self::broadcast::{BroadcastContext, broadcast, spawn_broadcast}; | 
|  | pub use self::join::{join, join_context}; | 
|  | use self::registry::{CustomSpawn, DefaultSpawn, ThreadSpawn}; | 
|  | pub use self::registry::{Registry, ThreadBuilder, mark_blocked, mark_unblocked}; | 
|  | pub use self::scope::{Scope, ScopeFifo, in_place_scope, in_place_scope_fifo, scope, scope_fifo}; | 
|  | pub use self::spawn::{spawn, spawn_fifo}; | 
|  | pub use self::thread_pool::{ | 
|  | ThreadPool, Yield, current_thread_has_pending_tasks, current_thread_index, yield_local, | 
|  | yield_now, | 
|  | }; | 
|  |  | 
|  | /// Returns the maximum number of threads that Rayon supports in a single thread-pool. | 
|  | /// | 
|  | /// If a higher thread count is requested by calling `ThreadPoolBuilder::num_threads` or by setting | 
|  | /// the `RAYON_NUM_THREADS` environment variable, then it will be reduced to this maximum. | 
|  | /// | 
|  | /// The value may vary between different targets, and is subject to change in new Rayon versions. | 
|  | pub fn max_num_threads() -> usize { | 
|  | // We are limited by the bits available in the sleep counter's `AtomicUsize`. | 
|  | crate::sleep::THREADS_MAX | 
|  | } | 
|  |  | 
|  | /// Returns the number of threads in the current registry. If this | 
|  | /// code is executing within a Rayon thread-pool, then this will be | 
|  | /// the number of threads for the thread-pool of the current | 
|  | /// thread. Otherwise, it will be the number of threads for the global | 
|  | /// thread-pool. | 
|  | /// | 
|  | /// This can be useful when trying to judge how many times to split | 
|  | /// parallel work (the parallel iterator traits use this value | 
|  | /// internally for this purpose). | 
|  | /// | 
|  | /// # Future compatibility note | 
|  | /// | 
|  | /// Note that unless this thread-pool was created with a | 
|  | /// builder that specifies the number of threads, then this | 
|  | /// number may vary over time in future versions (see [the | 
|  | /// `num_threads()` method for details][snt]). | 
|  | /// | 
|  | /// [snt]: struct.ThreadPoolBuilder.html#method.num_threads | 
|  | pub fn current_num_threads() -> usize { | 
|  | crate::registry::Registry::current_num_threads() | 
|  | } | 
|  |  | 
|  | /// Error when initializing a thread pool. | 
|  | #[derive(Debug)] | 
|  | pub struct ThreadPoolBuildError { | 
|  | kind: ErrorKind, | 
|  | } | 
|  |  | 
|  | #[derive(Debug)] | 
|  | enum ErrorKind { | 
|  | GlobalPoolAlreadyInitialized, | 
|  | IOError(io::Error), | 
|  | } | 
|  |  | 
|  | /// Used to create a new [`ThreadPool`] or to configure the global rayon thread pool. | 
|  | /// ## Creating a ThreadPool | 
|  | /// The following creates a thread pool with 22 threads. | 
|  | /// | 
|  | /// ```rust | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// let pool = rayon::ThreadPoolBuilder::new().num_threads(22).build().unwrap(); | 
|  | /// ``` | 
|  | /// | 
|  | /// To instead configure the global thread pool, use [`build_global()`]: | 
|  | /// | 
|  | /// ```rust | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// rayon::ThreadPoolBuilder::new().num_threads(22).build_global().unwrap(); | 
|  | /// ``` | 
|  | /// | 
|  | /// [`ThreadPool`]: struct.ThreadPool.html | 
|  | /// [`build_global()`]: struct.ThreadPoolBuilder.html#method.build_global | 
|  | pub struct ThreadPoolBuilder<S = DefaultSpawn> { | 
|  | /// The number of threads in the rayon thread pool. | 
|  | /// If zero will use the RAYON_NUM_THREADS environment variable. | 
|  | /// If RAYON_NUM_THREADS is invalid or zero will use the default. | 
|  | num_threads: usize, | 
|  |  | 
|  | /// Custom closure, if any, to handle a panic that we cannot propagate | 
|  | /// anywhere else. | 
|  | panic_handler: Option<Box<PanicHandler>>, | 
|  |  | 
|  | /// Closure to compute the name of a thread. | 
|  | get_thread_name: Option<Box<dyn FnMut(usize) -> String>>, | 
|  |  | 
|  | /// The stack size for the created worker threads | 
|  | stack_size: Option<usize>, | 
|  |  | 
|  | /// Closure invoked on deadlock. | 
|  | deadlock_handler: Option<Box<DeadlockHandler>>, | 
|  |  | 
|  | /// Closure invoked on worker thread start. | 
|  | start_handler: Option<Box<StartHandler>>, | 
|  |  | 
|  | /// Closure invoked on worker thread exit. | 
|  | exit_handler: Option<Box<ExitHandler>>, | 
|  |  | 
|  | /// Closure invoked to spawn threads. | 
|  | spawn_handler: S, | 
|  |  | 
|  | /// Closure invoked when starting computations in a thread. | 
|  | acquire_thread_handler: Option<Box<AcquireThreadHandler>>, | 
|  |  | 
|  | /// Closure invoked when blocking in a thread. | 
|  | release_thread_handler: Option<Box<ReleaseThreadHandler>>, | 
|  |  | 
|  | /// If false, worker threads will execute spawned jobs in a | 
|  | /// "depth-first" fashion. If true, they will do a "breadth-first" | 
|  | /// fashion. Depth-first is the default. | 
|  | breadth_first: bool, | 
|  | } | 
|  |  | 
|  | /// Contains the rayon thread pool configuration. Use [`ThreadPoolBuilder`] instead. | 
|  | /// | 
|  | /// [`ThreadPoolBuilder`]: struct.ThreadPoolBuilder.html | 
|  | #[deprecated(note = "Use `ThreadPoolBuilder`")] | 
|  | #[derive(Default)] | 
|  | pub struct Configuration { | 
|  | builder: ThreadPoolBuilder, | 
|  | } | 
|  |  | 
|  | /// The type for a panic handling closure. Note that this same closure | 
|  | /// may be invoked multiple times in parallel. | 
|  | type PanicHandler = dyn Fn(Box<dyn Any + Send>) + Send + Sync; | 
|  |  | 
|  | /// The type for a closure that gets invoked when the Rayon thread pool deadlocks | 
|  | type DeadlockHandler = dyn Fn() + Send + Sync; | 
|  |  | 
|  | /// The type for a closure that gets invoked when a thread starts. The | 
|  | /// closure is passed the index of the thread on which it is invoked. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | type StartHandler = dyn Fn(usize) + Send + Sync; | 
|  |  | 
|  | /// The type for a closure that gets invoked when a thread exits. The | 
|  | /// closure is passed the index of the thread on which it is invoked. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | type ExitHandler = dyn Fn(usize) + Send + Sync; | 
|  |  | 
|  | // NB: We can't `#[derive(Default)]` because `S` is left ambiguous. | 
|  | impl Default for ThreadPoolBuilder { | 
|  | fn default() -> Self { | 
|  | ThreadPoolBuilder { | 
|  | num_threads: 0, | 
|  | panic_handler: None, | 
|  | get_thread_name: None, | 
|  | stack_size: None, | 
|  | start_handler: None, | 
|  | exit_handler: None, | 
|  | deadlock_handler: None, | 
|  | acquire_thread_handler: None, | 
|  | release_thread_handler: None, | 
|  | spawn_handler: DefaultSpawn, | 
|  | breadth_first: false, | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /// The type for a closure that gets invoked before starting computations in a thread. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | type AcquireThreadHandler = dyn Fn() + Send + Sync; | 
|  |  | 
|  | /// The type for a closure that gets invoked before blocking in a thread. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | type ReleaseThreadHandler = dyn Fn() + Send + Sync; | 
|  |  | 
|  | impl ThreadPoolBuilder { | 
|  | /// Creates and returns a valid rayon thread pool builder, but does not initialize it. | 
|  | pub fn new() -> Self { | 
|  | Self::default() | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Note: the `S: ThreadSpawn` constraint is an internal implementation detail for the | 
|  | /// default spawn and those set by [`spawn_handler`](#method.spawn_handler). | 
|  | impl<S> ThreadPoolBuilder<S> | 
|  | where | 
|  | S: ThreadSpawn, | 
|  | { | 
|  | /// Creates a new `ThreadPool` initialized using this configuration. | 
|  | pub fn build(self) -> Result<ThreadPool, ThreadPoolBuildError> { | 
|  | ThreadPool::build(self) | 
|  | } | 
|  |  | 
|  | /// Initializes the global thread pool. This initialization is | 
|  | /// **optional**. If you do not call this function, the thread pool | 
|  | /// will be automatically initialized with the default | 
|  | /// configuration. Calling `build_global` is not recommended, except | 
|  | /// in two scenarios: | 
|  | /// | 
|  | /// - You wish to change the default configuration. | 
|  | /// - You are running a benchmark, in which case initializing may | 
|  | ///   yield slightly more consistent results, since the worker threads | 
|  | ///   will already be ready to go even in the first iteration. But | 
|  | ///   this cost is minimal. | 
|  | /// | 
|  | /// Initialization of the global thread pool happens exactly | 
|  | /// once. Once started, the configuration cannot be | 
|  | /// changed. Therefore, if you call `build_global` a second time, it | 
|  | /// will return an error. An `Ok` result indicates that this | 
|  | /// is the first initialization of the thread pool. | 
|  | pub fn build_global(self) -> Result<(), ThreadPoolBuildError> { | 
|  | let registry = registry::init_global_registry(self)?; | 
|  | registry.wait_until_primed(); | 
|  | Ok(()) | 
|  | } | 
|  | } | 
|  |  | 
|  | impl ThreadPoolBuilder { | 
|  | /// Creates a scoped `ThreadPool` initialized using this configuration. | 
|  | /// | 
|  | /// This is a convenience function for building a pool using [`std::thread::scope`] | 
|  | /// to spawn threads in a [`spawn_handler`](#method.spawn_handler). | 
|  | /// The threads in this pool will start by calling `wrapper`, which should | 
|  | /// do initialization and continue by calling `ThreadBuilder::run()`. | 
|  | /// | 
|  | /// [`std::thread::scope`]: https://doc.rust-lang.org/std/thread/fn.scope.html | 
|  | /// | 
|  | /// # Examples | 
|  | /// | 
|  | /// A scoped pool may be useful in combination with scoped thread-local variables. | 
|  | /// | 
|  | /// ``` | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// | 
|  | /// scoped_tls::scoped_thread_local!(static POOL_DATA: Vec<i32>); | 
|  | /// | 
|  | /// fn main() -> Result<(), rayon::ThreadPoolBuildError> { | 
|  | ///     let pool_data = vec![1, 2, 3]; | 
|  | /// | 
|  | ///     // We haven't assigned any TLS data yet. | 
|  | ///     assert!(!POOL_DATA.is_set()); | 
|  | /// | 
|  | ///     rayon::ThreadPoolBuilder::new() | 
|  | ///         .build_scoped( | 
|  | ///             // Borrow `pool_data` in TLS for each thread. | 
|  | ///             |thread| POOL_DATA.set(&pool_data, || thread.run()), | 
|  | ///             // Do some work that needs the TLS data. | 
|  | ///             |pool| pool.install(|| assert!(POOL_DATA.is_set())), | 
|  | ///         )?; | 
|  | /// | 
|  | ///     // Once we've returned, `pool_data` is no longer borrowed. | 
|  | ///     drop(pool_data); | 
|  | ///     Ok(()) | 
|  | /// } | 
|  | /// ``` | 
|  | pub fn build_scoped<W, F, R>(self, wrapper: W, with_pool: F) -> Result<R, ThreadPoolBuildError> | 
|  | where | 
|  | W: Fn(ThreadBuilder) + Sync, // expected to call `run()` | 
|  | F: FnOnce(&ThreadPool) -> R, | 
|  | { | 
|  | std::thread::scope(|scope| { | 
|  | let pool = self | 
|  | .spawn_handler(|thread| { | 
|  | let mut builder = std::thread::Builder::new(); | 
|  | if let Some(name) = thread.name() { | 
|  | builder = builder.name(name.to_string()); | 
|  | } | 
|  | if let Some(size) = thread.stack_size() { | 
|  | builder = builder.stack_size(size); | 
|  | } | 
|  | builder.spawn_scoped(scope, || wrapper(thread))?; | 
|  | Ok(()) | 
|  | }) | 
|  | .build()?; | 
|  | let result = unwind::halt_unwinding(|| with_pool(&pool)); | 
|  | pool.wait_until_stopped(); | 
|  | match result { | 
|  | Ok(result) => Ok(result), | 
|  | Err(err) => unwind::resume_unwinding(err), | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | impl<S> ThreadPoolBuilder<S> { | 
|  | /// Sets a custom function for spawning threads. | 
|  | /// | 
|  | /// Note that the threads will not exit until after the pool is dropped. It | 
|  | /// is up to the caller to wait for thread termination if that is important | 
|  | /// for any invariants. For instance, threads created in [`std::thread::scope`] | 
|  | /// will be joined before that scope returns, and this will block indefinitely | 
|  | /// if the pool is leaked. Furthermore, the global thread pool doesn't terminate | 
|  | /// until the entire process exits! | 
|  | /// | 
|  | /// # Examples | 
|  | /// | 
|  | /// A minimal spawn handler just needs to call `run()` from an independent thread. | 
|  | /// | 
|  | /// ``` | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// fn main() -> Result<(), rayon::ThreadPoolBuildError> { | 
|  | ///     let pool = rayon::ThreadPoolBuilder::new() | 
|  | ///         .spawn_handler(|thread| { | 
|  | ///             std::thread::spawn(|| thread.run()); | 
|  | ///             Ok(()) | 
|  | ///         }) | 
|  | ///         .build()?; | 
|  | /// | 
|  | ///     pool.install(|| println!("Hello from my custom thread!")); | 
|  | ///     Ok(()) | 
|  | /// } | 
|  | /// ``` | 
|  | /// | 
|  | /// The default spawn handler sets the name and stack size if given, and propagates | 
|  | /// any errors from the thread builder. | 
|  | /// | 
|  | /// ``` | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// fn main() -> Result<(), rayon::ThreadPoolBuildError> { | 
|  | ///     let pool = rayon::ThreadPoolBuilder::new() | 
|  | ///         .spawn_handler(|thread| { | 
|  | ///             let mut b = std::thread::Builder::new(); | 
|  | ///             if let Some(name) = thread.name() { | 
|  | ///                 b = b.name(name.to_owned()); | 
|  | ///             } | 
|  | ///             if let Some(stack_size) = thread.stack_size() { | 
|  | ///                 b = b.stack_size(stack_size); | 
|  | ///             } | 
|  | ///             b.spawn(|| thread.run())?; | 
|  | ///             Ok(()) | 
|  | ///         }) | 
|  | ///         .build()?; | 
|  | /// | 
|  | ///     pool.install(|| println!("Hello from my fully custom thread!")); | 
|  | ///     Ok(()) | 
|  | /// } | 
|  | /// ``` | 
|  | /// | 
|  | /// This can also be used for a pool of scoped threads like [`crossbeam::scope`], | 
|  | /// or [`std::thread::scope`] introduced in Rust 1.63, which is encapsulated in | 
|  | /// [`build_scoped`](#method.build_scoped). | 
|  | /// | 
|  | /// [`crossbeam::scope`]: https://docs.rs/crossbeam/0.8/crossbeam/fn.scope.html | 
|  | /// [`std::thread::scope`]: https://doc.rust-lang.org/std/thread/fn.scope.html | 
|  | /// | 
|  | /// ``` | 
|  | /// # use rustc_thread_pool as rayon; | 
|  | /// fn main() -> Result<(), rayon::ThreadPoolBuildError> { | 
|  | ///     std::thread::scope(|scope| { | 
|  | ///         let pool = rayon::ThreadPoolBuilder::new() | 
|  | ///             .spawn_handler(|thread| { | 
|  | ///                 let mut builder = std::thread::Builder::new(); | 
|  | ///                 if let Some(name) = thread.name() { | 
|  | ///                     builder = builder.name(name.to_string()); | 
|  | ///                 } | 
|  | ///                 if let Some(size) = thread.stack_size() { | 
|  | ///                     builder = builder.stack_size(size); | 
|  | ///                 } | 
|  | ///                 builder.spawn_scoped(scope, || { | 
|  | ///                     // Add any scoped initialization here, then run! | 
|  | ///                     thread.run() | 
|  | ///                 })?; | 
|  | ///                 Ok(()) | 
|  | ///             }) | 
|  | ///             .build()?; | 
|  | /// | 
|  | ///         pool.install(|| println!("Hello from my custom scoped thread!")); | 
|  | ///         Ok(()) | 
|  | ///     }) | 
|  | /// } | 
|  | /// ``` | 
|  | pub fn spawn_handler<F>(self, spawn: F) -> ThreadPoolBuilder<CustomSpawn<F>> | 
|  | where | 
|  | F: FnMut(ThreadBuilder) -> io::Result<()>, | 
|  | { | 
|  | ThreadPoolBuilder { | 
|  | spawn_handler: CustomSpawn::new(spawn), | 
|  | // ..self | 
|  | num_threads: self.num_threads, | 
|  | panic_handler: self.panic_handler, | 
|  | get_thread_name: self.get_thread_name, | 
|  | stack_size: self.stack_size, | 
|  | start_handler: self.start_handler, | 
|  | exit_handler: self.exit_handler, | 
|  | deadlock_handler: self.deadlock_handler, | 
|  | acquire_thread_handler: self.acquire_thread_handler, | 
|  | release_thread_handler: self.release_thread_handler, | 
|  | breadth_first: self.breadth_first, | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Returns a reference to the current spawn handler. | 
|  | fn get_spawn_handler(&mut self) -> &mut S { | 
|  | &mut self.spawn_handler | 
|  | } | 
|  |  | 
|  | /// Get the number of threads that will be used for the thread | 
|  | /// pool. See `num_threads()` for more information. | 
|  | fn get_num_threads(&self) -> usize { | 
|  | if self.num_threads > 0 { | 
|  | self.num_threads | 
|  | } else { | 
|  | let default = || thread::available_parallelism().map(|n| n.get()).unwrap_or(1); | 
|  |  | 
|  | match env::var("RAYON_NUM_THREADS").ok().and_then(|s| usize::from_str(&s).ok()) { | 
|  | Some(x @ 1..) => return x, | 
|  | Some(0) => return default(), | 
|  | _ => {} | 
|  | } | 
|  |  | 
|  | // Support for deprecated `RAYON_RS_NUM_CPUS`. | 
|  | match env::var("RAYON_RS_NUM_CPUS").ok().and_then(|s| usize::from_str(&s).ok()) { | 
|  | Some(x @ 1..) => x, | 
|  | _ => default(), | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Get the thread name for the thread with the given index. | 
|  | fn get_thread_name(&mut self, index: usize) -> Option<String> { | 
|  | let f = self.get_thread_name.as_mut()?; | 
|  | Some(f(index)) | 
|  | } | 
|  |  | 
|  | /// Sets a closure which takes a thread index and returns | 
|  | /// the thread's name. | 
|  | pub fn thread_name<F>(mut self, closure: F) -> Self | 
|  | where | 
|  | F: FnMut(usize) -> String + 'static, | 
|  | { | 
|  | self.get_thread_name = Some(Box::new(closure)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Sets the number of threads to be used in the rayon threadpool. | 
|  | /// | 
|  | /// If you specify a non-zero number of threads using this | 
|  | /// function, then the resulting thread-pools are guaranteed to | 
|  | /// start at most this number of threads. | 
|  | /// | 
|  | /// If `num_threads` is 0, or you do not call this function, then | 
|  | /// the Rayon runtime will select the number of threads | 
|  | /// automatically. At present, this is based on the | 
|  | /// `RAYON_NUM_THREADS` environment variable (if set), | 
|  | /// or the number of logical CPUs (otherwise). | 
|  | /// In the future, however, the default behavior may | 
|  | /// change to dynamically add or remove threads as needed. | 
|  | /// | 
|  | /// **Future compatibility warning:** Given the default behavior | 
|  | /// may change in the future, if you wish to rely on a fixed | 
|  | /// number of threads, you should use this function to specify | 
|  | /// that number. To reproduce the current default behavior, you | 
|  | /// may wish to use [`std::thread::available_parallelism`] | 
|  | /// to query the number of CPUs dynamically. | 
|  | /// | 
|  | /// **Old environment variable:** `RAYON_NUM_THREADS` is a one-to-one | 
|  | /// replacement of the now deprecated `RAYON_RS_NUM_CPUS` environment | 
|  | /// variable. If both variables are specified, `RAYON_NUM_THREADS` will | 
|  | /// be preferred. | 
|  | pub fn num_threads(mut self, num_threads: usize) -> Self { | 
|  | self.num_threads = num_threads; | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Returns a copy of the current panic handler. | 
|  | fn take_panic_handler(&mut self) -> Option<Box<PanicHandler>> { | 
|  | self.panic_handler.take() | 
|  | } | 
|  |  | 
|  | /// Normally, whenever Rayon catches a panic, it tries to | 
|  | /// propagate it to someplace sensible, to try and reflect the | 
|  | /// semantics of sequential execution. But in some cases, | 
|  | /// particularly with the `spawn()` APIs, there is no | 
|  | /// obvious place where we should propagate the panic to. | 
|  | /// In that case, this panic handler is invoked. | 
|  | /// | 
|  | /// If no panic handler is set, the default is to abort the | 
|  | /// process, under the principle that panics should not go | 
|  | /// unobserved. | 
|  | /// | 
|  | /// If the panic handler itself panics, this will abort the | 
|  | /// process. To prevent this, wrap the body of your panic handler | 
|  | /// in a call to `std::panic::catch_unwind()`. | 
|  | pub fn panic_handler<H>(mut self, panic_handler: H) -> Self | 
|  | where | 
|  | H: Fn(Box<dyn Any + Send>) + Send + Sync + 'static, | 
|  | { | 
|  | self.panic_handler = Some(Box::new(panic_handler)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Get the stack size of the worker threads | 
|  | fn get_stack_size(&self) -> Option<usize> { | 
|  | self.stack_size | 
|  | } | 
|  |  | 
|  | /// Sets the stack size of the worker threads | 
|  | pub fn stack_size(mut self, stack_size: usize) -> Self { | 
|  | self.stack_size = Some(stack_size); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// **(DEPRECATED)** Suggest to worker threads that they execute | 
|  | /// spawned jobs in a "breadth-first" fashion. | 
|  | /// | 
|  | /// Typically, when a worker thread is idle or blocked, it will | 
|  | /// attempt to execute the job from the *top* of its local deque of | 
|  | /// work (i.e., the job most recently spawned). If this flag is set | 
|  | /// to true, however, workers will prefer to execute in a | 
|  | /// *breadth-first* fashion -- that is, they will search for jobs at | 
|  | /// the *bottom* of their local deque. (At present, workers *always* | 
|  | /// steal from the bottom of other workers' deques, regardless of | 
|  | /// the setting of this flag.) | 
|  | /// | 
|  | /// If you think of the tasks as a tree, where a parent task | 
|  | /// spawns its children in the tree, then this flag loosely | 
|  | /// corresponds to doing a breadth-first traversal of the tree, | 
|  | /// whereas the default would be to do a depth-first traversal. | 
|  | /// | 
|  | /// **Note that this is an "execution hint".** Rayon's task | 
|  | /// execution is highly dynamic and the precise order in which | 
|  | /// independent tasks are executed is not intended to be | 
|  | /// guaranteed. | 
|  | /// | 
|  | /// This `breadth_first()` method is now deprecated per [RFC #1], | 
|  | /// and in the future its effect may be removed. Consider using | 
|  | /// [`scope_fifo()`] for a similar effect. | 
|  | /// | 
|  | /// [RFC #1]: https://github.com/rayon-rs/rfcs/blob/master/accepted/rfc0001-scope-scheduling.md | 
|  | /// [`scope_fifo()`]: fn.scope_fifo.html | 
|  | #[deprecated(note = "use `scope_fifo` and `spawn_fifo` for similar effect")] | 
|  | pub fn breadth_first(mut self) -> Self { | 
|  | self.breadth_first = true; | 
|  | self | 
|  | } | 
|  |  | 
|  | fn get_breadth_first(&self) -> bool { | 
|  | self.breadth_first | 
|  | } | 
|  |  | 
|  | /// Takes the current acquire thread callback, leaving `None`. | 
|  | fn take_acquire_thread_handler(&mut self) -> Option<Box<AcquireThreadHandler>> { | 
|  | self.acquire_thread_handler.take() | 
|  | } | 
|  |  | 
|  | /// Set a callback to be invoked when starting computations in a thread. | 
|  | pub fn acquire_thread_handler<H>(mut self, acquire_thread_handler: H) -> Self | 
|  | where | 
|  | H: Fn() + Send + Sync + 'static, | 
|  | { | 
|  | self.acquire_thread_handler = Some(Box::new(acquire_thread_handler)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Takes the current release thread callback, leaving `None`. | 
|  | fn take_release_thread_handler(&mut self) -> Option<Box<ReleaseThreadHandler>> { | 
|  | self.release_thread_handler.take() | 
|  | } | 
|  |  | 
|  | /// Set a callback to be invoked when blocking in thread. | 
|  | pub fn release_thread_handler<H>(mut self, release_thread_handler: H) -> Self | 
|  | where | 
|  | H: Fn() + Send + Sync + 'static, | 
|  | { | 
|  | self.release_thread_handler = Some(Box::new(release_thread_handler)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Takes the current deadlock callback, leaving `None`. | 
|  | fn take_deadlock_handler(&mut self) -> Option<Box<DeadlockHandler>> { | 
|  | self.deadlock_handler.take() | 
|  | } | 
|  |  | 
|  | /// Set a callback to be invoked on current deadlock. | 
|  | pub fn deadlock_handler<H>(mut self, deadlock_handler: H) -> Self | 
|  | where | 
|  | H: Fn() + Send + Sync + 'static, | 
|  | { | 
|  | self.deadlock_handler = Some(Box::new(deadlock_handler)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Takes the current thread start callback, leaving `None`. | 
|  | fn take_start_handler(&mut self) -> Option<Box<StartHandler>> { | 
|  | self.start_handler.take() | 
|  | } | 
|  |  | 
|  | /// Sets a callback to be invoked on thread start. | 
|  | /// | 
|  | /// The closure is passed the index of the thread on which it is invoked. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | /// If this closure panics, the panic will be passed to the panic handler. | 
|  | /// If that handler returns, then startup will continue normally. | 
|  | pub fn start_handler<H>(mut self, start_handler: H) -> Self | 
|  | where | 
|  | H: Fn(usize) + Send + Sync + 'static, | 
|  | { | 
|  | self.start_handler = Some(Box::new(start_handler)); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Returns a current thread exit callback, leaving `None`. | 
|  | fn take_exit_handler(&mut self) -> Option<Box<ExitHandler>> { | 
|  | self.exit_handler.take() | 
|  | } | 
|  |  | 
|  | /// Sets a callback to be invoked on thread exit. | 
|  | /// | 
|  | /// The closure is passed the index of the thread on which it is invoked. | 
|  | /// Note that this same closure may be invoked multiple times in parallel. | 
|  | /// If this closure panics, the panic will be passed to the panic handler. | 
|  | /// If that handler returns, then the thread will exit normally. | 
|  | pub fn exit_handler<H>(mut self, exit_handler: H) -> Self | 
|  | where | 
|  | H: Fn(usize) + Send + Sync + 'static, | 
|  | { | 
|  | self.exit_handler = Some(Box::new(exit_handler)); | 
|  | self | 
|  | } | 
|  | } | 
|  |  | 
|  | #[allow(deprecated)] | 
|  | impl Configuration { | 
|  | /// Creates and return a valid rayon thread pool configuration, but does not initialize it. | 
|  | pub fn new() -> Configuration { | 
|  | Configuration { builder: ThreadPoolBuilder::new() } | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::build`. | 
|  | pub fn build(self) -> Result<ThreadPool, Box<dyn Error + 'static>> { | 
|  | self.builder.build().map_err(Box::from) | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::thread_name`. | 
|  | pub fn thread_name<F>(mut self, closure: F) -> Self | 
|  | where | 
|  | F: FnMut(usize) -> String + 'static, | 
|  | { | 
|  | self.builder = self.builder.thread_name(closure); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::num_threads`. | 
|  | pub fn num_threads(mut self, num_threads: usize) -> Configuration { | 
|  | self.builder = self.builder.num_threads(num_threads); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::panic_handler`. | 
|  | pub fn panic_handler<H>(mut self, panic_handler: H) -> Configuration | 
|  | where | 
|  | H: Fn(Box<dyn Any + Send>) + Send + Sync + 'static, | 
|  | { | 
|  | self.builder = self.builder.panic_handler(panic_handler); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::stack_size`. | 
|  | pub fn stack_size(mut self, stack_size: usize) -> Self { | 
|  | self.builder = self.builder.stack_size(stack_size); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::breadth_first`. | 
|  | pub fn breadth_first(mut self) -> Self { | 
|  | self.builder = self.builder.breadth_first(); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::start_handler`. | 
|  | pub fn start_handler<H>(mut self, start_handler: H) -> Configuration | 
|  | where | 
|  | H: Fn(usize) + Send + Sync + 'static, | 
|  | { | 
|  | self.builder = self.builder.start_handler(start_handler); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::exit_handler`. | 
|  | pub fn exit_handler<H>(mut self, exit_handler: H) -> Configuration | 
|  | where | 
|  | H: Fn(usize) + Send + Sync + 'static, | 
|  | { | 
|  | self.builder = self.builder.exit_handler(exit_handler); | 
|  | self | 
|  | } | 
|  |  | 
|  | /// Returns a ThreadPoolBuilder with identical parameters. | 
|  | fn into_builder(self) -> ThreadPoolBuilder { | 
|  | self.builder | 
|  | } | 
|  | } | 
|  |  | 
|  | impl ThreadPoolBuildError { | 
|  | fn new(kind: ErrorKind) -> ThreadPoolBuildError { | 
|  | ThreadPoolBuildError { kind } | 
|  | } | 
|  |  | 
|  | fn is_unsupported(&self) -> bool { | 
|  | matches!(&self.kind, ErrorKind::IOError(e) if e.kind() == io::ErrorKind::Unsupported) | 
|  | } | 
|  | } | 
|  |  | 
|  | impl Error for ThreadPoolBuildError { | 
|  | fn source(&self) -> Option<&(dyn Error + 'static)> { | 
|  | match &self.kind { | 
|  | ErrorKind::GlobalPoolAlreadyInitialized => None, | 
|  | ErrorKind::IOError(e) => Some(e), | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | impl fmt::Display for ThreadPoolBuildError { | 
|  | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | 
|  | match &self.kind { | 
|  | ErrorKind::GlobalPoolAlreadyInitialized => { | 
|  | "The global thread pool has already been initialized.".fmt(f) | 
|  | } | 
|  | ErrorKind::IOError(e) => e.fmt(f), | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Deprecated in favor of `ThreadPoolBuilder::build_global`. | 
|  | #[deprecated(note = "use `ThreadPoolBuilder::build_global`")] | 
|  | #[allow(deprecated)] | 
|  | pub fn initialize(config: Configuration) -> Result<(), Box<dyn Error>> { | 
|  | config.into_builder().build_global().map_err(Box::from) | 
|  | } | 
|  |  | 
|  | impl<S> fmt::Debug for ThreadPoolBuilder<S> { | 
|  | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | 
|  | let ThreadPoolBuilder { | 
|  | ref num_threads, | 
|  | ref get_thread_name, | 
|  | ref panic_handler, | 
|  | ref stack_size, | 
|  | ref deadlock_handler, | 
|  | ref start_handler, | 
|  | ref exit_handler, | 
|  | ref acquire_thread_handler, | 
|  | ref release_thread_handler, | 
|  | spawn_handler: _, | 
|  | ref breadth_first, | 
|  | } = *self; | 
|  |  | 
|  | // Just print `Some(<closure>)` or `None` to the debug | 
|  | // output. | 
|  | struct ClosurePlaceholder; | 
|  | impl fmt::Debug for ClosurePlaceholder { | 
|  | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | 
|  | f.write_str("<closure>") | 
|  | } | 
|  | } | 
|  | let get_thread_name = get_thread_name.as_ref().map(|_| ClosurePlaceholder); | 
|  | let panic_handler = panic_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  | let deadlock_handler = deadlock_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  | let start_handler = start_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  | let exit_handler = exit_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  | let acquire_thread_handler = acquire_thread_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  | let release_thread_handler = release_thread_handler.as_ref().map(|_| ClosurePlaceholder); | 
|  |  | 
|  | f.debug_struct("ThreadPoolBuilder") | 
|  | .field("num_threads", num_threads) | 
|  | .field("get_thread_name", &get_thread_name) | 
|  | .field("panic_handler", &panic_handler) | 
|  | .field("stack_size", &stack_size) | 
|  | .field("deadlock_handler", &deadlock_handler) | 
|  | .field("start_handler", &start_handler) | 
|  | .field("exit_handler", &exit_handler) | 
|  | .field("acquire_thread_handler", &acquire_thread_handler) | 
|  | .field("release_thread_handler", &release_thread_handler) | 
|  | .field("breadth_first", &breadth_first) | 
|  | .finish() | 
|  | } | 
|  | } | 
|  |  | 
|  | #[allow(deprecated)] | 
|  | impl fmt::Debug for Configuration { | 
|  | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | 
|  | self.builder.fmt(f) | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Provides the calling context to a closure called by `join_context`. | 
|  | #[derive(Debug)] | 
|  | pub struct FnContext { | 
|  | migrated: bool, | 
|  |  | 
|  | /// disable `Send` and `Sync`, just for a little future-proofing. | 
|  | _marker: PhantomData<*mut ()>, | 
|  | } | 
|  |  | 
|  | impl FnContext { | 
|  | #[inline] | 
|  | fn new(migrated: bool) -> Self { | 
|  | FnContext { migrated, _marker: PhantomData } | 
|  | } | 
|  | } | 
|  |  | 
|  | impl FnContext { | 
|  | /// Returns `true` if the closure was called from a different thread | 
|  | /// than it was provided from. | 
|  | #[inline] | 
|  | pub fn migrated(&self) -> bool { | 
|  | self.migrated | 
|  | } | 
|  | } |