Skip to content

Commit 7820b62

Browse files
committed
Auto merge of rust-lang#105117 - pitaj:debug_asserts, r=the8472
Add more debug assertions to unsafe functions related to rust-lang#51713
2 parents 73c8d2d + cd35794 commit 7820b62

File tree

3 files changed

+50
-35
lines changed

3 files changed

+50
-35
lines changed

library/core/src/slice/index.rs

+2-3
Original file line numberDiff line numberDiff line change
@@ -371,12 +371,11 @@ unsafe impl<T> const SliceIndex<[T]> for ops::Range<usize> {
371371

372372
#[inline]
373373
unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] {
374-
let this = ops::Range { start: self.start, end: self.end };
374+
let this = ops::Range { ..self };
375375
// SAFETY: the caller guarantees that `slice` is not dangling, so it
376376
// cannot be longer than `isize::MAX`. They also guarantee that
377377
// `self` is in bounds of `slice` so `self` cannot overflow an `isize`,
378378
// so the call to `add` is safe.
379-
380379
unsafe {
381380
assert_unsafe_precondition!(
382381
"slice::get_unchecked requires that the range is within the slice",
@@ -389,7 +388,7 @@ unsafe impl<T> const SliceIndex<[T]> for ops::Range<usize> {
389388

390389
#[inline]
391390
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] {
392-
let this = ops::Range { start: self.start, end: self.end };
391+
let this = ops::Range { ..self };
393392
// SAFETY: see comments for `get_unchecked` above.
394393
unsafe {
395394
assert_unsafe_precondition!(

library/core/src/slice/mod.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -1695,7 +1695,13 @@ impl<T> [T] {
16951695
let ptr = self.as_ptr();
16961696

16971697
// SAFETY: Caller has to check that `0 <= mid <= self.len()`
1698-
unsafe { (from_raw_parts(ptr, mid), from_raw_parts(ptr.add(mid), len - mid)) }
1698+
unsafe {
1699+
assert_unsafe_precondition!(
1700+
"slice::split_at_unchecked requires the index to be within the slice",
1701+
(mid: usize, len: usize) => mid <= len
1702+
);
1703+
(from_raw_parts(ptr, mid), from_raw_parts(ptr.add(mid), len - mid))
1704+
}
16991705
}
17001706

17011707
/// Divides one mutable slice into two at an index, without doing bounds checking.

library/core/src/str/traits.rs

+41-31
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
//! Trait implementations for `str`.
22
33
use crate::cmp::Ordering;
4+
use crate::intrinsics::assert_unsafe_precondition;
45
use crate::ops;
56
use crate::ptr;
67
use crate::slice::SliceIndex;
@@ -194,15 +195,37 @@ unsafe impl const SliceIndex<str> for ops::Range<usize> {
194195
let slice = slice as *const [u8];
195196
// SAFETY: the caller guarantees that `self` is in bounds of `slice`
196197
// which satisfies all the conditions for `add`.
197-
let ptr = unsafe { slice.as_ptr().add(self.start) };
198+
let ptr = unsafe {
199+
let this = ops::Range { ..self };
200+
assert_unsafe_precondition!(
201+
"str::get_unchecked requires that the range is within the string slice",
202+
(this: ops::Range<usize>, slice: *const [u8]) =>
203+
// We'd like to check that the bounds are on char boundaries,
204+
// but there's not really a way to do so without reading
205+
// behind the pointer, which has aliasing implications.
206+
// It's also not possible to move this check up to
207+
// `str::get_unchecked` without adding a special function
208+
// to `SliceIndex` just for this.
209+
this.end >= this.start && this.end <= slice.len()
210+
);
211+
slice.as_ptr().add(self.start)
212+
};
198213
let len = self.end - self.start;
199214
ptr::slice_from_raw_parts(ptr, len) as *const str
200215
}
201216
#[inline]
202217
unsafe fn get_unchecked_mut(self, slice: *mut str) -> *mut Self::Output {
203218
let slice = slice as *mut [u8];
204219
// SAFETY: see comments for `get_unchecked`.
205-
let ptr = unsafe { slice.as_mut_ptr().add(self.start) };
220+
let ptr = unsafe {
221+
let this = ops::Range { ..self };
222+
assert_unsafe_precondition!(
223+
"str::get_unchecked_mut requires that the range is within the string slice",
224+
(this: ops::Range<usize>, slice: *mut [u8]) =>
225+
this.end >= this.start && this.end <= slice.len()
226+
);
227+
slice.as_mut_ptr().add(self.start)
228+
};
206229
let len = self.end - self.start;
207230
ptr::slice_from_raw_parts_mut(ptr, len) as *mut str
208231
}
@@ -272,15 +295,13 @@ unsafe impl const SliceIndex<str> for ops::RangeTo<usize> {
272295
}
273296
#[inline]
274297
unsafe fn get_unchecked(self, slice: *const str) -> *const Self::Output {
275-
let slice = slice as *const [u8];
276-
let ptr = slice.as_ptr();
277-
ptr::slice_from_raw_parts(ptr, self.end) as *const str
298+
// SAFETY: the caller has to uphold the safety contract for `get_unchecked`.
299+
unsafe { (0..self.end).get_unchecked(slice) }
278300
}
279301
#[inline]
280302
unsafe fn get_unchecked_mut(self, slice: *mut str) -> *mut Self::Output {
281-
let slice = slice as *mut [u8];
282-
let ptr = slice.as_mut_ptr();
283-
ptr::slice_from_raw_parts_mut(ptr, self.end) as *mut str
303+
// SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`.
304+
unsafe { (0..self.end).get_unchecked_mut(slice) }
284305
}
285306
#[inline]
286307
fn index(self, slice: &str) -> &Self::Output {
@@ -343,20 +364,15 @@ unsafe impl const SliceIndex<str> for ops::RangeFrom<usize> {
343364
}
344365
#[inline]
345366
unsafe fn get_unchecked(self, slice: *const str) -> *const Self::Output {
346-
let slice = slice as *const [u8];
347-
// SAFETY: the caller guarantees that `self` is in bounds of `slice`
348-
// which satisfies all the conditions for `add`.
349-
let ptr = unsafe { slice.as_ptr().add(self.start) };
350-
let len = slice.len() - self.start;
351-
ptr::slice_from_raw_parts(ptr, len) as *const str
367+
let len = (slice as *const [u8]).len();
368+
// SAFETY: the caller has to uphold the safety contract for `get_unchecked`.
369+
unsafe { (self.start..len).get_unchecked(slice) }
352370
}
353371
#[inline]
354372
unsafe fn get_unchecked_mut(self, slice: *mut str) -> *mut Self::Output {
355-
let slice = slice as *mut [u8];
356-
// SAFETY: identical to `get_unchecked`.
357-
let ptr = unsafe { slice.as_mut_ptr().add(self.start) };
358-
let len = slice.len() - self.start;
359-
ptr::slice_from_raw_parts_mut(ptr, len) as *mut str
373+
let len = (slice as *mut [u8]).len();
374+
// SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`.
375+
unsafe { (self.start..len).get_unchecked_mut(slice) }
360376
}
361377
#[inline]
362378
fn index(self, slice: &str) -> &Self::Output {
@@ -452,35 +468,29 @@ unsafe impl const SliceIndex<str> for ops::RangeToInclusive<usize> {
452468
type Output = str;
453469
#[inline]
454470
fn get(self, slice: &str) -> Option<&Self::Output> {
455-
if self.end == usize::MAX { None } else { (..self.end + 1).get(slice) }
471+
(0..=self.end).get(slice)
456472
}
457473
#[inline]
458474
fn get_mut(self, slice: &mut str) -> Option<&mut Self::Output> {
459-
if self.end == usize::MAX { None } else { (..self.end + 1).get_mut(slice) }
475+
(0..=self.end).get_mut(slice)
460476
}
461477
#[inline]
462478
unsafe fn get_unchecked(self, slice: *const str) -> *const Self::Output {
463479
// SAFETY: the caller must uphold the safety contract for `get_unchecked`.
464-
unsafe { (..self.end + 1).get_unchecked(slice) }
480+
unsafe { (0..=self.end).get_unchecked(slice) }
465481
}
466482
#[inline]
467483
unsafe fn get_unchecked_mut(self, slice: *mut str) -> *mut Self::Output {
468484
// SAFETY: the caller must uphold the safety contract for `get_unchecked_mut`.
469-
unsafe { (..self.end + 1).get_unchecked_mut(slice) }
485+
unsafe { (0..=self.end).get_unchecked_mut(slice) }
470486
}
471487
#[inline]
472488
fn index(self, slice: &str) -> &Self::Output {
473-
if self.end == usize::MAX {
474-
str_index_overflow_fail();
475-
}
476-
(..self.end + 1).index(slice)
489+
(0..=self.end).index(slice)
477490
}
478491
#[inline]
479492
fn index_mut(self, slice: &mut str) -> &mut Self::Output {
480-
if self.end == usize::MAX {
481-
str_index_overflow_fail();
482-
}
483-
(..self.end + 1).index_mut(slice)
493+
(0..=self.end).index_mut(slice)
484494
}
485495
}
486496

0 commit comments

Comments
 (0)