From 26a029d407be480d791972afb5975cf62c9360a6 Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Fri, 19 Apr 2024 02:47:55 +0200 Subject: Adding upstream version 124.0.1. Signed-off-by: Daniel Baumann --- third_party/rust/threadbound/src/lib.rs | 143 ++++++++++++++++++++++++++++++++ 1 file changed, 143 insertions(+) create mode 100644 third_party/rust/threadbound/src/lib.rs (limited to 'third_party/rust/threadbound/src') diff --git a/third_party/rust/threadbound/src/lib.rs b/third_party/rust/threadbound/src/lib.rs new file mode 100644 index 0000000000..45a84f896f --- /dev/null +++ b/third_party/rust/threadbound/src/lib.rs @@ -0,0 +1,143 @@ +//! [![github]](https://github.com/dtolnay/threadbound) [![crates-io]](https://crates.io/crates/threadbound) [![docs-rs]](https://docs.rs/threadbound) +//! +//! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github +//! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust +//! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logo=docs.rs +//! +//!
+//! +//! [`ThreadBound`] is a wrapper that binds a value to its original thread. +//! The wrapper gets to be [`Sync`] and [`Send`] but only the original thread on +//! which the ThreadBound was constructed can retrieve the underlying value. +//! +//! [`ThreadBound`]: struct.ThreadBound.html +//! [`Sync`]: https://doc.rust-lang.org/std/marker/trait.Sync.html +//! [`Send`]: https://doc.rust-lang.org/std/marker/trait.Send.html +//! +//! # Example +//! +//! ``` +//! use std::marker::PhantomData; +//! use std::rc::Rc; +//! use std::sync::Arc; +//! use threadbound::ThreadBound; +//! +//! // Neither Send nor Sync. Maybe the index points into a +//! // thread-local interner. +//! #[derive(Copy, Clone)] +//! struct Span { +//! index: u32, +//! marker: PhantomData>, +//! } +//! +//! // Error types are always supposed to be Send and Sync. +//! // We can use ThreadBound to make it so. +//! struct Error { +//! span: ThreadBound, +//! message: String, +//! } +//! +//! fn main() { +//! let err = Error { +//! span: ThreadBound::new(Span { +//! index: 99, +//! marker: PhantomData, +//! }), +//! message: "fearless concurrency".to_owned(), +//! }; +//! +//! // Original thread can see the contents. +//! assert_eq!(err.span.get_ref().unwrap().index, 99); +//! +//! let err = Arc::new(err); +//! let err2 = err.clone(); +//! std::thread::spawn(move || { +//! // Other threads cannot get access. Maybe they use +//! // a default value or a different codepath. +//! assert!(err2.span.get_ref().is_none()); +//! }); +//! +//! // Original thread can still see the contents. +//! assert_eq!(err.span.get_ref().unwrap().index, 99); +//! } +//! ``` + +#![doc(html_root_url = "https://docs.rs/threadbound/0.1.5")] +#![allow(clippy::doc_markdown)] + +use std::fmt::{self, Debug}; +use std::thread::{self, ThreadId}; + +/// ThreadBound is a Sync-maker and Send-maker that allows accessing a value +/// of type T only from the original thread on which the ThreadBound was +/// constructed. +/// +/// Refer to the [crate-level documentation] for a usage example. +/// +/// [crate-level documentation]: index.html +pub struct ThreadBound { + value: T, + thread_id: ThreadId, +} + +unsafe impl Sync for ThreadBound {} + +// Send bound requires Copy, as otherwise Drop could run in the wrong place. +unsafe impl Send for ThreadBound {} + +impl ThreadBound { + /// Binds a value to the current thread. The wrapper can be sent around to + /// other threads, but no other threads will be able to access the + /// underlying value. + pub fn new(value: T) -> Self { + ThreadBound { + value, + thread_id: thread::current().id(), + } + } + + /// Accesses a reference to the underlying value if this is its original + /// thread, otherwise `None`. + pub fn get_ref(&self) -> Option<&T> { + if thread::current().id() == self.thread_id { + Some(&self.value) + } else { + None + } + } + + /// Accesses a mutable reference to the underlying value if this is its + /// original thread, otherwise `None`. + pub fn get_mut(&mut self) -> Option<&mut T> { + if thread::current().id() == self.thread_id { + Some(&mut self.value) + } else { + None + } + } + + /// Extracts ownership of the underlying value if this is its original + /// thread, otherwise `None`. + pub fn into_inner(self) -> Option { + if thread::current().id() == self.thread_id { + Some(self.value) + } else { + None + } + } +} + +impl Default for ThreadBound { + fn default() -> Self { + ThreadBound::new(Default::default()) + } +} + +impl Debug for ThreadBound { + fn fmt(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + match self.get_ref() { + Some(value) => Debug::fmt(value, formatter), + None => formatter.write_str("unknown"), + } + } +} -- cgit v1.2.3