Skip to content

Commit 41d5c9c

Browse files
committed
Auto merge of rust-lang#126556 - saethlin:layout-precondition, r=<try>
Add a precondition check for Layout::from_size_align_unchecked Ran into this while looking into rust-lang/miri#3679. This is of course not the cause of the ICE, but the reproducer doesn't encounter a precondition check and it ought to.
2 parents cb8a7ea + 44b1911 commit 41d5c9c

File tree

3 files changed

+27
-5
lines changed

3 files changed

+27
-5
lines changed

library/alloc/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,7 @@
138138
#![feature(maybe_uninit_uninit_array_transpose)]
139139
#![feature(panic_internals)]
140140
#![feature(pattern)]
141+
#![feature(ptr_alignment_type)]
141142
#![feature(ptr_internals)]
142143
#![feature(ptr_metadata)]
143144
#![feature(ptr_sub_ptr)]

library/alloc/src/raw_vec.rs

+16-5
Original file line numberDiff line numberDiff line change
@@ -297,7 +297,20 @@ impl<T, A: Allocator> RawVec<T, A> {
297297
}
298298

299299
fn current_memory(&self) -> Option<(NonNull<u8>, Layout)> {
300-
if T::IS_ZST || self.cap.0 == 0 {
300+
// Reduce the amount of code we need to monomorphize per `T`.
301+
#[inline]
302+
#[rustc_no_mir_inline]
303+
unsafe fn inner(size: usize, align: usize, cap: usize) -> Layout {
304+
// SAFETY: Precondition guaranteed by the caller
305+
unsafe {
306+
let size = size.unchecked_mul(cap);
307+
Layout::from_size_align_unchecked(size, align)
308+
}
309+
}
310+
311+
let cap = self.cap.0;
312+
313+
if T::IS_ZST || cap == 0 {
301314
None
302315
} else {
303316
// We could use Layout::array here which ensures the absence of isize and usize overflows
@@ -306,10 +319,8 @@ impl<T, A: Allocator> RawVec<T, A> {
306319
// support such types. So we can do better by skipping some checks and avoid an unwrap.
307320
const { assert!(mem::size_of::<T>() % mem::align_of::<T>() == 0) };
308321
unsafe {
309-
let align = mem::align_of::<T>();
310-
let size = mem::size_of::<T>().unchecked_mul(self.cap.0);
311-
let layout = Layout::from_size_align_unchecked(size, align);
312-
Some((self.ptr.cast().into(), layout))
322+
let layout = inner(mem::size_of::<T>(), mem::align_of::<T>(), cap);
323+
Some((self.non_null().cast(), layout))
313324
}
314325
}
315326
}

library/core/src/alloc/layout.rs

+10
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
// collections, resulting in having to optimize down excess IR multiple times.
55
// Your performance intuition is useless. Run perf.
66

7+
use crate::assert_unsafe_precondition;
78
use crate::cmp;
89
use crate::error::Error;
910
use crate::fmt;
@@ -118,6 +119,15 @@ impl Layout {
118119
#[inline]
119120
#[rustc_allow_const_fn_unstable(ptr_alignment_type)]
120121
pub const unsafe fn from_size_align_unchecked(size: usize, align: usize) -> Self {
122+
assert_unsafe_precondition!(
123+
check_library_ub,
124+
"Layout::from_size_align_unchecked requires that align is a power of 2 \
125+
and the rounded-up allocation size does not exceed isize::MAX",
126+
(
127+
size: usize = size,
128+
align: usize = align,
129+
) => Layout::from_size_align(size, align).is_ok()
130+
);
121131
// SAFETY: the caller is required to uphold the preconditions.
122132
unsafe { Layout { size, align: Alignment::new_unchecked(align) } }
123133
}

0 commit comments

Comments
 (0)