Skip to content

Commit 03596a1

Browse files
authored
Unrolled build for rust-lang#132459
Rollup merge of rust-lang#132459 - RalfJung:byte_sub_ptr, r=scottmcm feat(byte_sub_ptr): unstably add ptr::byte_sub_ptr This is an API that naturally should exist as a combination of byte_offset_from and sub_ptr both existing (they showed up at similar times so this union was never made). Adding these is a logical (and perhaps final) precondition of stabilizing ptr_sub_ptr (rust-lang#95892). Original PR by ``@Gankra`` (rust-lang#121919), I am just reviving it. The 2nd commit (with a small docs tweak) is by me.
2 parents 705cfe0 + c388655 commit 03596a1

File tree

3 files changed

+66
-9
lines changed

3 files changed

+66
-9
lines changed

library/core/src/ptr/const_ptr.rs

+22-3
Original file line numberDiff line numberDiff line change
@@ -582,7 +582,7 @@ impl<T: ?Sized> *const T {
582582
intrinsics::ptr_mask(self.cast::<()>(), mask).with_metadata_of(self)
583583
}
584584

585-
/// Calculates the distance between two pointers. The returned value is in
585+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
586586
/// units of T: the distance in bytes divided by `mem::size_of::<T>()`.
587587
///
588588
/// This is equivalent to `(self as isize - origin as isize) / (mem::size_of::<T>() as isize)`,
@@ -677,7 +677,7 @@ impl<T: ?Sized> *const T {
677677
unsafe { intrinsics::ptr_offset_from(self, origin) }
678678
}
679679

680-
/// Calculates the distance between two pointers. The returned value is in
680+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
681681
/// units of **bytes**.
682682
///
683683
/// This is purely a convenience for casting to a `u8` pointer and
@@ -695,7 +695,7 @@ impl<T: ?Sized> *const T {
695695
unsafe { self.cast::<u8>().offset_from(origin.cast::<u8>()) }
696696
}
697697

698-
/// Calculates the distance between two pointers, *where it's known that
698+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
699699
/// `self` is equal to or greater than `origin`*. The returned value is in
700700
/// units of T: the distance in bytes is divided by `mem::size_of::<T>()`.
701701
///
@@ -790,6 +790,25 @@ impl<T: ?Sized> *const T {
790790
unsafe { intrinsics::ptr_offset_from_unsigned(self, origin) }
791791
}
792792

793+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
794+
/// `self` is equal to or greater than `origin`*. The returned value is in
795+
/// units of **bytes**.
796+
///
797+
/// This is purely a convenience for casting to a `u8` pointer and
798+
/// using [`sub_ptr`][pointer::sub_ptr] on it. See that method for
799+
/// documentation and safety requirements.
800+
///
801+
/// For non-`Sized` pointees this operation considers only the data pointers,
802+
/// ignoring the metadata.
803+
#[unstable(feature = "ptr_sub_ptr", issue = "95892")]
804+
#[rustc_const_unstable(feature = "const_ptr_sub_ptr", issue = "95892")]
805+
#[inline]
806+
#[cfg_attr(miri, track_caller)] // even without panics, this helps for Miri backtraces
807+
pub const unsafe fn byte_sub_ptr<U: ?Sized>(self, origin: *const U) -> usize {
808+
// SAFETY: the caller must uphold the safety contract for `sub_ptr`.
809+
unsafe { self.cast::<u8>().sub_ptr(origin.cast::<u8>()) }
810+
}
811+
793812
/// Returns whether two pointers are guaranteed to be equal.
794813
///
795814
/// At runtime this function behaves like `Some(self == other)`.

library/core/src/ptr/mut_ptr.rs

+22-3
Original file line numberDiff line numberDiff line change
@@ -746,7 +746,7 @@ impl<T: ?Sized> *mut T {
746746
(self as *const T).guaranteed_ne(other as _)
747747
}
748748

749-
/// Calculates the distance between two pointers. The returned value is in
749+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
750750
/// units of T: the distance in bytes divided by `mem::size_of::<T>()`.
751751
///
752752
/// This is equivalent to `(self as isize - origin as isize) / (mem::size_of::<T>() as isize)`,
@@ -839,7 +839,7 @@ impl<T: ?Sized> *mut T {
839839
unsafe { (self as *const T).offset_from(origin) }
840840
}
841841

842-
/// Calculates the distance between two pointers. The returned value is in
842+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
843843
/// units of **bytes**.
844844
///
845845
/// This is purely a convenience for casting to a `u8` pointer and
@@ -857,7 +857,7 @@ impl<T: ?Sized> *mut T {
857857
unsafe { self.cast::<u8>().offset_from(origin.cast::<u8>()) }
858858
}
859859

