-
Notifications
You must be signed in to change notification settings - Fork 14k
Description
/*
[dependencies]
bumpalo = { version = "3", features = ["allocator_api"] }
*/
#![feature(allocator_api)]
use bumpalo::Bump;
use std::rc::Rc;
fn main() {
let bump = Bump::new();
let boxx = Box::new_in(42, &bump);
let rc = Rc::<i32, _>::from(boxx);
}Output:
free(): invalid pointer
Miri:
error: Undefined Behavior: deallocating 0x258bc[alloc1011]<2851> which does not point to the beginning of an object
--> /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/alloc.rs:117:14
|
117 | unsafe { __rust_dealloc(ptr, layout.size(), layout.align()) }
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ deallocating 0x258bc[alloc1011]<2851> which does not point to the beginning of an object
|
= help: this indicates a bug in the program: it performed an invalid operation, and caused Undefined Behavior
= help: see https://doc.rust-lang.org/nightly/reference/behavior-considered-undefined.html for further information
= note: BACKTRACE:
= note: inside `std::alloc::dealloc` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/alloc.rs:117:14: 117:64
= note: inside `<std::alloc::Global as std::alloc::Allocator>::deallocate` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/alloc.rs:254:22: 254:51
= note: inside `<std::boxed::Box<std::mem::ManuallyDrop<i32>> as std::ops::Drop>::drop` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/boxed.rs:1244:17: 1244:66
= note: inside `std::ptr::drop_in_place::<std::boxed::Box<std::mem::ManuallyDrop<i32>>> - shim(Some(std::boxed::Box<std::mem::ManuallyDrop<i32>>))` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/core/src/ptr/mod.rs:507:1: 507:56
= note: inside `std::mem::drop::<std::boxed::Box<std::mem::ManuallyDrop<i32>>>` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/core/src/mem/mod.rs:992:24: 992:25
= note: inside `std::rc::Rc::<i32, &bumpalo::Bump>::from_box_in` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/rc.rs:1928:13: 1928:22
= note: inside `<std::rc::Rc<i32, &bumpalo::Bump> as std::convert::From<std::boxed::Box<i32, &bumpalo::Bump>>>::from` at /home/frank/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library/alloc/src/rc.rs:2504:9: 2504:27
note: inside `main`
--> src/main.rs:16:14
|
16 | let rc = Rc::<i32, _>::from(boxx);
| ^^^^^^^^^^^^^^^^^^^^^^^^
note: some details are omitted, run with `MIRIFLAGS=-Zmiri-backtrace=full` for a verbose backtrace
The underlying issue was found when reviewing the implementation of From<Box<T, A>> for Rc<T, A>. The code is a reproduction to trigger the unsoundness. The implementation in question features:
Lines 1912 to 1932 in ca663b0
| #[cfg(not(no_global_oom_handling))] | |
| fn from_box_in(src: Box<T, A>) -> Rc<T, A> { | |
| unsafe { | |
| let value_size = size_of_val(&*src); | |
| let ptr = Self::allocate_for_ptr_in(&*src, Box::allocator(&src)); | |
| // Copy value as bytes | |
| ptr::copy_nonoverlapping( | |
| &*src as *const T as *const u8, | |
| &mut (*ptr).value as *mut _ as *mut u8, | |
| value_size, | |
| ); | |
| // Free the allocation without dropping its contents | |
| let (bptr, alloc) = Box::into_raw_with_allocator(src); | |
| let src = Box::from_raw(bptr as *mut mem::ManuallyDrop<T>); | |
| drop(src); | |
| Self::from_ptr_in(ptr, alloc) | |
| } | |
| } |
Where Box<T, A> is turned into Box<MaybeUninit<T>> and dropped. Yes, that’s Box<MaybeUninit<T>, Global>, not Box<MaybeUninit<T>, A>.
The code for Arc seems to be doing the same thing.
rust/library/alloc/src/sync.rs
Lines 1858 to 1877 in ca663b0
| fn from_box_in(src: Box<T, A>) -> Arc<T, A> { | |
| unsafe { | |
| let value_size = size_of_val(&*src); | |
| let ptr = Self::allocate_for_ptr_in(&*src, Box::allocator(&src)); | |
| // Copy value as bytes | |
| ptr::copy_nonoverlapping( | |
| &*src as *const T as *const u8, | |
| &mut (*ptr).data as *mut _ as *mut u8, | |
| value_size, | |
| ); | |
| // Free the allocation without dropping its contents | |
| let (bptr, alloc) = Box::into_raw_with_allocator(src); | |
| let src = Box::from_raw(bptr as *mut mem::ManuallyDrop<T>); | |
| drop(src); | |
| Self::from_ptr_in(ptr, alloc) | |
| } | |
| } |
For comparison: The code for From<Vec<T, A>> for Rc<[T]> (and the same thing for Arc) seems to properly use a Vec<MaybeUninit<T>, &A>… no actually, that’s a Vec<T, &A> with zero lengths… notably the allocator is correct though.
Lines 2521 to 2535 in ca663b0
| #[inline] | |
| fn from(v: Vec<T, A>) -> Rc<[T], A> { | |
| unsafe { | |
| let (vec_ptr, len, cap, alloc) = v.into_raw_parts_with_alloc(); | |
| let rc_ptr = Self::allocate_for_slice_in(len, &alloc); | |
| ptr::copy_nonoverlapping(vec_ptr, &mut (*rc_ptr).value as *mut [T] as *mut T, len); | |
| // Create a `Vec<T, &A>` with length 0, to deallocate the buffer | |
| // without dropping its contents or the allocator | |
| let _ = Vec::from_raw_parts_in(vec_ptr, 0, cap, &alloc); | |
| Self::from_ptr_in(rc_ptr, alloc) | |
| } | |
| } |
So the buggy code for Box should just switch to create a Box<MaybeUninit<T>, &A> presumably.
@rustbot label I-unsound requires-nightly T-libs