Skip to content

Commit ef8804b

Browse files
committed
Add precondition to Layout that the align fit in a u32.
This precondition takes the form of a behavorial change in `Layout::from_size_align` (so it returns `None` if the `align` is too large) and a new requirement for safe usage of `Layout::from_size_align_unchecked`. Fix #30170.
1 parent b2c0707 commit ef8804b

File tree

1 file changed

+15
-7
lines changed

1 file changed

+15
-7
lines changed

src/liballoc/allocator.rs

+15-7
Original file line numberDiff line numberDiff line change
@@ -65,11 +65,13 @@ pub struct Layout {
6565

6666
impl Layout {
6767
/// Constructs a `Layout` from a given `size` and `align`,
68-
/// or returns `None` if either of the following conditions
68+
/// or returns `None` if any of the following conditions
6969
/// are not met:
7070
///
7171
/// * `align` must be a power of two,
7272
///
73+
/// * `align` must not exceed 2^31 (i.e. `1 << 31`),
74+
///
7375
/// * `size`, when rounded up to the nearest multiple of `align`,
7476
/// must not overflow (i.e. the rounded value must be less than
7577
/// `usize::MAX`).
@@ -79,6 +81,10 @@ impl Layout {
7981
return None;
8082
}
8183

84+
if align > (1 << 31) {
85+
return None;
86+
}
87+
8288
// (power-of-two implies align != 0.)
8389

8490
// Rounded up size is:
@@ -106,8 +112,10 @@ impl Layout {
106112
///
107113
/// # Unsafety
108114
///
109-
/// This function is unsafe as it does not verify that `align` is a power of
110-
/// two nor that `size` aligned to `align` fits within the address space.
115+
/// This function is unsafe as it does not verify that `align` is
116+
/// a power-of-two that is also less than or equal to 2^31, nor
117+
/// that `size` aligned to `align` fits within the address space
118+
/// (i.e. the `Layout::from_size_align` preconditions).
111119
#[inline]
112120
pub unsafe fn from_size_align_unchecked(size: usize, align: usize) -> Layout {
113121
Layout { size: size, align: align }
@@ -217,10 +225,10 @@ impl Layout {
217225
Some(alloc_size) => alloc_size,
218226
};
219227

220-
// We can assume that `self.align` is a power-of-two.
221-
// Furthermore, `alloc_size` has alreayd been rounded up
222-
// to a multiple of `self.align`; therefore, the call
223-
// to `Layout::from_size_align` below should never panic.
228+
// We can assume that `self.align` is a power-of-two that does
229+
// not exceed 2^31. Furthermore, `alloc_size` has already been
230+
// rounded up to a multiple of `self.align`; therefore, the
231+
// call to `Layout::from_size_align` below should never panic.
224232
Some((Layout::from_size_align(alloc_size, self.align).unwrap(), padded_size))
225233
}
226234

0 commit comments

Comments
 (0)