summaryrefslogtreecommitdiffstats
path: root/third_party/rust/tokio-executor
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
commit2aa4a82499d4becd2284cdb482213d541b8804dd (patch)
treeb80bf8bf13c3766139fbacc530efd0dd9d54394c /third_party/rust/tokio-executor
parentInitial commit. (diff)
downloadfirefox-2aa4a82499d4becd2284cdb482213d541b8804dd.tar.xz
firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.zip
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'third_party/rust/tokio-executor')
-rw-r--r--third_party/rust/tokio-executor/.cargo-checksum.json1
-rw-r--r--third_party/rust/tokio-executor/CHANGELOG.md49
-rw-r--r--third_party/rust/tokio-executor/Cargo.toml30
-rw-r--r--third_party/rust/tokio-executor/LICENSE25
-rw-r--r--third_party/rust/tokio-executor/README.md47
-rw-r--r--third_party/rust/tokio-executor/src/enter.rs164
-rw-r--r--third_party/rust/tokio-executor/src/error.rs50
-rw-r--r--third_party/rust/tokio-executor/src/executor.rs151
-rw-r--r--third_party/rust/tokio-executor/src/global.rs290
-rw-r--r--third_party/rust/tokio-executor/src/lib.rs68
-rw-r--r--third_party/rust/tokio-executor/src/park.rs226
-rw-r--r--third_party/rust/tokio-executor/src/typed.rs181
-rw-r--r--third_party/rust/tokio-executor/tests/executor.rs29
13 files changed, 1311 insertions, 0 deletions
diff --git a/third_party/rust/tokio-executor/.cargo-checksum.json b/third_party/rust/tokio-executor/.cargo-checksum.json
new file mode 100644
index 0000000000..0b529dfe13
--- /dev/null
+++ b/third_party/rust/tokio-executor/.cargo-checksum.json
@@ -0,0 +1 @@
+{"files":{"CHANGELOG.md":"95eeaa7acd4fc1203637784a611d1202e5e5d6c644cb1c4f4d3c8b0da42ea920","Cargo.toml":"651887528b10435036b410b87f5a95edde0159ff4923102f92b1902f6b778a33","LICENSE":"898b1ae9821e98daf8964c8d6c7f61641f5f5aa78ad500020771c0939ee0dea1","README.md":"28d453f140f96a9bc110aab49854a3dbc67d17b8302d3203aaf78d74f6bc8f04","src/enter.rs":"fd96bb8f3fcd621cbbde99cd15cdcfff65d0d8f10fb22bbe01e95453af76bf01","src/error.rs":"142ec440d9bd6a500ea58fbcfe7c1a80aef699fa00c9a451b7cf4b237c03b0a1","src/executor.rs":"c8fce212c14f3e07aa1076023df0e4ee2290a65efa838e5f4fb0ab271e66c379","src/global.rs":"a6979e156b1e2989b872ded3b5965773545ce736da2891c46bf5da714ef6837c","src/lib.rs":"26689bd517d5b97996e06e4c777e47f29b89447a067de7223c45538ed50690a1","src/park.rs":"6d590f4ef8c161fd51b33e583cf014e4b53ef81d974663d76d4827b150938327","src/typed.rs":"0068829ab9429f51651bab97e066c970665e4a418a56bc62946a0d0e29c2e71b","tests/executor.rs":"ff891380e285651a11f87610b46ab7f9c88c0b67a390b43316b7c03df463df51"},"package":"ca6df436c42b0c3330a82d855d2ef017cd793090ad550a6bc2184f4b933532ab"} \ No newline at end of file
diff --git a/third_party/rust/tokio-executor/CHANGELOG.md b/third_party/rust/tokio-executor/CHANGELOG.md
new file mode 100644
index 0000000000..328085b0da
--- /dev/null
+++ b/third_party/rust/tokio-executor/CHANGELOG.md
@@ -0,0 +1,49 @@
+# 0.1.9 (November 27, 2019)
+
+### Added
+- Add `executor::set_default` which behaves like `with_default` but returns a
+ drop guard (#1725).
+
+# 0.1.8 (June 2, 2019)
+
+### Added
+- Add `executor::exit` to allow other executors inside `threadpool::blocking` (#1155).
+
+# 0.1.7 (March 22, 2019)
+
+### Added
+- `TypedExecutor` for spawning futures of a specific type (#993).
+
+# 0.1.6 (January 6, 2019)
+
+* Implement `Unpark` for `Arc<Unpark>` (#802).
+* Switch to crossbeam's Parker / Unparker (#528).
+
+# 0.1.5 (September 26, 2018)
+
+* Implement `futures::Executor` for `DefaultExecutor` (#563).
+* Add `Enter::block_on(future)` (#646)
+
+# 0.1.4 (August 23, 2018)
+
+* Implement `std::error::Error` for error types (#511).
+
+# 0.1.3 (August 6, 2018)
+
+* Implement `Executor` for `Box<E: Executor>` (#420).
+* Improve `EnterError` debug message (#410).
+* Implement `status`, `Send`, and `Sync` for `DefaultExecutor` (#463, #472).
+* Fix race in `ParkThread` (#507).
+* Handle recursive calls into `DefaultExecutor` (#473).
+
+# 0.1.2 (March 30, 2018)
+
+* Implement `Unpark` for `Box<Unpark>`.
+
+# 0.1.1 (March 22, 2018)
+
+* Optionally support futures 0.2.
+
+# 0.1.0 (March 09, 2018)
+
+* Initial release
diff --git a/third_party/rust/tokio-executor/Cargo.toml b/third_party/rust/tokio-executor/Cargo.toml
new file mode 100644
index 0000000000..37579636fb
--- /dev/null
+++ b/third_party/rust/tokio-executor/Cargo.toml
@@ -0,0 +1,30 @@
+# THIS FILE IS AUTOMATICALLY GENERATED BY CARGO
+#
+# When uploading crates to the registry Cargo will automatically
+# "normalize" Cargo.toml files for maximal compatibility
+# with all versions of Cargo and also rewrite `path` dependencies
+# to registry (e.g., crates.io) dependencies
+#
+# If you believe there's an error in this file please file an
+# issue against the rust-lang/cargo repository. If you're
+# editing this file be aware that the upstream Cargo.toml
+# will likely look very different (and much more reasonable)
+
+[package]
+name = "tokio-executor"
+version = "0.1.9"
+authors = ["Carl Lerche <me@carllerche.com>"]
+description = "Future execution primitives\n"
+homepage = "https://github.com/tokio-rs/tokio"
+documentation = "https://docs.rs/tokio-executor/0.1.9/tokio_executor"
+keywords = ["futures", "tokio"]
+categories = ["concurrency", "asynchronous"]
+license = "MIT"
+repository = "https://github.com/tokio-rs/tokio"
+[dependencies.crossbeam-utils]
+version = "0.6.2"
+
+[dependencies.futures]
+version = "0.1.19"
+[dev-dependencies.tokio]
+version = "0.1.18"
diff --git a/third_party/rust/tokio-executor/LICENSE b/third_party/rust/tokio-executor/LICENSE
new file mode 100644
index 0000000000..cdb28b4b56
--- /dev/null
+++ b/third_party/rust/tokio-executor/LICENSE
@@ -0,0 +1,25 @@
+Copyright (c) 2019 Tokio Contributors
+
+Permission is hereby granted, free of charge, to any
+person obtaining a copy of this software and associated
+documentation files (the "Software"), to deal in the
+Software without restriction, including without
+limitation the rights to use, copy, modify, merge,
+publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software
+is furnished to do so, subject to the following
+conditions:
+
+The above copyright notice and this permission notice
+shall be included in all copies or substantial portions
+of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
+ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
+TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
+PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
+SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
+IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/third_party/rust/tokio-executor/README.md b/third_party/rust/tokio-executor/README.md
new file mode 100644
index 0000000000..1a996107aa
--- /dev/null
+++ b/third_party/rust/tokio-executor/README.md
@@ -0,0 +1,47 @@
+# tokio-executor
+
+Task execution related traits and utilities.
+
+[Documentation](https://docs.rs/tokio-executor/0.1.9/tokio_executor)
+
+## Overview
+
+In the Tokio execution model, futures are lazy. When a future is created, no
+work is performed. In order for the work defined by the future to happen, the
+future must be submitted to an executor. A future that is submitted to an
+executor is called a "task".
+
+The executor is responsible for ensuring that [`Future::poll`] is called
+whenever the task is [notified]. Notification happens when the internal state of
+a task transitions from "not ready" to ready. For example, a socket might have
+received data and a call to `read` will now be able to succeed.
+
+This crate provides traits and utilities that are necessary for building an
+executor, including:
+
+* The [`Executor`] trait describes the API for spawning a future onto an
+ executor.
+
+* [`enter`] marks that the current thread is entering an execution
+ context. This prevents a second executor from accidentally starting from
+ within the context of one that is already running.
+
+* [`DefaultExecutor`] spawns tasks onto the default executor for the current
+ context.
+
+* [`Park`] abstracts over blocking and unblocking the current thread.
+
+[`Executor`]: https://docs.rs/tokio-executor/0.1.9/tokio_executor/trait.Executor.html
+[`enter`]: https://docs.rs/tokio-executor/0.1.9/tokio_executor/fn.enter.html
+[`DefaultExecutor`]: https://docs.rs/tokio-executor/0.1.9/tokio_executor/struct.DefaultExecutor.html
+[`Park`]: https://docs.rs/tokio-executor/0.1.9/tokio_executor/park/trait.Park.html
+
+## License
+
+This project is licensed under the [MIT license](LICENSE).
+
+### Contribution
+
+Unless you explicitly state otherwise, any contribution intentionally submitted
+for inclusion in Tokio by you, shall be licensed as MIT, without any additional
+terms or conditions.
diff --git a/third_party/rust/tokio-executor/src/enter.rs b/third_party/rust/tokio-executor/src/enter.rs
new file mode 100644
index 0000000000..1f8ef1a701
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/enter.rs
@@ -0,0 +1,164 @@
+use std::cell::Cell;
+use std::error::Error;
+use std::fmt;
+use std::prelude::v1::*;
+
+use futures::{self, Future};
+
+thread_local!(static ENTERED: Cell<bool> = Cell::new(false));
+
+/// Represents an executor context.
+///
+/// For more details, see [`enter` documentation](fn.enter.html)
+pub struct Enter {
+ on_exit: Vec<Box<dyn Callback>>,
+ permanent: bool,
+}
+
+/// An error returned by `enter` if an execution scope has already been
+/// entered.
+pub struct EnterError {
+ _a: (),
+}
+
+impl fmt::Debug for EnterError {
+ fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
+ f.debug_struct("EnterError")
+ .field("reason", &self.description())
+ .finish()
+ }
+}
+
+impl fmt::Display for EnterError {
+ fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
+ write!(fmt, "{}", self.description())
+ }
+}
+
+impl Error for EnterError {
+ fn description(&self) -> &str {
+ "attempted to run an executor while another executor is already running"
+ }
+}
+
+/// Marks the current thread as being within the dynamic extent of an
+/// executor.
+///
+/// Executor implementations should call this function before blocking the
+/// thread. If `None` is returned, the executor should fail by panicking or
+/// taking some other action without blocking the current thread. This prevents
+/// deadlocks due to multiple executors competing for the same thread.
+///
+/// # Error
+///
+/// Returns an error if the current thread is already marked
+pub fn enter() -> Result<Enter, EnterError> {
+ ENTERED.with(|c| {
+ if c.get() {
+ Err(EnterError { _a: () })
+ } else {
+ c.set(true);
+
+ Ok(Enter {
+ on_exit: Vec::new(),
+ permanent: false,
+ })
+ }
+ })
+}
+
+// Forces the current "entered" state to be cleared while the closure
+// is executed.
+//
+// # Warning
+//
+// This is hidden for a reason. Do not use without fully understanding
+// executors. Misuing can easily cause your program to deadlock.
+#[doc(hidden)]
+pub fn exit<F: FnOnce() -> R, R>(f: F) -> R {
+ // Reset in case the closure panics
+ struct Reset;
+ impl Drop for Reset {
+ fn drop(&mut self) {
+ ENTERED.with(|c| {
+ c.set(true);
+ });
+ }
+ }
+
+ ENTERED.with(|c| {
+ debug_assert!(c.get());
+ c.set(false);
+ });
+
+ let reset = Reset;
+ let ret = f();
+ ::std::mem::forget(reset);
+
+ ENTERED.with(|c| {
+ assert!(!c.get(), "closure claimed permanent executor");
+ c.set(true);
+ });
+
+ ret
+}
+
+impl Enter {
+ /// Register a callback to be invoked if and when the thread
+ /// ceased to act as an executor.
+ pub fn on_exit<F>(&mut self, f: F)
+ where
+ F: FnOnce() + 'static,
+ {
+ self.on_exit.push(Box::new(f));
+ }
+
+ /// Treat the remainder of execution on this thread as part of an
+ /// executor; used mostly for thread pool worker threads.
+ ///
+ /// All registered `on_exit` callbacks are *dropped* without being
+ /// invoked.
+ pub fn make_permanent(mut self) {
+ self.permanent = true;
+ }
+
+ /// Blocks the thread on the specified future, returning the value with
+ /// which that future completes.
+ pub fn block_on<F: Future>(&mut self, f: F) -> Result<F::Item, F::Error> {
+ futures::executor::spawn(f).wait_future()
+ }
+}
+
+impl fmt::Debug for Enter {
+ fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
+ f.debug_struct("Enter").finish()
+ }
+}
+
+impl Drop for Enter {
+ fn drop(&mut self) {
+ ENTERED.with(|c| {
+ assert!(c.get());
+
+ if self.permanent {
+ return;
+ }
+
+ for callback in self.on_exit.drain(..) {
+ callback.call();
+ }
+
+ c.set(false);
+ });
+ }
+}
+
+trait Callback: 'static {
+ fn call(self: Box<Self>);
+}
+
+impl<F: FnOnce() + 'static> Callback for F {
+ fn call(self: Box<Self>) {
+ (*self)()
+ }
+}
diff --git a/third_party/rust/tokio-executor/src/error.rs b/third_party/rust/tokio-executor/src/error.rs
new file mode 100644
index 0000000000..579239f089
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/error.rs
@@ -0,0 +1,50 @@
+use std::error::Error;
+use std::fmt;
+
+/// Errors returned by `Executor::spawn`.
+///
+/// Spawn errors should represent relatively rare scenarios. Currently, the two
+/// scenarios represented by `SpawnError` are:
+///
+/// * An executor being at capacity or full. As such, the executor is not able
+/// to accept a new future. This error state is expected to be transient.
+/// * An executor has been shutdown and can no longer accept new futures. This
+/// error state is expected to be permanent.
+#[derive(Debug)]
+pub struct SpawnError {
+ is_shutdown: bool,
+}
+
+impl SpawnError {
+ /// Return a new `SpawnError` reflecting a shutdown executor failure.
+ pub fn shutdown() -> Self {
+ SpawnError { is_shutdown: true }
+ }
+
+ /// Return a new `SpawnError` reflecting an executor at capacity failure.
+ pub fn at_capacity() -> Self {
+ SpawnError { is_shutdown: false }
+ }
+
+ /// Returns `true` if the error reflects a shutdown executor failure.
+ pub fn is_shutdown(&self) -> bool {
+ self.is_shutdown
+ }
+
+ /// Returns `true` if the error reflects an executor at capacity failure.
+ pub fn is_at_capacity(&self) -> bool {
+ !self.is_shutdown
+ }
+}
+
+impl fmt::Display for SpawnError {
+ fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
+ write!(fmt, "{}", self.description())
+ }
+}
+
+impl Error for SpawnError {
+ fn description(&self) -> &str {
+ "attempted to spawn task while the executor is at capacity or shut down"
+ }
+}
diff --git a/third_party/rust/tokio-executor/src/executor.rs b/third_party/rust/tokio-executor/src/executor.rs
new file mode 100644
index 0000000000..2a8abc6824
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/executor.rs
@@ -0,0 +1,151 @@
+use futures::Future;
+use SpawnError;
+
+/// A value that executes futures.
+///
+/// The [`spawn`] function is used to submit a future to an executor. Once
+/// submitted, the executor takes ownership of the future and becomes
+/// responsible for driving the future to completion.
+///
+/// The strategy employed by the executor to handle the future is less defined
+/// and is left up to the `Executor` implementation. The `Executor` instance is
+/// expected to call [`poll`] on the future once it has been notified, however
+/// the "when" and "how" can vary greatly.
+///
+/// For example, the executor might be a thread pool, in which case a set of
+/// threads have already been spawned up and the future is inserted into a
+/// queue. A thread will acquire the future and poll it.
+///
+/// The `Executor` trait is only for futures that **are** `Send`. These are most
+/// common. There currently is no trait that describes executors that operate
+/// entirely on the current thread (i.e., are able to spawn futures that are not
+/// `Send`). Note that single threaded executors can still implement `Executor`,
+/// but only futures that are `Send` can be spawned via the trait.
+///
+/// This trait is primarily intended to implemented by executors and used to
+/// back `tokio::spawn`. Libraries and applications **may** use this trait to
+/// bound generics, but doing so will limit usage to futures that implement
+/// `Send`. Instead, libraries and applications are recommended to use
+/// [`TypedExecutor`] as a bound.
+///
+/// # Errors
+///
+/// The [`spawn`] function returns `Result` with an error type of `SpawnError`.
+/// This error type represents the reason that the executor was unable to spawn
+/// the future. The two current represented scenarios are:
+///
+/// * An executor being at capacity or full. As such, the executor is not able
+/// to accept a new future. This error state is expected to be transient.
+/// * An executor has been shutdown and can no longer accept new futures. This
+/// error state is expected to be permanent.
+///
+/// If a caller encounters an at capacity error, the caller should try to shed
+/// load. This can be as simple as dropping the future that was spawned.
+///
+/// If the caller encounters a shutdown error, the caller should attempt to
+/// gracefully shutdown.
+///
+/// # Examples
+///
+/// ```rust
+/// # extern crate futures;
+/// # extern crate tokio_executor;
+/// # use tokio_executor::Executor;
+/// # fn docs(my_executor: &mut Executor) {
+/// use futures::future::lazy;
+/// my_executor.spawn(Box::new(lazy(|| {
+/// println!("running on the executor");
+/// Ok(())
+/// }))).unwrap();
+/// # }
+/// # fn main() {}
+/// ```
+///
+/// [`spawn`]: #tymethod.spawn
+/// [`poll`]: https://docs.rs/futures/0.1/futures/future/trait.Future.html#tymethod.poll
+/// [`TypedExecutor`]: ../trait.TypedExecutor.html
+pub trait Executor {
+ /// Spawns a future object to run on this executor.
+ ///
+ /// `future` is passed to the executor, which will begin running it. The
+ /// future may run on the current thread or another thread at the discretion
+ /// of the `Executor` implementation.
+ ///
+ /// # Panics
+ ///
+ /// Implementations are encouraged to avoid panics. However, panics are
+ /// permitted and the caller should check the implementation specific
+ /// documentation for more details on possible panics.
+ ///
+ /// # Examples
+ ///
+ /// ```rust
+ /// # extern crate futures;
+ /// # extern crate tokio_executor;
+ /// # use tokio_executor::Executor;
+ /// # fn docs(my_executor: &mut Executor) {
+ /// use futures::future::lazy;
+ /// my_executor.spawn(Box::new(lazy(|| {
+ /// println!("running on the executor");
+ /// Ok(())
+ /// }))).unwrap();
+ /// # }
+ /// # fn main() {}
+ /// ```
+ fn spawn(
+ &mut self,
+ future: Box<dyn Future<Item = (), Error = ()> + Send>,
+ ) -> Result<(), SpawnError>;
+
+ /// Provides a best effort **hint** to whether or not `spawn` will succeed.
+ ///
+ /// This function may return both false positives **and** false negatives.
+ /// If `status` returns `Ok`, then a call to `spawn` will *probably*
+ /// succeed, but may fail. If `status` returns `Err`, a call to `spawn` will
+ /// *probably* fail, but may succeed.
+ ///
+ /// This allows a caller to avoid creating the task if the call to `spawn`
+ /// has a high likelihood of failing.
+ ///
+ /// # Panics
+ ///
+ /// This function must not panic. Implementers must ensure that panics do
+ /// not happen.
+ ///
+ /// # Examples
+ ///
+ /// ```rust
+ /// # extern crate futures;
+ /// # extern crate tokio_executor;
+ /// # use tokio_executor::Executor;
+ /// # fn docs(my_executor: &mut Executor) {
+ /// use futures::future::lazy;
+ ///
+ /// if my_executor.status().is_ok() {
+ /// my_executor.spawn(Box::new(lazy(|| {
+ /// println!("running on the executor");
+ /// Ok(())
+ /// }))).unwrap();
+ /// } else {
+ /// println!("the executor is not in a good state");
+ /// }
+ /// # }
+ /// # fn main() {}
+ /// ```
+ fn status(&self) -> Result<(), SpawnError> {
+ Ok(())
+ }
+}
+
+impl<E: Executor + ?Sized> Executor for Box<E> {
+ fn spawn(
+ &mut self,
+ future: Box<dyn Future<Item = (), Error = ()> + Send>,
+ ) -> Result<(), SpawnError> {
+ (**self).spawn(future)
+ }
+
+ fn status(&self) -> Result<(), SpawnError> {
+ (**self).status()
+ }
+}
diff --git a/third_party/rust/tokio-executor/src/global.rs b/third_party/rust/tokio-executor/src/global.rs
new file mode 100644
index 0000000000..5012276088
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/global.rs
@@ -0,0 +1,290 @@
+use super::{Enter, Executor, SpawnError};
+
+use futures::{future, Future};
+
+use std::cell::Cell;
+
+/// Executes futures on the default executor for the current execution context.
+///
+/// `DefaultExecutor` implements `Executor` and can be used to spawn futures
+/// without referencing a specific executor.
+///
+/// When an executor starts, it sets the `DefaultExecutor` handle to point to an
+/// executor (usually itself) that is used to spawn new tasks.
+///
+/// The current `DefaultExecutor` reference is tracked using a thread-local
+/// variable and is set using `tokio_executor::with_default`
+#[derive(Debug, Clone)]
+pub struct DefaultExecutor {
+ _dummy: (),
+}
+
+/// Ensures that the executor is removed from the thread-local context
+/// when leaving the scope. This handles cases that involve panicking.
+#[derive(Debug)]
+pub struct DefaultGuard {
+ _p: (),
+}
+
+impl DefaultExecutor {
+ /// Returns a handle to the default executor for the current context.
+ ///
+ /// Futures may be spawned onto the default executor using this handle.
+ ///
+ /// The returned handle will reference whichever executor is configured as
+ /// the default **at the time `spawn` is called**. This enables
+ /// `DefaultExecutor::current()` to be called before an execution context is
+ /// setup, then passed **into** an execution context before it is used.
+ ///
+ /// This is also true for sending the handle across threads, so calling
+ /// `DefaultExecutor::current()` on thread A and then sending the result to
+ /// thread B will _not_ reference the default executor that was set on thread A.
+ pub fn current() -> DefaultExecutor {
+ DefaultExecutor { _dummy: () }
+ }
+
+ #[inline]
+ fn with_current<F: FnOnce(&mut dyn Executor) -> R, R>(f: F) -> Option<R> {
+ EXECUTOR.with(
+ |current_executor| match current_executor.replace(State::Active) {
+ State::Ready(executor_ptr) => {
+ let executor = unsafe { &mut *executor_ptr };
+ let result = f(executor);
+ current_executor.set(State::Ready(executor_ptr));
+ Some(result)
+ }
+ State::Empty | State::Active => None,
+ },
+ )
+ }
+}
+
+#[derive(Clone, Copy)]
+enum State {
+ // default executor not defined
+ Empty,
+ // default executor is defined and ready to be used
+ Ready(*mut dyn Executor),
+ // default executor is currently active (used to detect recursive calls)
+ Active,
+}
+
+thread_local! {
+ /// Thread-local tracking the current executor
+ static EXECUTOR: Cell<State> = Cell::new(State::Empty)
+}
+
+// ===== impl DefaultExecutor =====
+
+impl super::Executor for DefaultExecutor {
+ fn spawn(
+ &mut self,
+ future: Box<dyn Future<Item = (), Error = ()> + Send>,
+ ) -> Result<(), SpawnError> {
+ DefaultExecutor::with_current(|executor| executor.spawn(future))
+ .unwrap_or_else(|| Err(SpawnError::shutdown()))
+ }
+
+ fn status(&self) -> Result<(), SpawnError> {
+ DefaultExecutor::with_current(|executor| executor.status())
+ .unwrap_or_else(|| Err(SpawnError::shutdown()))
+ }
+}
+
+impl<T> super::TypedExecutor<T> for DefaultExecutor
+where
+ T: Future<Item = (), Error = ()> + Send + 'static,
+{
+ fn spawn(&mut self, future: T) -> Result<(), SpawnError> {
+ super::Executor::spawn(self, Box::new(future))
+ }
+
+ fn status(&self) -> Result<(), SpawnError> {
+ super::Executor::status(self)
+ }
+}
+
+impl<T> future::Executor<T> for DefaultExecutor
+where
+ T: Future<Item = (), Error = ()> + Send + 'static,
+{
+ fn execute(&self, future: T) -> Result<(), future::ExecuteError<T>> {
+ if let Err(e) = super::Executor::status(self) {
+ let kind = if e.is_at_capacity() {
+ future::ExecuteErrorKind::NoCapacity
+ } else {
+ future::ExecuteErrorKind::Shutdown
+ };
+
+ return Err(future::ExecuteError::new(kind, future));
+ }
+
+ let _ = DefaultExecutor::with_current(|executor| executor.spawn(Box::new(future)));
+ Ok(())
+ }
+}
+
+// ===== global spawn fns =====
+
+/// Submits a future for execution on the default executor -- usually a
+/// threadpool.
+///
+/// Futures are lazy constructs. When they are defined, no work happens. In
+/// order for the logic defined by the future to be run, the future must be
+/// spawned on an executor. This function is the easiest way to do so.
+///
+/// This function must be called from an execution context, i.e. from a future
+/// that has been already spawned onto an executor.
+///
+/// Once spawned, the future will execute. The details of how that happens is
+/// left up to the executor instance. If the executor is a thread pool, the
+/// future will be pushed onto a queue that a worker thread polls from. If the
+/// executor is a "current thread" executor, the future might be polled
+/// immediately from within the call to `spawn` or it might be pushed onto an
+/// internal queue.
+///
+/// # Panics
+///
+/// This function will panic if the default executor is not set or if spawning
+/// onto the default executor returns an error. To avoid the panic, use the
+/// `DefaultExecutor` handle directly.
+///
+/// # Examples
+///
+/// ```rust
+/// # extern crate futures;
+/// # extern crate tokio_executor;
+/// # use tokio_executor::spawn;
+/// # pub fn dox() {
+/// use futures::future::lazy;
+///
+/// spawn(lazy(|| {
+/// println!("running on the default executor");
+/// Ok(())
+/// }));
+/// # }
+/// # pub fn main() {}
+/// ```
+pub fn spawn<T>(future: T)
+where
+ T: Future<Item = (), Error = ()> + Send + 'static,
+{
+ DefaultExecutor::current().spawn(Box::new(future)).unwrap()
+}
+
+/// Set the default executor for the duration of the closure
+///
+/// # Panics
+///
+/// This function panics if there already is a default executor set.
+pub fn with_default<T, F, R>(executor: &mut T, enter: &mut Enter, f: F) -> R
+where
+ T: Executor,
+ F: FnOnce(&mut Enter) -> R,
+{
+ unsafe fn hide_lt<'a>(p: *mut (dyn Executor + 'a)) -> *mut (dyn Executor + 'static) {
+ use std::mem;
+ mem::transmute(p)
+ }
+
+ EXECUTOR.with(|cell| {
+ match cell.get() {
+ State::Ready(_) | State::Active => {
+ panic!("default executor already set for execution context")
+ }
+ _ => {}
+ }
+
+ // Ensure that the executor is removed from the thread-local context
+ // when leaving the scope. This handles cases that involve panicking.
+ struct Reset<'a>(&'a Cell<State>);
+
+ impl<'a> Drop for Reset<'a> {
+ fn drop(&mut self) {
+ self.0.set(State::Empty);
+ }
+ }
+
+ let _reset = Reset(cell);
+
+ // While scary, this is safe. The function takes a
+ // `&mut Executor`, which guarantees that the reference lives for the
+ // duration of `with_default`.
+ //
+ // Because we are always clearing the TLS value at the end of the
+ // function, we can cast the reference to 'static which thread-local
+ // cells require.
+ let executor = unsafe { hide_lt(executor as &mut _ as *mut _) };
+
+ cell.set(State::Ready(executor));
+
+ f(enter)
+ })
+}
+
+/// Sets `executor` as the default executor, returning a guard that unsets it when
+/// dropped.
+///
+/// # Panics
+///
+/// This function panics if there already is a default executor set.
+pub fn set_default<T>(executor: T) -> DefaultGuard
+where
+ T: Executor + 'static,
+{
+ EXECUTOR.with(|cell| {
+ match cell.get() {
+ State::Ready(_) | State::Active => {
+ panic!("default executor already set for execution context")
+ }
+ _ => {}
+ }
+
+ // Ensure that the executor will outlive the call to set_default, even
+ // if the drop guard is never dropped due to calls to `mem::forget` or
+ // similar.
+ let executor = Box::new(executor);
+
+ cell.set(State::Ready(Box::into_raw(executor)));
+ });
+
+ DefaultGuard { _p: () }
+}
+
+impl Drop for DefaultGuard {
+ fn drop(&mut self) {
+ let _ = EXECUTOR.try_with(|cell| {
+ if let State::Ready(prev) = cell.replace(State::Empty) {
+ // drop the previous executor.
+ unsafe {
+ let prev = Box::from_raw(prev);
+ drop(prev);
+ };
+ }
+ });
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::{with_default, DefaultExecutor, Executor};
+
+ #[test]
+ fn default_executor_is_send_and_sync() {
+ fn assert_send_sync<T: Send + Sync>() {}
+
+ assert_send_sync::<DefaultExecutor>();
+ }
+
+ #[test]
+ fn nested_default_executor_status() {
+ let mut enter = super::super::enter().unwrap();
+ let mut executor = DefaultExecutor::current();
+
+ let result = with_default(&mut executor, &mut enter, |_| {
+ DefaultExecutor::current().status()
+ });
+
+ assert!(result.err().unwrap().is_shutdown())
+ }
+}
diff --git a/third_party/rust/tokio-executor/src/lib.rs b/third_party/rust/tokio-executor/src/lib.rs
new file mode 100644
index 0000000000..67aa4895fc
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/lib.rs
@@ -0,0 +1,68 @@
+#![deny(missing_docs, missing_debug_implementations)]
+#![doc(html_root_url = "https://docs.rs/tokio-executor/0.1.9")]
+
+//! Task execution related traits and utilities.
+//!
+//! In the Tokio execution model, futures are lazy. When a future is created, no
+//! work is performed. In order for the work defined by the future to happen,
+//! the future must be submitted to an executor. A future that is submitted to
+//! an executor is called a "task".
+//!
+//! The executor is responsible for ensuring that [`Future::poll`] is called
+//! whenever the task is notified. Notification happens when the internal
+//! state of a task transitions from *not ready* to *ready*. For example, a
+//! socket might have received data and a call to `read` will now be able to
+//! succeed.
+//!
+//! This crate provides traits and utilities that are necessary for building an
+//! executor, including:
+//!
+//! * The [`Executor`] trait spawns future object onto an executor.
+//!
+//! * The [`TypedExecutor`] trait spawns futures of a specific type onto an
+//! executor. This is used to be generic over executors that spawn futures
+//! that are either `Send` or `!Send` or implement executors that apply to
+//! specific futures.
+//!
+//! * [`enter`] marks that the current thread is entering an execution
+//! context. This prevents a second executor from accidentally starting from
+//! within the context of one that is already running.
+//!
+//! * [`DefaultExecutor`] spawns tasks onto the default executor for the current
+//! context.
+//!
+//! * [`Park`] abstracts over blocking and unblocking the current thread.
+//!
+//! # Implementing an executor
+//!
+//! Executors should always implement `TypedExecutor`. This usually is the bound
+//! that applications and libraries will use when generic over an executor. See
+//! the [trait documentation][`TypedExecutor`] for more details.
+//!
+//! If the executor is able to spawn all futures that are `Send`, then the
+//! executor should also implement the `Executor` trait. This trait is rarely
+//! used directly by applications and libraries. Instead, `tokio::spawn` is
+//! configured to dispatch to type that implements `Executor`.
+//!
+//! [`Executor`]: trait.Executor.html
+//! [`TypedExecutor`]: trait.TypedExecutor.html
+//! [`enter`]: fn.enter.html
+//! [`DefaultExecutor`]: struct.DefaultExecutor.html
+//! [`Park`]: park/index.html
+//! [`Future::poll`]: https://docs.rs/futures/0.1/futures/future/trait.Future.html#tymethod.poll
+
+extern crate crossbeam_utils;
+extern crate futures;
+
+mod enter;
+mod error;
+mod executor;
+mod global;
+pub mod park;
+mod typed;
+
+pub use enter::{enter, exit, Enter, EnterError};
+pub use error::SpawnError;
+pub use executor::Executor;
+pub use global::{set_default, spawn, with_default, DefaultExecutor, DefaultGuard};
+pub use typed::TypedExecutor;
diff --git a/third_party/rust/tokio-executor/src/park.rs b/third_party/rust/tokio-executor/src/park.rs
new file mode 100644
index 0000000000..8626f089a1
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/park.rs
@@ -0,0 +1,226 @@
+//! Abstraction over blocking and unblocking the current thread.
+//!
+//! Provides an abstraction over blocking the current thread. This is similar to
+//! the park / unpark constructs provided by [`std`] but made generic. This
+//! allows embedding custom functionality to perform when the thread is blocked.
+//!
+//! A blocked [`Park`][p] instance is unblocked by calling [`unpark`] on its
+//! [`Unpark`][up] handle.
+//!
+//! The [`ParkThread`] struct implements [`Park`][p] using
+//! [`thread::park`][`std`] to put the thread to sleep. The Tokio reactor also
+//! implements park, but uses [`mio::Poll`][mio] to block the thread instead.
+//!
+//! The [`Park`][p] trait is composable. A timer implementation might decorate a
+//! [`Park`][p] implementation by checking if any timeouts have elapsed after
+//! the inner [`Park`][p] implementation unblocks.
+//!
+//! # Model
+//!
+//! Conceptually, each [`Park`][p] instance has an associated token, which is
+//! initially not present:
+//!
+//! * The [`park`] method blocks the current thread unless or until the token
+//! is available, at which point it atomically consumes the token.
+//! * The [`unpark`] method atomically makes the token available if it wasn't
+//! already.
+//!
+//! Some things to note:
+//!
+//! * If [`unpark`] is called before [`park`], the next call to [`park`] will
+//! **not** block the thread.
+//! * **Spurious** wakeups are permitted, i.e., the [`park`] method may unblock
+//! even if [`unpark`] was not called.
+//! * [`park_timeout`] does the same as [`park`] but allows specifying a maximum
+//! time to block the thread for.
+//!
+//! [`std`]: https://doc.rust-lang.org/std/thread/fn.park.html
+//! [`thread::park`]: https://doc.rust-lang.org/std/thread/fn.park.html
+//! [`ParkThread`]: struct.ParkThread.html
+//! [p]: trait.Park.html
+//! [`park`]: trait.Park.html#tymethod.park
+//! [`park_timeout`]: trait.Park.html#tymethod.park_timeout
+//! [`unpark`]: trait.Unpark.html#tymethod.unpark
+//! [up]: trait.Unpark.html
+//! [mio]: https://docs.rs/mio/0.6/mio/struct.Poll.html
+
+use std::marker::PhantomData;
+use std::rc::Rc;
+use std::sync::Arc;
+use std::time::Duration;
+
+use crossbeam_utils::sync::{Parker, Unparker};
+
+/// Block the current thread.
+///
+/// See [module documentation][mod] for more details.
+///
+/// [mod]: ../index.html
+pub trait Park {
+ /// Unpark handle type for the `Park` implementation.
+ type Unpark: Unpark;
+
+ /// Error returned by `park`
+ type Error;
+
+ /// Get a new `Unpark` handle associated with this `Park` instance.
+ fn unpark(&self) -> Self::Unpark;
+
+ /// Block the current thread unless or until the token is available.
+ ///
+ /// A call to `park` does not guarantee that the thread will remain blocked
+ /// forever, and callers should be prepared for this possibility. This
+ /// function may wakeup spuriously for any reason.
+ ///
+ /// See [module documentation][mod] for more details.
+ ///
+ /// # Panics
+ ///
+ /// This function **should** not panic, but ultimately, panics are left as
+ /// an implementation detail. Refer to the documentation for the specific
+ /// `Park` implementation
+ ///
+ /// [mod]: ../index.html
+ fn park(&mut self) -> Result<(), Self::Error>;
+
+ /// Park the current thread for at most `duration`.
+ ///
+ /// This function is the same as `park` but allows specifying a maximum time
+ /// to block the thread for.
+ ///
+ /// Same as `park`, there is no guarantee that the thread will remain
+ /// blocked for any amount of time. Spurious wakeups are permitted for any
+ /// reason.
+ ///
+ /// See [module documentation][mod] for more details.
+ ///
+ /// # Panics
+ ///
+ /// This function **should** not panic, but ultimately, panics are left as
+ /// an implementation detail. Refer to the documentation for the specific
+ /// `Park` implementation
+ ///
+ /// [mod]: ../index.html
+ fn park_timeout(&mut self, duration: Duration) -> Result<(), Self::Error>;
+}
+
+/// Unblock a thread blocked by the associated [`Park`] instance.
+///
+/// See [module documentation][mod] for more details.
+///
+/// [mod]: ../index.html
+/// [`Park`]: trait.Park.html
+pub trait Unpark: Sync + Send + 'static {
+ /// Unblock a thread that is blocked by the associated `Park` handle.
+ ///
+ /// Calling `unpark` atomically makes available the unpark token, if it is
+ /// not already available.
+ ///
+ /// See [module documentation][mod] for more details.
+ ///
+ /// # Panics
+ ///
+ /// This function **should** not panic, but ultimately, panics are left as
+ /// an implementation detail. Refer to the documentation for the specific
+ /// `Unpark` implementation
+ ///
+ /// [mod]: ../index.html
+ fn unpark(&self);
+}
+
+impl Unpark for Box<dyn Unpark> {
+ fn unpark(&self) {
+ (**self).unpark()
+ }
+}
+
+impl Unpark for Arc<dyn Unpark> {
+ fn unpark(&self) {
+ (**self).unpark()
+ }
+}
+
+/// Blocks the current thread using a condition variable.
+///
+/// Implements the [`Park`] functionality by using a condition variable. An
+/// atomic variable is also used to avoid using the condition variable if
+/// possible.
+///
+/// The condition variable is cached in a thread-local variable and is shared
+/// across all `ParkThread` instances created on the same thread. This also
+/// means that an instance of `ParkThread` might be unblocked by a handle
+/// associated with a different `ParkThread` instance.
+#[derive(Debug)]
+pub struct ParkThread {
+ _anchor: PhantomData<Rc<()>>,
+}
+
+/// Error returned by [`ParkThread`]
+///
+/// This currently is never returned, but might at some point in the future.
+///
+/// [`ParkThread`]: struct.ParkThread.html
+#[derive(Debug)]
+pub struct ParkError {
+ _p: (),
+}
+
+/// Unblocks a thread that was blocked by `ParkThread`.
+#[derive(Clone, Debug)]
+pub struct UnparkThread {
+ inner: Unparker,
+}
+
+thread_local! {
+ static CURRENT_PARKER: Parker = Parker::new();
+}
+
+// ===== impl ParkThread =====
+
+impl ParkThread {
+ /// Create a new `ParkThread` handle for the current thread.
+ ///
+ /// This type cannot be moved to other threads, so it should be created on
+ /// the thread that the caller intends to park.
+ pub fn new() -> ParkThread {
+ ParkThread {
+ _anchor: PhantomData,
+ }
+ }
+
+ /// Get a reference to the `ParkThread` handle for this thread.
+ fn with_current<F, R>(&self, f: F) -> R
+ where
+ F: FnOnce(&Parker) -> R,
+ {
+ CURRENT_PARKER.with(|inner| f(inner))
+ }
+}
+
+impl Park for ParkThread {
+ type Unpark = UnparkThread;
+ type Error = ParkError;
+
+ fn unpark(&self) -> Self::Unpark {
+ let inner = self.with_current(|inner| inner.unparker().clone());
+ UnparkThread { inner }
+ }
+
+ fn park(&mut self) -> Result<(), Self::Error> {
+ self.with_current(|inner| inner.park());
+ Ok(())
+ }
+
+ fn park_timeout(&mut self, duration: Duration) -> Result<(), Self::Error> {
+ self.with_current(|inner| inner.park_timeout(duration));
+ Ok(())
+ }
+}
+
+// ===== impl UnparkThread =====
+
+impl Unpark for UnparkThread {
+ fn unpark(&self) {
+ self.inner.unpark();
+ }
+}
diff --git a/third_party/rust/tokio-executor/src/typed.rs b/third_party/rust/tokio-executor/src/typed.rs
new file mode 100644
index 0000000000..22edf29d12
--- /dev/null
+++ b/third_party/rust/tokio-executor/src/typed.rs
@@ -0,0 +1,181 @@
+use SpawnError;
+
+/// A value that spawns futures of a specific type.
+///
+/// The trait is generic over `T`: the type of future that can be spawened. This
+/// is useful for implementing an executor that is only able to spawn a specific
+/// type of future.
+///
+/// The [`spawn`] function is used to submit the future to the executor. Once
+/// submitted, the executor takes ownership of the future and becomes
+/// responsible for driving the future to completion.
+///
+/// This trait is useful as a bound for applications and libraries in order to
+/// be generic over futures that are `Send` vs. `!Send`.
+///
+/// # Examples
+///
+/// Consider a function that provides an API for draining a `Stream` in the
+/// background. To do this, a task must be spawned to perform the draining. As
+/// such, the function takes a stream and an executor on which the background
+/// task is spawned.
+///
+/// ```rust
+/// #[macro_use]
+/// extern crate futures;
+/// extern crate tokio;
+///
+/// use futures::{Future, Stream, Poll};
+/// use tokio::executor::TypedExecutor;
+/// use tokio::sync::oneshot;
+///
+/// pub fn drain<T, E>(stream: T, executor: &mut E)
+/// -> impl Future<Item = (), Error = ()>
+/// where
+/// T: Stream,
+/// E: TypedExecutor<Drain<T>>
+/// {
+/// let (tx, rx) = oneshot::channel();
+///
+/// executor.spawn(Drain {
+/// stream,
+/// tx: Some(tx),
+/// }).unwrap();
+///
+/// rx.map_err(|_| ())
+/// }
+///
+/// // The background task
+/// pub struct Drain<T: Stream> {
+/// stream: T,
+/// tx: Option<oneshot::Sender<()>>,
+/// }
+///
+/// impl<T: Stream> Future for Drain<T> {
+/// type Item = ();
+/// type Error = ();
+///
+/// fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
+/// loop {
+/// let item = try_ready!(
+/// self.stream.poll()
+/// .map_err(|_| ())
+/// );
+///
+/// if item.is_none() { break; }
+/// }
+///
+/// self.tx.take().unwrap().send(()).map_err(|_| ());
+/// Ok(().into())
+/// }
+/// }
+/// # pub fn main() {}
+/// ```
+///
+/// By doing this, the `drain` fn can accept a stream that is `!Send` as long as
+/// the supplied executor is able to spawn `!Send` types.
+pub trait TypedExecutor<T> {
+ /// Spawns a future to run on this executor.
+ ///
+ /// `future` is passed to the executor, which will begin running it. The
+ /// executor takes ownership of the future and becomes responsible for
+ /// driving the future to completion.
+ ///
+ /// # Panics
+ ///
+ /// Implementations are encouraged to avoid panics. However, panics are
+ /// permitted and the caller should check the implementation specific
+ /// documentation for more details on possible panics.
+ ///
+ /// # Examples
+ ///
+ /// ```rust
+ /// # extern crate futures;
+ /// # extern crate tokio_executor;
+ /// # use tokio_executor::TypedExecutor;
+ /// # use futures::{Future, Poll};
+ /// fn example<T>(my_executor: &mut T)
+ /// where
+ /// T: TypedExecutor<MyFuture>,
+ /// {
+ /// my_executor.spawn(MyFuture).unwrap();
+ /// }
+ ///
+ /// struct MyFuture;
+ ///
+ /// impl Future for MyFuture {
+ /// type Item = ();
+ /// type Error = ();
+ ///
+ /// fn poll(&mut self) -> Poll<(), ()> {
+ /// println!("running on the executor");
+ /// Ok(().into())
+ /// }
+ /// }
+ /// # fn main() {}
+ /// ```
+ fn spawn(&mut self, future: T) -> Result<(), SpawnError>;
+
+ /// Provides a best effort **hint** to whether or not `spawn` will succeed.
+ ///
+ /// This function may return both false positives **and** false negatives.
+ /// If `status` returns `Ok`, then a call to `spawn` will *probably*
+ /// succeed, but may fail. If `status` returns `Err`, a call to `spawn` will
+ /// *probably* fail, but may succeed.
+ ///
+ /// This allows a caller to avoid creating the task if the call to `spawn`
+ /// has a high likelihood of failing.
+ ///
+ /// # Panics
+ ///
+ /// This function must not panic. Implementers must ensure that panics do
+ /// not happen.
+ ///
+ /// # Examples
+ ///
+ /// ```rust
+ /// # extern crate futures;
+ /// # extern crate tokio_executor;
+ /// # use tokio_executor::TypedExecutor;
+ /// # use futures::{Future, Poll};
+ /// fn example<T>(my_executor: &mut T)
+ /// where
+ /// T: TypedExecutor<MyFuture>,
+ /// {
+ /// if my_executor.status().is_ok() {
+ /// my_executor.spawn(MyFuture).unwrap();
+ /// } else {
+ /// println!("the executor is not in a good state");
+ /// }
+ /// }
+ ///
+ /// struct MyFuture;
+ ///
+ /// impl Future for MyFuture {
+ /// type Item = ();
+ /// type Error = ();
+ ///
+ /// fn poll(&mut self) -> Poll<(), ()> {
+ /// println!("running on the executor");
+ /// Ok(().into())
+ /// }
+ /// }
+ /// # fn main() {}
+ /// ```
+ fn status(&self) -> Result<(), SpawnError> {
+ Ok(())
+ }
+}
+
+impl<E, T> TypedExecutor<T> for Box<E>
+where
+ E: TypedExecutor<T>,
+{
+ fn spawn(&mut self, future: T) -> Result<(), SpawnError> {
+ (**self).spawn(future)
+ }
+
+ fn status(&self) -> Result<(), SpawnError> {
+ (**self).status()
+ }
+}
diff --git a/third_party/rust/tokio-executor/tests/executor.rs b/third_party/rust/tokio-executor/tests/executor.rs
new file mode 100644
index 0000000000..8d82c893cd
--- /dev/null
+++ b/third_party/rust/tokio-executor/tests/executor.rs
@@ -0,0 +1,29 @@
+extern crate futures;
+extern crate tokio_executor;
+
+use futures::{future::lazy, Future};
+use tokio_executor::DefaultExecutor;
+
+mod out_of_executor_context {
+ use super::*;
+ use tokio_executor::Executor;
+
+ fn test<F, E>(spawn: F)
+ where
+ F: Fn(Box<dyn Future<Item = (), Error = ()> + Send>) -> Result<(), E>,
+ {
+ let res = spawn(Box::new(lazy(|| Ok(()))));
+ assert!(res.is_err());
+ }
+
+ #[test]
+ fn spawn() {
+ test(|f| DefaultExecutor::current().spawn(f));
+ }
+
+ #[test]
+ fn execute() {
+ use futures::future::Executor as FuturesExecutor;
+ test(|f| DefaultExecutor::current().execute(f));
+ }
+}