Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cfi: Store type erasure witness for Argument #115954

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
32 changes: 32 additions & 0 deletions library/core/src/fmt/rt.rs
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,12 @@ enum ArgumentType<'a> {
// was derived from a `&'a T`.
value: NonNull<()>,
formatter: unsafe fn(NonNull<()>, &mut Formatter<'_>) -> Result,
#[cfg(any(sanitize = "cfi", sanitize = "kcfi"))]
cast_stub: unsafe fn(
unsafe fn(NonNull<()>, &mut Formatter<'_>) -> Result,
NonNull<()>,
f: &mut Formatter<'_>,
) -> Result,
_lifetime: PhantomData<&'a ()>,
},
Count(usize),
Expand All @@ -93,6 +99,21 @@ pub struct Argument<'a> {
ty: ArgumentType<'a>,
}

/// This function acts as a witness to an earlier type erasure when constructing an
/// `ArgumentType`. The type parameter `T` should be instantiated to the erased type.
/// SAFETY: This function should be called on a value and formatter which underwent a
/// T->Opaque cast at their definition site.
#[cfg(any(sanitize = "cfi", sanitize = "kcfi"))]
unsafe fn cast_stub<T>(
formatter: unsafe fn(NonNull<()>, &mut Formatter<'_>) -> Result,
value: NonNull<()>,
f: &mut Formatter<'_>,
) -> Result {
let value: &T = mem::transmute(value);
let formatter: fn(&T, &mut Formatter<'_>) -> Result = mem::transmute(formatter);
formatter(value, f)
}

#[rustc_diagnostic_item = "ArgumentMethods"]
impl<'a> Argument<'a> {
#[inline(always)]
Expand All @@ -104,6 +125,8 @@ impl<'a> Argument<'a> {
value: NonNull::from(x).cast(),
// SAFETY: function pointers always have the same layout.
formatter: unsafe { mem::transmute(f) },
#[cfg(any(sanitize = "cfi", sanitize = "kcfi"))]
cast_stub: cast_stub::<T>,
_lifetime: PhantomData,
},
}
Expand Down Expand Up @@ -163,6 +186,15 @@ impl<'a> Argument<'a> {
#[inline(always)]
pub(super) unsafe fn fmt(&self, f: &mut Formatter<'_>) -> Result {
match self.ty {
#[cfg(any(sanitize = "cfi", sanitize = "kcfi"))]
// SAFETY:
// This is the `cast_stub` that was prepared alongside the
// formatter and value, so it should have the `T` instantiated to
// the erased type that `value` points at.
ArgumentType::Placeholder { formatter, value, cast_stub, .. } => unsafe {
cast_stub(formatter, value, f)
},
#[cfg(not(any(sanitize = "cfi", sanitize = "kcfi")))]
// SAFETY:
// Because of the invariant that if `formatter` had the type
// `fn(&T, _) -> _` then `value` has type `&'b T` where `'b` is
Expand Down
Loading