Skip to content

Commit f6faef4

Browse files
committed
Auto merge of #114795 - RalfJung:cell-swap, r=dtolnay
make Cell::swap panic if the Cells partially overlap The following function ought to be sound: ```rust fn as_cell_of_array<T, const N: usize>(c: &[Cell<T>; N]) -> &Cell<[T; N]> { unsafe { transmute(c) } } ``` However, due to `Cell::swap`, it currently is not -- safe code can [cause a use-after-free](https://play.rust-lang.org/?version=nightly&mode=debug&edition=2018&gist=c9415799722d985ff7d2c2c997b724ca). This PR fixes that. Fixes #80778
2 parents cedbe5c + e7a1e42 commit f6faef4

File tree

2 files changed

+16
-3
lines changed

2 files changed

+16
-3
lines changed

library/core/src/cell.rs

+15-2
Original file line numberDiff line numberDiff line change
@@ -237,6 +237,7 @@
237237

238238
use crate::cmp::Ordering;
239239
use crate::fmt::{self, Debug, Display};
240+
use crate::intrinsics::is_nonoverlapping;
240241
use crate::marker::{PhantomData, Unsize};
241242
use crate::mem;
242243
use crate::ops::{CoerceUnsized, Deref, DerefMut, DispatchFromDyn};
@@ -415,6 +416,12 @@ impl<T> Cell<T> {
415416
/// Swaps the values of two `Cell`s.
416417
/// Difference with `std::mem::swap` is that this function doesn't require `&mut` reference.
417418
///
419+
/// # Panics
420+
///
421+
/// This function will panic if `self` and `other` are different `Cell`s that partially overlap.
422+
/// (Using just standard library methods, it is impossible to create such partially overlapping `Cell`s.
423+
/// However, unsafe code is allowed to e.g. create two `&Cell<[i32; 2]>` that partially overlap.)
424+
///
418425
/// # Examples
419426
///
420427
/// ```
@@ -430,14 +437,20 @@ impl<T> Cell<T> {
430437
#[stable(feature = "move_cell", since = "1.17.0")]
431438
pub fn swap(&self, other: &Self) {
432439
if ptr::eq(self, other) {
440+
// Swapping wouldn't change anything.
433441
return;
434442
}
443+
if !is_nonoverlapping(self, other, 1) {
444+
// See <https://github.com/rust-lang/rust/issues/80778> for why we need to stop here.
445+
panic!("`Cell::swap` on overlapping non-identical `Cell`s");
446+
}
435447
// SAFETY: This can be risky if called from separate threads, but `Cell`
436448
// is `!Sync` so this won't happen. This also won't invalidate any
437449
// pointers since `Cell` makes sure nothing else will be pointing into
438-
// either of these `Cell`s.
450+
// either of these `Cell`s. We also excluded shenanigans like partially overlapping `Cell`s,
451+
// so `swap` will just properly copy two full values of type `T` back and forth.
439452
unsafe {
440-
ptr::swap(self.value.get(), other.value.get());
453+
mem::swap(&mut *self.value.get(), &mut *other.value.get());
441454
}
442455
}
443456

library/core/src/intrinsics.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2567,7 +2567,7 @@ pub(crate) fn is_nonoverlapping<T>(src: *const T, dst: *const T, count: usize) -
25672567
let size = mem::size_of::<T>()
25682568
.checked_mul(count)
25692569
.expect("is_nonoverlapping: `size_of::<T>() * count` overflows a usize");
2570-
let diff = if src_usize > dst_usize { src_usize - dst_usize } else { dst_usize - src_usize };
2570+
let diff = src_usize.abs_diff(dst_usize);
25712571
// If the absolute distance between the ptrs is at least as big as the size of the buffer,
25722572
// they do not overlap.
25732573
diff >= size

0 commit comments

Comments
 (0)