860-
/// Calculates the distance between two pointers, *where it's known that
860+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
861861
/// `self` is equal to or greater than `origin`*. The returned value is in
862862
/// units of T: the distance in bytes is divided by `mem::size_of::<T>()`.
863863
///
@@ -930,6 +930,25 @@ impl<T: ?Sized> *mut T {
930930
unsafe { (self as *const T).sub_ptr(origin) }
931931
}
932932

933+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
934+
/// `self` is equal to or greater than `origin`*. The returned value is in
935+
/// units of **bytes**.
936+
///
937+
/// This is purely a convenience for casting to a `u8` pointer and
938+
/// using [`sub_ptr`][pointer::sub_ptr] on it. See that method for
939+
/// documentation and safety requirements.
940+
///
941+
/// For non-`Sized` pointees this operation considers only the data pointers,
942+
/// ignoring the metadata.
943+
#[unstable(feature = "ptr_sub_ptr", issue = "95892")]
944+
#[rustc_const_unstable(feature = "const_ptr_sub_ptr", issue = "95892")]
945+
#[inline]
946+
#[cfg_attr(miri, track_caller)] // even without panics, this helps for Miri backtraces
947+
pub const unsafe fn byte_sub_ptr<U: ?Sized>(self, origin: *mut U) -> usize {
948+
// SAFETY: the caller must uphold the safety contract for `byte_sub_ptr`.
949+
unsafe { (self as *const T).byte_sub_ptr(origin) }
950+
}
951+
933952
/// Adds an unsigned offset to a pointer.
934953
///
935954
/// This can only move the pointer forward (or not move it). If you need to move forward or

library/core/src/ptr/non_null.rs

+22-3
Original file line numberDiff line numberDiff line change
@@ -676,7 +676,7 @@ impl<T: ?Sized> NonNull<T> {
676676
unsafe { NonNull { pointer: self.pointer.byte_sub(count) } }
677677
}
678678

679-
/// Calculates the distance between two pointers. The returned value is in
679+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
680680
/// units of T: the distance in bytes divided by `mem::size_of::<T>()`.
681681
///
682682
/// This is equivalent to `(self as isize - origin as isize) / (mem::size_of::<T>() as isize)`,
@@ -773,7 +773,7 @@ impl<T: ?Sized> NonNull<T> {
773773
unsafe { self.pointer.offset_from(origin.pointer) }
774774
}
775775

776-
/// Calculates the distance between two pointers. The returned value is in
776+
/// Calculates the distance between two pointers within the same allocation. The returned value is in
777777
/// units of **bytes**.
778778
///
779779
/// This is purely a convenience for casting to a `u8` pointer and
@@ -793,7 +793,7 @@ impl<T: ?Sized> NonNull<T> {
793793

794794
// N.B. `wrapping_offset``, `wrapping_add`, etc are not implemented because they can wrap to null
795795

796-
/// Calculates the distance between two pointers, *where it's known that
796+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
797797
/// `self` is equal to or greater than `origin`*. The returned value is in
798798
/// units of T: the distance in bytes is divided by `mem::size_of::<T>()`.
799799
///
@@ -866,6 +866,25 @@ impl<T: ?Sized> NonNull<T> {
866866
unsafe { self.pointer.sub_ptr(subtracted.pointer) }
867867
}
868868

869+
/// Calculates the distance between two pointers within the same allocation, *where it's known that
870+
/// `self` is equal to or greater than `origin`*. The returned value is in
871+
/// units of **bytes**.
872+
///
873+
/// This is purely a convenience for casting to a `u8` pointer and
874+
/// using [`sub_ptr`][NonNull::sub_ptr] on it. See that method for
875+
/// documentation and safety requirements.
876+
///
877+
/// For non-`Sized` pointees this operation considers only the data pointers,
878+
/// ignoring the metadata.
879+
#[inline(always)]
880+
#[cfg_attr(miri, track_caller)] // even without panics, this helps for Miri backtraces
881+
#[unstable(feature = "ptr_sub_ptr", issue = "95892")]
882+
#[rustc_const_unstable(feature = "const_ptr_sub_ptr", issue = "95892")]
883+
pub const unsafe fn byte_sub_ptr<U: ?Sized>(self, origin: NonNull<U>) -> usize {
884+
// SAFETY: the caller must uphold the safety contract for `byte_sub_ptr`.
885+
unsafe { self.pointer.byte_sub_ptr(origin.pointer) }
886+
}
887+
869888
/// Reads the value from `self` without moving it. This leaves the
870889
/// memory in `self` unchanged.
871890
///

0 commit comments

Comments
 (0)