summaryrefslogtreecommitdiffstats
path: root/third_party/rust/ouroboros/src/lib.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/ouroboros/src/lib.rs')
-rw-r--r--third_party/rust/ouroboros/src/lib.rs249
1 files changed, 249 insertions, 0 deletions
diff --git a/third_party/rust/ouroboros/src/lib.rs b/third_party/rust/ouroboros/src/lib.rs
new file mode 100644
index 0000000000..12f053a4fe
--- /dev/null
+++ b/third_party/rust/ouroboros/src/lib.rs
@@ -0,0 +1,249 @@
+//! A crate for creating safe self-referencing structs.
+//!
+//! See the documentation of [`ouroboros_examples`](https://docs.rs/ouroboros_examples) for
+//! sample documentation of structs which have had the macro applied to them.
+
+#![allow(clippy::needless_doctest_main)]
+
+/// This macro is used to turn a regular struct into a self-referencing one. An example:
+/// ```rust
+/// use ouroboros::self_referencing;
+///
+/// #[self_referencing]
+/// struct MyStruct {
+/// int_data: Box<i32>,
+/// float_data: Box<f32>,
+/// #[borrows(int_data)]
+/// int_reference: &'this i32,
+/// #[borrows(mut float_data)]
+/// float_reference: &'this mut f32,
+/// }
+///
+/// fn main() {
+/// let mut my_value = MyStructBuilder {
+/// int_data: Box::new(42),
+/// float_data: Box::new(3.14),
+/// int_reference_builder: |int_data: &i32| int_data,
+/// float_reference_builder: |float_data: &mut f32| float_data,
+/// }.build();
+///
+/// // Prints 42
+/// println!("{:?}", my_value.borrow_int_data());
+/// // Prints 3.14
+/// println!("{:?}", my_value.borrow_float_reference());
+/// // Sets the value of float_data to 84.0
+/// my_value.with_mut(|fields| {
+/// **fields.float_reference = (**fields.int_reference as f32) * 2.0;
+/// });
+///
+/// // We can hold on to this reference...
+/// let int_ref = *my_value.borrow_int_reference();
+/// println!("{:?}", *int_ref);
+/// // As long as the struct is still alive.
+/// drop(my_value);
+/// // This will cause an error!
+/// // println!("{:?}", *int_ref);
+/// }
+/// ```
+/// To explain the features and limitations of this crate, some definitions are necessary:
+/// # Definitions
+/// - **immutably borrowed field**: a field which is immutably borrowed by at least one other field.
+/// - **mutably borrowed field**: a field which is mutably borrowed by exactly one other field.
+/// - **self-referencing field**: a field which borrows at least one other field.
+/// - **head field**: a field which does not borrow any other fields, I.E. not self-referencing.
+/// This does not include fields with empty borrows annotations (`#[borrows()]`.)
+/// - **tail field**: a field which is not borrowed by any other fields.
+///
+/// # Usage
+/// To make a self-referencing struct, you must write a struct definition and place
+/// `#[self_referencing]` on top. For every field that borrows other fields, you must place
+/// `#[borrows()]` on top and place inside the parenthesis a list of fields that it borrows. Mut can
+/// be prefixed to indicate that a mutable borrow is required. For example,
+/// `#[borrows(a, b, mut c)]` indicates that the first two fields need to be borrowed immutably and
+/// the third needs to be borrowed mutably. You can also use `#[borrows()]` without any arguments to
+/// indicate a field that will eventually borrow from the struct, but does not borrow anything when
+/// first created. For example, you could use this on a field like `error: Option<&'this str>`.
+///
+/// # You must comply with these limitations
+/// - Fields must be declared before the first time they are borrowed.
+/// - Normal borrowing rules apply, E.G. a field cannot be borrowed mutably twice.
+/// - Fields that are borrowed must be of a data type that implement
+/// [`StableDeref`](https://docs.rs/stable_deref_trait/1.2.0/stable_deref_trait/trait.StableDeref.html).
+/// Normally this just means `Box<T>`.
+/// - Fields that use the `'this` lifetime must have a corresponding `#[borrows()]` annotation.
+/// The error for this needs some work, currently you will get an error saying that `'this` is
+/// undefined at the location it was illegally used in.
+///
+/// Violating them will result in an error message directly pointing out the violated rule.
+///
+/// # Flexibility of this crate
+/// The example above uses plain references as the self-referencing part of the struct, but you can
+/// use anything that is dependent on lifetimes of objects inside the struct. For example, you could
+/// do something like this:
+/// ```rust
+/// use ouroboros::self_referencing;
+///
+/// pub struct ComplexData<'a, 'b> {
+/// aref: &'a i32,
+/// bref: &'b mut i32,
+/// number: i32,
+/// }
+///
+/// impl<'a, 'b> ComplexData<'a, 'b> {
+/// fn new(aref: &'a i32, bref: &'b mut i32, number: i32) -> Self {
+/// Self { aref, bref, number }
+/// }
+///
+/// /// Copies the value aref points to into what bref points to.
+/// fn transfer(&mut self) {
+/// *self.bref = *self.aref;
+/// }
+///
+/// /// Prints the value bref points to.
+/// fn print_bref(&self) {
+/// println!("{}", *self.bref);
+/// }
+/// }
+///
+/// fn main() {
+/// #[self_referencing]
+/// struct DataStorage {
+/// immutable: Box<i32>,
+/// mutable: Box<i32>,
+/// #[borrows(immutable, mut mutable)]
+/// complex_data: ComplexData<'this, 'this>,
+/// }
+///
+/// let mut data_storage = DataStorageBuilder {
+/// immutable: Box::new(10),
+/// mutable: Box::new(20),
+/// complex_data_builder: |i: &i32, m: &mut i32| ComplexData::new(i, m, 12345),
+/// }.build();
+/// data_storage.with_complex_data_mut(|data| {
+/// // Copies the value in immutable into mutable.
+/// data.transfer();
+/// // Prints 10
+/// data.print_bref();
+/// });
+/// }
+/// ```
+/// # Using `chain_hack`
+/// Unfortunately, as of September 2020, Rust has a
+/// [known limitation in its type checker](https://users.rust-lang.org/t/why-does-this-not-compile-box-t-target-t/49027/7?u=aaaaa)
+/// which prevents chained references from working (I.E. structs where field C references field B
+/// which references field A.) To counteract this problem, you can use
+/// `#[self_referencing(chain_hack)]` to allow creating these kinds of structs at the cost of
+/// additional restrictions and possible loss of clarity in some error messages. The main limitation
+/// is that all fields that are borrowed must be of type `Box<T>`. A nice error message will be
+/// generated if you use a different type. There should be no other limitations, but some
+/// configurations may produce strange compiler errors. If you find such a configuration, please
+/// open an issue on the [Github repository](https://github.com/joshua-maros/ouroboros/issues).
+/// You can view a documented example of a struct which uses `chain_hack` [here](https://docs.rs/ouroboros_examples/latest/ouroboros_examples/struct.ChainHack.html).
+///
+/// # What does the macro generate?
+/// The `#[self_referencing]` struct will replace your definition with an unsafe self-referencing
+/// struct with a safe public interface. Many functions will be generated depending on your original
+/// struct definition. Documentation is generated for all items, so building documentation for
+/// your project allows accessing detailed information about available functions. Using
+/// `#[self_referencing(no_doc)]` will hide the generated items from documentation if it is becoming
+/// too cluttered.
+///
+/// ### A quick note on visibility
+/// The visibility of generated items is dependent on one of two things. If the
+/// generated item is related to a specific field of the struct, it uses the visibility of the
+/// original field. (The actual field in the struct will be made private since accessing it could cause
+/// undefined behavior.) If the generated item is not related to any particular field, it will by
+/// default only be visible to the module the struct is declared in. (This includes things like
+/// `new()` and `with()`.) You can use `#[self_referencing(pub_extras)]` to make these items have the
+/// same visibility as the struct itself.
+///
+/// # List of generated items
+/// ### `MyStruct::new(fields...) -> MyStruct`
+/// A basic constructor. It accepts values for each field in the order you declared them in. For
+/// **head fields**, you only need to pass in what value it should have and it will be moved in
+/// to the output. For **self-referencing fields**, you must provide a function or closure which creates
+/// the value based on the values it borrows. A field using the earlier example of
+/// `#[borrow(a, b, mut c)]` would require a function typed as
+/// `FnOnce(a: &_, b: &_, c: &mut _) -> _`. Fields which have an empty borrows annotation
+/// (`#[borrows()]`) should have their value directly passed in. A field using the earlier example
+/// of `Option<&'this str>` would require an input of `None`. Do not pass a function. Do not collect
+/// 200 dollars.
+/// ### `MyStructBuilder`
+/// This is the preferred way to create a new instance of your struct. It is similar to using the
+/// `MyStruct { a, b, c, d }` syntax instead of `MyStruct::new(a, b, c, d)`. It contains one field
+/// for every argument in the actual constructor. **Head fields** have the same name that you
+/// originally defined them with. **self-referencing fields** are suffixed with `_builder` since you need
+/// to provide a function instead of a value. Fields with an empty borrows annotation are not
+/// initialized using builders. Calling `.build()` on an instance of `MyStructBuilder`
+/// will convert it to an instance of `MyStruct`.
+/// ### `MyStruct::try_new<E>(fields...) -> Result<MyStruct, E>`
+/// Similar to the regular `new()` function, except the functions wich create values for all
+/// **self-referencing fields** can return `Result<>`s. If any of those are `Err`s, that error will be
+/// returned instead of an instance of `MyStruct`. The preferred way to use this function is through
+/// `MyStructTryBuilder` and its `try_build()` function.
+/// ### `MyStruct::try_new_or_recover<E>(fields...) -> Result<MyStruct, (E, Heads)>`
+/// Similar to the `try_new()` function, except that all the **head fields** are returned along side
+/// the original error in case of an error. The preferred way to use this function is through
+/// `MyStructTryBuilder` and its `try_build_or_recover()` function.
+/// ### `MyStruct::with_FIELD<R>(&self, user: FnOnce(field: &FieldType) -> R) -> R`
+/// This function is generated for every **tail and immutably-borrowed field** in your struct. It
+/// allows safely accessing
+/// a reference to that value. The function generates the reference and passes it to `user`. You
+/// can do anything you want with the reference, it is constructed to not outlive the struct.
+/// ### `MyStruct::borrow_FIELD(&self) -> &FieldType`
+/// This function is generated for every **tail and immutably-borrowed field** in your struct. It
+/// is equivalent to calling `my_struct.with_FIELD(|field| field)`. Note that certain types of
+/// fields would cause this function to generate a compiler error, so it is ommitted. Generally, if
+/// your field uses `'this` as a lifetime parameter, the corresponding `borrow_FIELD` function will
+/// not be generated. There is no `borrow_FIELD_mut`, unfortunately, as Rust's
+/// borrow checker is currently not capable of ensuring that such a method would be used safely.
+/// ### `MyStruct::with_FIELD_mut<R>(&mut self, user: FnOnce(field: &mut FieldType) -> R) -> R`
+/// This function is generated for every **tail field** in your struct. It is the mutable version
+/// of `with_FIELD`.
+/// ### `MyStruct::with<R>(&self, user: FnOnce(fields: AllFields) -> R) -> R`
+/// Allows borrowing all **tail and immutably-borrowed fields** at once. Functions similarly to
+/// `with_FIELD`.
+/// ### `MyStruct::with_mut<R>(&self, user: FnOnce(fields: AllFields) -> R) -> R`
+/// Allows mutably borrowing all **tail fields** and immutably borrowing all **immutably-borrowed**
+/// fields at once. Functions similarly to `with_FIELD_mut`, except that you can borrow multiple
+/// fields as mutable at the same time and also have immutable access to any remaining fields.
+/// ### `MyStruct::into_heads(self) -> Heads`
+/// Drops all self-referencing fields and returns a struct containing all **head fields**.
+pub use ouroboros_macro::self_referencing;
+
+#[doc(hidden)]
+pub mod macro_help {
+ use stable_deref_trait::StableDeref;
+ use std::ops::DerefMut;
+
+ /// Converts a reference to an object implementing Deref to a static reference to the data it
+ /// Derefs to. This is obviously unsafe because the compiler can no longer guarantee that the
+ /// data outlives the reference. This function is templated to only work for containers that
+ /// implement StableDeref, E.G. Box and Rc. The intent is that the data that is being pointed
+ /// to will never move as long as the container itself is not dropped. It is up to the consumer
+ /// to get rid of the reference before the container is dropped. The + 'static ensures that
+ /// whatever we are referring to will remain valid indefinitely, that there are no limitations
+ /// on how long the pointer itself can live.
+ ///
+ /// # Safety
+ ///
+ /// The caller must ensure that the returned reference is not used after the originally passed
+ /// reference would become invalid.
+ pub unsafe fn stable_deref_and_strip_lifetime<'a, T: StableDeref + 'static>(
+ data: &'a T,
+ ) -> &'static T::Target {
+ &*((&**data) as *const _)
+ }
+
+ /// Like stable_deref_and_strip_lifetime, but for mutable references.
+ ///
+ /// # Safety
+ ///
+ /// The caller must ensure that the returned reference is not used after the originally passed
+ /// reference would become invalid.
+ pub unsafe fn stable_deref_and_strip_lifetime_mut<'a, T: StableDeref + DerefMut + 'static>(
+ data: &'a mut T,
+ ) -> &'static mut T::Target {
+ &mut *((&mut **data) as *mut _)
+ }
+}