Rollup merge of #60963 - blkerby:boxed_docs, r=alexcrichton
Update boxed::Box docs on memory layout The existing docs for the `Box` type state that "the way `Box` allocates and releases memory is unspecified", and that therefore the only valid pointer to pass to `Box::from_raw` is one obtained from `Box::into_raw`. This is inconsistent with the module-level docs which specify, > It is valid to convert both ways between a Box and a raw pointer allocated with the Global allocator, given that the Layout used with the allocator is correct for the type. More precisely, a value: *mut T that has been allocated with the Global allocator with Layout::for_value(&*value) may be converted into a box using Box::<T>::from_raw(value). Conversely, the memory backing a value: *mut T obtained from Box::<T>::into_raw may be deallocated using the Global allocator with Layout::for_value(&*value). This pull request updates the docs for `Box` to make them consistent with the module-level docs and adds some examples of how to use the global allocator in conjunction with `Box::from_raw` and `Box::into_raw`.
This commit is contained in:
commit
40a18a1df5
@ -4,16 +4,6 @@
|
|||||||
//! heap allocation in Rust. Boxes provide ownership for this allocation, and
|
//! heap allocation in Rust. Boxes provide ownership for this allocation, and
|
||||||
//! drop their contents when they go out of scope.
|
//! drop their contents when they go out of scope.
|
||||||
//!
|
//!
|
||||||
//! For non-zero-sized values, a [`Box`] will use the [`Global`] allocator for
|
|
||||||
//! its allocation. It is valid to convert both ways between a [`Box`] and a
|
|
||||||
//! raw pointer allocated with the [`Global`] allocator, given that the
|
|
||||||
//! [`Layout`] used with the allocator is correct for the type. More precisely,
|
|
||||||
//! a `value: *mut T` that has been allocated with the [`Global`] allocator
|
|
||||||
//! with `Layout::for_value(&*value)` may be converted into a box using
|
|
||||||
//! `Box::<T>::from_raw(value)`. Conversely, the memory backing a `value: *mut
|
|
||||||
//! T` obtained from `Box::<T>::into_raw` may be deallocated using the
|
|
||||||
//! [`Global`] allocator with `Layout::for_value(&*value)`.
|
|
||||||
//!
|
|
||||||
//! # Examples
|
//! # Examples
|
||||||
//!
|
//!
|
||||||
//! Move a value from the stack to the heap by creating a [`Box`]:
|
//! Move a value from the stack to the heap by creating a [`Box`]:
|
||||||
@ -61,6 +51,19 @@
|
|||||||
//! for a `Cons`. By introducing a `Box`, which has a defined size, we know how
|
//! for a `Cons`. By introducing a `Box`, which has a defined size, we know how
|
||||||
//! big `Cons` needs to be.
|
//! big `Cons` needs to be.
|
||||||
//!
|
//!
|
||||||
|
//! # Memory layout
|
||||||
|
//!
|
||||||
|
//! For non-zero-sized values, a [`Box`] will use the [`Global`] allocator for
|
||||||
|
//! its allocation. It is valid to convert both ways between a [`Box`] and a
|
||||||
|
//! raw pointer allocated with the [`Global`] allocator, given that the
|
||||||
|
//! [`Layout`] used with the allocator is correct for the type. More precisely,
|
||||||
|
//! a `value: *mut T` that has been allocated with the [`Global`] allocator
|
||||||
|
//! with `Layout::for_value(&*value)` may be converted into a box using
|
||||||
|
//! `Box::<T>::from_raw(value)`. Conversely, the memory backing a `value: *mut
|
||||||
|
//! T` obtained from `Box::<T>::into_raw` may be deallocated using the
|
||||||
|
//! [`Global`] allocator with `Layout::for_value(&*value)`.
|
||||||
|
//!
|
||||||
|
//!
|
||||||
//! [dereferencing]: ../../std/ops/trait.Deref.html
|
//! [dereferencing]: ../../std/ops/trait.Deref.html
|
||||||
//! [`Box`]: struct.Box.html
|
//! [`Box`]: struct.Box.html
|
||||||
//! [`Global`]: ../alloc/struct.Global.html
|
//! [`Global`]: ../alloc/struct.Global.html
|
||||||
@ -127,24 +130,38 @@ impl<T: ?Sized> Box<T> {
|
|||||||
///
|
///
|
||||||
/// After calling this function, the raw pointer is owned by the
|
/// After calling this function, the raw pointer is owned by the
|
||||||
/// resulting `Box`. Specifically, the `Box` destructor will call
|
/// resulting `Box`. Specifically, the `Box` destructor will call
|
||||||
/// the destructor of `T` and free the allocated memory. Since the
|
/// the destructor of `T` and free the allocated memory. For this
|
||||||
/// way `Box` allocates and releases memory is unspecified, the
|
/// to be safe, the memory must have been allocated in accordance
|
||||||
/// only valid pointer to pass to this function is the one taken
|
/// with the [memory layout] used by `Box` .
|
||||||
/// from another `Box` via the [`Box::into_raw`] function.
|
///
|
||||||
|
/// # Safety
|
||||||
///
|
///
|
||||||
/// This function is unsafe because improper use may lead to
|
/// This function is unsafe because improper use may lead to
|
||||||
/// memory problems. For example, a double-free may occur if the
|
/// memory problems. For example, a double-free may occur if the
|
||||||
/// function is called twice on the same raw pointer.
|
/// function is called twice on the same raw pointer.
|
||||||
///
|
///
|
||||||
/// [`Box::into_raw`]: struct.Box.html#method.into_raw
|
|
||||||
///
|
|
||||||
/// # Examples
|
/// # Examples
|
||||||
///
|
/// Recreate a `Box` which was previously converted to a raw pointer
|
||||||
|
/// using [`Box::into_raw`]:
|
||||||
/// ```
|
/// ```
|
||||||
/// let x = Box::new(5);
|
/// let x = Box::new(5);
|
||||||
/// let ptr = Box::into_raw(x);
|
/// let ptr = Box::into_raw(x);
|
||||||
/// let x = unsafe { Box::from_raw(ptr) };
|
/// let x = unsafe { Box::from_raw(ptr) };
|
||||||
/// ```
|
/// ```
|
||||||
|
/// Manually create a `Box` from scratch by using the global allocator:
|
||||||
|
/// ```
|
||||||
|
/// use std::alloc::{alloc, Layout};
|
||||||
|
///
|
||||||
|
/// unsafe {
|
||||||
|
/// let ptr = alloc(Layout::new::<i32>()) as *mut i32;
|
||||||
|
/// *ptr = 5;
|
||||||
|
/// let x = Box::from_raw(ptr);
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// [memory layout]: index.html#memory-layout
|
||||||
|
/// [`Layout`]: ../alloc/struct.Layout.html
|
||||||
|
/// [`Box::into_raw`]: struct.Box.html#method.into_raw
|
||||||
#[stable(feature = "box_raw", since = "1.4.0")]
|
#[stable(feature = "box_raw", since = "1.4.0")]
|
||||||
#[inline]
|
#[inline]
|
||||||
pub unsafe fn from_raw(raw: *mut T) -> Self {
|
pub unsafe fn from_raw(raw: *mut T) -> Self {
|
||||||
@ -157,22 +174,40 @@ impl<T: ?Sized> Box<T> {
|
|||||||
///
|
///
|
||||||
/// After calling this function, the caller is responsible for the
|
/// After calling this function, the caller is responsible for the
|
||||||
/// memory previously managed by the `Box`. In particular, the
|
/// memory previously managed by the `Box`. In particular, the
|
||||||
/// caller should properly destroy `T` and release the memory. The
|
/// caller should properly destroy `T` and release the memory, taking
|
||||||
/// proper way to do so is to convert the raw pointer back into a
|
/// into account the [memory layout] used by `Box`. The easiest way to
|
||||||
/// `Box` with the [`Box::from_raw`] function.
|
/// do this is to convert the raw pointer back into a `Box` with the
|
||||||
|
/// [`Box::from_raw`] function, allowing the `Box` destructor to perform
|
||||||
|
/// the cleanup.
|
||||||
///
|
///
|
||||||
/// Note: this is an associated function, which means that you have
|
/// Note: this is an associated function, which means that you have
|
||||||
/// to call it as `Box::into_raw(b)` instead of `b.into_raw()`. This
|
/// to call it as `Box::into_raw(b)` instead of `b.into_raw()`. This
|
||||||
/// is so that there is no conflict with a method on the inner type.
|
/// is so that there is no conflict with a method on the inner type.
|
||||||
///
|
///
|
||||||
/// [`Box::from_raw`]: struct.Box.html#method.from_raw
|
|
||||||
///
|
|
||||||
/// # Examples
|
/// # Examples
|
||||||
///
|
/// Converting the raw pointer back into a `Box` with [`Box::from_raw`]
|
||||||
|
/// for automatic cleanup:
|
||||||
/// ```
|
/// ```
|
||||||
/// let x = Box::new(5);
|
/// let x = Box::new(String::from("Hello"));
|
||||||
/// let ptr = Box::into_raw(x);
|
/// let ptr = Box::into_raw(x);
|
||||||
|
/// let x = unsafe { Box::from_raw(ptr) };
|
||||||
/// ```
|
/// ```
|
||||||
|
/// Manual cleanup by explicitly running the destructor and deallocating
|
||||||
|
/// the memory:
|
||||||
|
/// ```
|
||||||
|
/// use std::alloc::{dealloc, Layout};
|
||||||
|
/// use std::ptr;
|
||||||
|
///
|
||||||
|
/// let x = Box::new(String::from("Hello"));
|
||||||
|
/// let p = Box::into_raw(x);
|
||||||
|
/// unsafe {
|
||||||
|
/// ptr::drop_in_place(p);
|
||||||
|
/// dealloc(p as *mut u8, Layout::new::<String>());
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// [memory layout]: index.html#memory-layout
|
||||||
|
/// [`Box::from_raw`]: struct.Box.html#method.from_raw
|
||||||
#[stable(feature = "box_raw", since = "1.4.0")]
|
#[stable(feature = "box_raw", since = "1.4.0")]
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn into_raw(b: Box<T>) -> *mut T {
|
pub fn into_raw(b: Box<T>) -> *mut T {
|
||||||
@ -184,7 +219,7 @@ impl<T: ?Sized> Box<T> {
|
|||||||
/// After calling this function, the caller is responsible for the
|
/// After calling this function, the caller is responsible for the
|
||||||
/// memory previously managed by the `Box`. In particular, the
|
/// memory previously managed by the `Box`. In particular, the
|
||||||
/// caller should properly destroy `T` and release the memory. The
|
/// caller should properly destroy `T` and release the memory. The
|
||||||
/// proper way to do so is to convert the `NonNull<T>` pointer
|
/// easiest way to do so is to convert the `NonNull<T>` pointer
|
||||||
/// into a raw pointer and back into a `Box` with the [`Box::from_raw`]
|
/// into a raw pointer and back into a `Box` with the [`Box::from_raw`]
|
||||||
/// function.
|
/// function.
|
||||||
///
|
///
|
||||||
@ -203,6 +238,10 @@ impl<T: ?Sized> Box<T> {
|
|||||||
/// fn main() {
|
/// fn main() {
|
||||||
/// let x = Box::new(5);
|
/// let x = Box::new(5);
|
||||||
/// let ptr = Box::into_raw_non_null(x);
|
/// let ptr = Box::into_raw_non_null(x);
|
||||||
|
///
|
||||||
|
/// // Clean up the memory by converting the NonNull pointer back
|
||||||
|
/// // into a Box and letting the Box be dropped.
|
||||||
|
/// let x = unsafe { Box::from_raw(ptr.as_ptr()) };
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
#[unstable(feature = "box_into_raw_non_null", issue = "47336")]
|
#[unstable(feature = "box_into_raw_non_null", issue = "47336")]
|
||||||
|
Loading…
x
Reference in New Issue
Block a user