Skip to content

Commit

Permalink
Deprecate offset_to; switch core&alloc to using offset_from instead
Browse files Browse the repository at this point in the history
Bonus: might make code than uses `.len()` on slice iterators faster
  • Loading branch information
scottmcm committed Apr 1, 2018
1 parent 80785a5 commit b394165
Show file tree
Hide file tree
Showing 4 changed files with 19 additions and 13 deletions.
2 changes: 1 addition & 1 deletion src/liballoc/lib.rs
Expand Up @@ -103,13 +103,13 @@
#![feature(lang_items)]
#![feature(needs_allocator)]
#![feature(nonzero)]
#![feature(offset_to)]
#![feature(optin_builtin_traits)]
#![feature(pattern)]
#![feature(pin)]
#![feature(placement_in_syntax)]
#![feature(placement_new_protocol)]
#![feature(ptr_internals)]
#![feature(ptr_offset_from)]
#![feature(rustc_attrs)]
#![feature(slice_get_slice)]
#![feature(slice_rsplit)]
Expand Down
7 changes: 4 additions & 3 deletions src/liballoc/vec.rs
Expand Up @@ -2338,9 +2338,10 @@ impl<T> Iterator for IntoIter<T> {

#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
let exact = match self.ptr.offset_to(self.end) {
Some(x) => x as usize,
None => (self.end as usize).wrapping_sub(self.ptr as usize),
let exact = if mem::size_of::<T>() == 0 {
(self.end as usize).wrapping_sub(self.ptr as usize)
} else {
unsafe { self.end.offset_from(self.ptr) as usize }
};
(exact, Some(exact))
}
Expand Down
12 changes: 8 additions & 4 deletions src/libcore/ptr.rs
Expand Up @@ -677,6 +677,7 @@ impl<T: ?Sized> *const T {
///
/// ```
/// #![feature(offset_to)]
/// #![allow(deprecated)]
///
/// fn main() {
/// let a = [0; 5];
Expand All @@ -689,14 +690,15 @@ impl<T: ?Sized> *const T {
/// }
/// ```
#[unstable(feature = "offset_to", issue = "41079")]
#[rustc_deprecated(since = "1.27.0", reason = "Replaced by `wrapping_offset_from`, with the \
opposite argument order. If you're writing unsafe code, consider `offset_from`.")]
#[inline]
pub fn offset_to(self, other: *const T) -> Option<isize> where T: Sized {
let size = mem::size_of::<T>();
if size == 0 {
None
} else {
let diff = (other as isize).wrapping_sub(self as isize);
Some(diff / size as isize)
Some(other.wrapping_offset_from(self))
}
}

Expand Down Expand Up @@ -1442,6 +1444,7 @@ impl<T: ?Sized> *mut T {
///
/// ```
/// #![feature(offset_to)]
/// #![allow(deprecated)]
///
/// fn main() {
/// let mut a = [0; 5];
Expand All @@ -1454,14 +1457,15 @@ impl<T: ?Sized> *mut T {
/// }
/// ```
#[unstable(feature = "offset_to", issue = "41079")]
#[rustc_deprecated(since = "1.27.0", reason = "Replaced by `wrapping_offset_from`, with the \
opposite argument order. If you're writing unsafe code, consider `offset_from`.")]
#[inline]
pub fn offset_to(self, other: *const T) -> Option<isize> where T: Sized {
let size = mem::size_of::<T>();
if size == 0 {
None
} else {
let diff = (other as isize).wrapping_sub(self as isize);
Some(diff / size as isize)
Some(other.wrapping_offset_from(self))
}
}

Expand Down
11 changes: 6 additions & 5 deletions src/libcore/slice/mod.rs
Expand Up @@ -1185,7 +1185,7 @@ macro_rules! iterator {

#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
let exact = ptrdistance(self.ptr, self.end);
let exact = unsafe { ptrdistance(self.ptr, self.end) };
(exact, Some(exact))
}

Expand Down Expand Up @@ -1593,10 +1593,11 @@ unsafe impl<'a, T> TrustedLen for IterMut<'a, T> {}
// Return the number of elements of `T` from `start` to `end`.
// Return the arithmetic difference if `T` is zero size.
#[inline(always)]
fn ptrdistance<T>(start: *const T, end: *const T) -> usize {
match start.offset_to(end) {
Some(x) => x as usize,
None => (end as usize).wrapping_sub(start as usize),
unsafe fn ptrdistance<T>(start: *const T, end: *const T) -> usize {
if mem::size_of::<T>() == 0 {
(end as usize).wrapping_sub(start as usize)
} else {
end.offset_from(start) as usize
}
}

Expand Down

0 comments on commit b394165

Please sign in to comment.