Doc fixes from review
This commit is contained in:
parent
c155d5149f
commit
8d49e948a8
2 changed files with 26 additions and 7 deletions
|
@ -40,8 +40,8 @@ pub use drop::TaggedPtr;
|
||||||
/// [`into_ptr`] must be valid for writes (and thus calling [`NonNull::as_mut`]
|
/// [`into_ptr`] must be valid for writes (and thus calling [`NonNull::as_mut`]
|
||||||
/// on it must be safe).
|
/// on it must be safe).
|
||||||
///
|
///
|
||||||
/// The [`BITS`] constant must be correct. At least [`BITS`] least significant
|
/// The [`BITS`] constant must be correct. [`BITS`] least-significant bits,
|
||||||
/// bits, must be zero on all pointers returned from [`into_ptr`].
|
/// must be zero on all pointers returned from [`into_ptr`].
|
||||||
///
|
///
|
||||||
/// For example, if the alignment of [`Self::Target`] is 2, then `BITS` should be 1.
|
/// For example, if the alignment of [`Self::Target`] is 2, then `BITS` should be 1.
|
||||||
///
|
///
|
||||||
|
@ -52,9 +52,12 @@ pub use drop::TaggedPtr;
|
||||||
/// [`Self::Target`]: Deref::Target
|
/// [`Self::Target`]: Deref::Target
|
||||||
/// [`DerefMut`]: std::ops::DerefMut
|
/// [`DerefMut`]: std::ops::DerefMut
|
||||||
pub unsafe trait Pointer: Deref {
|
pub unsafe trait Pointer: Deref {
|
||||||
/// Number of unused (always zero) **least significant bits** in this
|
/// Number of unused (always zero) **least-significant bits** in this
|
||||||
/// pointer, usually related to the pointees alignment.
|
/// pointer, usually related to the pointees alignment.
|
||||||
///
|
///
|
||||||
|
/// For example if [`BITS`] = `2`, then given `ptr = Self::into_ptr(..)`,
|
||||||
|
/// `ptr.addr() & 0b11 == 0` must be true.
|
||||||
|
///
|
||||||
/// Most likely the value you want to use here is the following, unless
|
/// Most likely the value you want to use here is the following, unless
|
||||||
/// your [`Self::Target`] type is unsized (e.g., `ty::List<T>` in rustc)
|
/// your [`Self::Target`] type is unsized (e.g., `ty::List<T>` in rustc)
|
||||||
/// or your pointer is over/under aligned, in which case you'll need to
|
/// or your pointer is over/under aligned, in which case you'll need to
|
||||||
|
@ -71,6 +74,7 @@ pub unsafe trait Pointer: Deref {
|
||||||
/// # }
|
/// # }
|
||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
|
/// [`BITS`]: Pointer::BITS
|
||||||
/// [`Self::Target`]: Deref::Target
|
/// [`Self::Target`]: Deref::Target
|
||||||
const BITS: usize;
|
const BITS: usize;
|
||||||
|
|
||||||
|
@ -105,7 +109,7 @@ pub unsafe trait Pointer: Deref {
|
||||||
///
|
///
|
||||||
/// The [`BITS`] constant must be correct.
|
/// The [`BITS`] constant must be correct.
|
||||||
///
|
///
|
||||||
/// No more than [`BITS`] least significant bits may be set in the returned usize.
|
/// No more than [`BITS`] least-significant bits may be set in the returned usize.
|
||||||
///
|
///
|
||||||
/// [`BITS`]: Tag::BITS
|
/// [`BITS`]: Tag::BITS
|
||||||
pub unsafe trait Tag: Copy {
|
pub unsafe trait Tag: Copy {
|
||||||
|
|
|
@ -33,7 +33,7 @@ where
|
||||||
/// those are embeddable in instruction encoding, for example:
|
/// those are embeddable in instruction encoding, for example:
|
||||||
///
|
///
|
||||||
/// ```asm
|
/// ```asm
|
||||||
/// // (https://godbolt.org/z/jqcYPWEr3)
|
/// // (<https://godbolt.org/z/jqcYPWEr3>)
|
||||||
/// example::shift_read3:
|
/// example::shift_read3:
|
||||||
/// mov eax, dword ptr [8*rdi]
|
/// mov eax, dword ptr [8*rdi]
|
||||||
/// ret
|
/// ret
|
||||||
|
@ -49,7 +49,8 @@ where
|
||||||
/// - `shift_read3` uses `<< 3` (the tag is in the most-significant bits)
|
/// - `shift_read3` uses `<< 3` (the tag is in the most-significant bits)
|
||||||
/// - `mask_read3` uses `& !0b111` (the tag is in the least-significant bits)
|
/// - `mask_read3` uses `& !0b111` (the tag is in the least-significant bits)
|
||||||
///
|
///
|
||||||
/// The shift approach thus produces less instructions and is likely faster.
|
/// The shift approach thus produces less instructions and is likely faster
|
||||||
|
/// (see <https://godbolt.org/z/Y913sMdWb>).
|
||||||
///
|
///
|
||||||
/// Encoding diagram:
|
/// Encoding diagram:
|
||||||
/// ```text
|
/// ```text
|
||||||
|
@ -66,12 +67,21 @@ where
|
||||||
tag_ghost: PhantomData<T>,
|
tag_ghost: PhantomData<T>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Note that even though `CopyTaggedPtr` is only really expected to work with
|
||||||
|
// `P: Copy`, can't add `P: Copy` bound, because `CopyTaggedPtr` is used in the
|
||||||
|
// `TaggedPtr`'s implementation.
|
||||||
impl<P, T, const CP: bool> CopyTaggedPtr<P, T, CP>
|
impl<P, T, const CP: bool> CopyTaggedPtr<P, T, CP>
|
||||||
where
|
where
|
||||||
P: Pointer,
|
P: Pointer,
|
||||||
T: Tag,
|
T: Tag,
|
||||||
{
|
{
|
||||||
/// Tags `pointer` with `tag`.
|
/// Tags `pointer` with `tag`.
|
||||||
|
///
|
||||||
|
/// Note that this leaks `pointer`: it won't be dropped when
|
||||||
|
/// `CopyTaggedPtr` is dropped. If you have a pointer with a significant
|
||||||
|
/// drop, use [`TaggedPtr`] instead.
|
||||||
|
///
|
||||||
|
/// [`TaggedPtr`]: crate::tagged_ptr::TaggedPtr
|
||||||
pub fn new(pointer: P, tag: T) -> Self {
|
pub fn new(pointer: P, tag: T) -> Self {
|
||||||
Self { packed: Self::pack(P::into_ptr(pointer), tag), tag_ghost: PhantomData }
|
Self { packed: Self::pack(P::into_ptr(pointer), tag), tag_ghost: PhantomData }
|
||||||
}
|
}
|
||||||
|
@ -95,7 +105,8 @@ where
|
||||||
let tag = self.packed.addr().get() >> Self::TAG_BIT_SHIFT;
|
let tag = self.packed.addr().get() >> Self::TAG_BIT_SHIFT;
|
||||||
|
|
||||||
// Safety:
|
// Safety:
|
||||||
//
|
// The shift retrieves the original value from `T::into_usize`,
|
||||||
|
// satisfying `T::from_usize`'s preconditions.
|
||||||
unsafe { T::from_usize(tag) }
|
unsafe { T::from_usize(tag) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -152,6 +163,10 @@ where
|
||||||
//
|
//
|
||||||
// Semantically this is just `f(&self.pointer)` (where `self.pointer`
|
// Semantically this is just `f(&self.pointer)` (where `self.pointer`
|
||||||
// is non-packed original pointer).
|
// is non-packed original pointer).
|
||||||
|
//
|
||||||
|
// Note that even though `CopyTaggedPtr` is only really expected to
|
||||||
|
// work with `P: Copy`, we have to assume `P: ?Copy`, because
|
||||||
|
// `CopyTaggedPtr` is used in the `TaggedPtr`'s implementation.
|
||||||
let ptr = unsafe { ManuallyDrop::new(P::from_ptr(self.pointer_raw())) };
|
let ptr = unsafe { ManuallyDrop::new(P::from_ptr(self.pointer_raw())) };
|
||||||
f(&ptr)
|
f(&ptr)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue