-
Notifications
You must be signed in to change notification settings - Fork 13k
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
Add an intrinsic for ptr::from_raw_parts(_mut)
#123840
Conversation
Some changes occurred to MIR optimizations cc @rust-lang/wg-mir-opt This PR changes MIR cc @oli-obk, @RalfJung, @JakobDegen, @davidtwco, @celinval, @vakaras Some changes occurred to the CTFE / Miri engine cc @rust-lang/miri This PR changes Stable MIR |
_5 = ((*_1).1: usize); | ||
StorageLive(_6); | ||
_6 = _4 as *const () (PtrToPtr); | ||
_7 = *const [u8] from (_6, _5); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For comparison, on nightly this line is
_9 = std::ptr::metadata::PtrComponents::<[u8]> { data_pointer: _7, metadata: _3 };
_8 = std::ptr::metadata::PtrRepr::<[u8]> { const_ptr: move _9 };
_6 = (_8.0: *const [u8]);
if you put the threshold high enough to see it https://rust.godbolt.org/z/baT3bqEze
// SAFETY: Accessing the value from the `PtrRepr` union is safe since *const T | ||
// and PtrComponents<T> have the same memory layouts. Only std can make this | ||
// guarantee. | ||
unsafe { PtrRepr { components: PtrComponents { data_pointer, metadata } }.const_ptr } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sadly another PR to add an intrinsic for https://doc.rust-lang.org/nightly/std/ptr/fn.metadata.html will be needed before this can delete PtrRepr
and PtrComponents
. But the scroll bar was long enough already that I didn't want to do that in this change.
This comment has been minimized.
This comment has been minimized.
@@ -910,6 +911,8 @@ impl<'body, 'tcx> VnState<'body, 'tcx> { | |||
} | |||
// Do not track unions. | |||
AggregateKind::Adt(_, _, _, _, Some(_)) => return None, | |||
// FIXME: Do the extra work to GVN `from_raw_parts` | |||
AggregateKind::RawPtr(..) => return None, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I started down the path of an AggregateTy::RawPtr(ty)
, then decided that since GVN couldn't track the old code because of the union, I didn't need to shave this yak too right now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok.
Some changes occurred in compiler/rustc_codegen_cranelift cc @bjorn3 |
This comment has been minimized.
This comment has been minimized.
3cf58a9
to
aa35d1b
Compare
This comment has been minimized.
This comment has been minimized.
aa35d1b
to
c79c53a
Compare
Pulling something out of a review comment to provide some context:
Yeah, since
|
c105f86
to
ada5a3d
Compare
This comment has been minimized.
This comment has been minimized.
ada5a3d
to
70dee04
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for investigating this @scottmcm!
// raw pointer rvalues are checked in the MIR validator, | ||
// and since they only deal in things that have no lifetimes, | ||
// there's nothing that needs checking here. | ||
return; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If RawPtr
is only created by an intrinsic, this should bug!
.
@@ -717,6 +717,24 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> { | |||
OperandRef { val: OperandValue::Immediate(static_), layout } | |||
} | |||
mir::Rvalue::Use(ref operand) => self.codegen_operand(bx, operand), | |||
mir::Rvalue::Aggregate(box mir::AggregateKind::RawPtr(..), ref fields) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: should this appear in an earlier commit?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The first few commits here aren't split up super great, just checkpoints where it built and the tests pass. The tests "passing" for them wasn't terribly meaningful, though, since without an intrinsic for it and without the lowering using it, then the code wasn't actually ever running.
LMK if you're fine leaving it as-is or would rather I just merge the first few.
@@ -36,6 +36,7 @@ impl<'tcx> MirPass<'tcx> for InstSimplify { | |||
ctx.simplify_bool_cmp(&statement.source_info, rvalue); | |||
ctx.simplify_ref_deref(&statement.source_info, rvalue); | |||
ctx.simplify_len(&statement.source_info, rvalue); | |||
ctx.simplify_ptr_aggregate(&statement.source_info, rvalue); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could the new opt be in a standalone commit, to see its effect on tests?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added a new mir-opt test for byte_add
on pointers to show what this does: https://github.com/rust-lang/rust/pull/123840/files#diff-89ce90739dcfc9de9ddd90700c234f28642574e2baad03a1bd1b0b6fee9e4016
/// *Both* operands must always be included, even the unit value if this is | ||
/// creating a thin pointer. If you're just converting between thin pointers, | ||
/// you may want an [`Rvalue::Cast`] with [`CastKind::PtrToPtr`] instead. | ||
RawPtr(Ty<'tcx>, Mutability), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In terms of generality, should this support only raw pointers, or borrows too? Or to be left as a follow-up experiment?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right now I think that Rvalue::Ref
is the only thing that makes references (well, I guess Transmute
s too), so I'd rather leave it on just raw pointers, for now, with the &*_2
dance for turning it back into a reference. Especially since I don't want to worry about making sure that whatever SB/TB tagging is needed is emitting in Miri for things.
It's also nice in the library for the intrinsic to be able to be safe, which it can't if it makes references. That could be done with separate intrinsics so the raw pointer one can still be safe, but that's again making the size of the change bigger again, so I think not doing it in this PR, at least, is best.
@@ -910,6 +911,8 @@ impl<'body, 'tcx> VnState<'body, 'tcx> { | |||
} | |||
// Do not track unions. | |||
AggregateKind::Adt(_, _, _, _, Some(_)) => return None, | |||
// FIXME: Do the extra work to GVN `from_raw_parts` | |||
AggregateKind::RawPtr(..) => return None, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok.
bf66fec
to
1712f5c
Compare
Sorry for all the pushes, @cjgillot . Here's the diff since your review: https://github.com/rust-lang/rust/compare/70dee04ff7a06bebf69a63a373ee1f046e6b5878..1712f5c9aa7060425ed534f7d062992ab4055971 |
StorageDead(_3); | ||
StorageLive(_5); | ||
_5 = _4 as *const () (PtrToPtr); | ||
_0 = _4 as *const u32 (PtrToPtr); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As a serendipitous bonus, this PR means that using byte_add
on a thin pointer is no longer worse in MIR than casting to *const u8
and offsetting yourself, as with unit metadata there's no longer any union
dances needed, just the straight-forward casting.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh, even better -- with #123949 it also drops the irrelevant cast to *const ()
.
library/core/src/intrinsics.rs
Outdated
#[rustc_intrinsic_must_be_overridden] | ||
#[cfg(not(bootstrap))] | ||
pub const fn aggregate_raw_ptr<P: AggregateRawPtr<D, Metadata = M>, D, M>(_data: D, _meta: M) -> P { | ||
// No fallback because `libcore` doesn't want to know the layout |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// No fallback because `libcore` doesn't want to know the layout | |
// To implement a fallback we'd have to assume the layout, but the whole point of this intrinsic is to not do that. |
We do want to know the layout, we just can't :p
@@ -147,6 +157,30 @@ impl<'tcx> InstSimplifyContext<'tcx, '_> { | |||
} | |||
} | |||
|
|||
/// Transform "Aggregate(RawPtr, \[p, ()\])" ==> "Cast(PtrToPtr, p)". |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This escaping is really unfortunate. Is it possible to use backticks for code formatting instead?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure. I was copying the other two method headers that also used "
, so I've updated all three of them to backticks.
Huh, curious that it still passes CI. I'll take a look. |
ah must have been a different PR, the rollup without this one also failed, but the passing test was something with RefTarget which threw me off 😅 @bors r=cjgillot |
…iaskrgr Rollup of 5 pull requests Successful merges: - rust-lang#123840 (Add an intrinsic for `ptr::from_raw_parts(_mut)`) - rust-lang#124224 (cleanup: unnecessary clone during lower generics args) - rust-lang#124229 (Add gnullvm targets to manifest) - rust-lang#124231 (remove from reviewers) - rust-lang#124235 (Move some tests) r? `@ghost` `@rustbot` modify labels: rollup
Rollup merge of rust-lang#123840 - scottmcm:aggregate-kind-rawptr, r=cjgillot Add an intrinsic for `ptr::from_raw_parts(_mut)` Fixes rust-lang#123174 cc `@CAD97` `@saethlin` r? `@cjgillot` As suggested in rust-lang#123190 (comment), this adds a new `AggregateKind::RawPtr` for creating a pointer from its data pointer and its metadata. That means that `slice::from_raw_parts` and friends no longer need to hard-code pointer layout into `libcore`, and because it no longer does union hacks the MIR is shorter and more amenable to optimizations.
…ts, r=<try> Avoid a cast in `ptr::slice_from_raw_parts(_mut)` Casting to `*const ()` or `*mut ()` is no longer needed after rust-lang#123840 so let's make the MIR smaller (and more inline-able, as seen in the tests). If [ACP#362](rust-lang/libs-team#362) goes through we can keep calling `ptr::from_raw_parts(_mut)` in these also without the cast, but that hasn't had any libs-api attention yet, so I'm not waiting on it.
…ts, r=joboet Avoid a cast in `ptr::slice_from_raw_parts(_mut)` Casting to `*const ()` or `*mut ()` is no longer needed after rust-lang#123840 so let's make the MIR smaller (and more inline-able, as seen in the tests). If [ACP#362](rust-lang/libs-team#362) goes through we can keep calling `ptr::from_raw_parts(_mut)` in these also without the cast, but that hasn't had any libs-api attention yet, so I'm not waiting on it.
Unify `Rvalue::Aggregate` paths in cg_ssa In rust-lang#123840 and rust-lang#123886 I added two different codepaths for `Rvalue::Aggregate` in `cg_ssa`. This merges them into one, since raw pointers are also immediates that can be built from the immediates of their "fields".
Addresses the new `AggregateKind::RawPtr` added in rust-lang/rust#123840. Resolves #3161 By submitting this pull request, I confirm that my contribution is made under the terms of the Apache 2.0 and MIT licenses. --------- Co-authored-by: Kareem Khazem <[email protected]> Co-authored-by: Michael Tautschnig <[email protected]>
Fix ICE in non-operand `aggregate_raw_ptr` intrinsic codegen Introduced in rust-lang#123840 Found in rust-lang#121571, cc `@clarfonthey`
Rollup merge of rust-lang#125184 - scottmcm:fix-thin-ptr-ice, r=jieyouxu Fix ICE in non-operand `aggregate_raw_ptr` intrinsic codegen Introduced in rust-lang#123840 Found in rust-lang#121571, cc `@clarfonthey`
Add an intrinsic for `ptr::metadata` The follow-up to rust-lang#123840, so we can remove `PtrComponents` and `PtrRepr` from libcore entirely (well, after a bootstrap update). As discussed in <https://rust-lang.zulipchat.com/#narrow/stream/189540-t-compiler.2Fwg-mir-opt/topic/.60ptr_metadata.60.20in.20MIR/near/435637808>, this introduces `UnOp::PtrMetadata` taking a raw pointer and returning the associated metadata value. By no longer going through a `union`, this should also help future PRs better optimize pointer operations. r? `@oli-obk`
Add an intrinsic for `ptr::metadata` The follow-up to rust-lang#123840, so we can remove `PtrComponents` and `PtrRepr` from libcore entirely (well, after a bootstrap update). As discussed in <https://rust-lang.zulipchat.com/#narrow/stream/189540-t-compiler.2Fwg-mir-opt/topic/.60ptr_metadata.60.20in.20MIR/near/435637808>, this introduces `UnOp::PtrMetadata` taking a raw pointer and returning the associated metadata value. By no longer going through a `union`, this should also help future PRs better optimize pointer operations. r? `@oli-obk`
Add an intrinsic for `ptr::metadata` The follow-up to rust-lang#123840, so we can remove `PtrComponents` and `PtrRepr` from libcore entirely (well, after a bootstrap update). As discussed in <https://rust-lang.zulipchat.com/#narrow/stream/189540-t-compiler.2Fwg-mir-opt/topic/.60ptr_metadata.60.20in.20MIR/near/435637808>, this introduces `UnOp::PtrMetadata` taking a raw pointer and returning the associated metadata value. By no longer going through a `union`, this should also help future PRs better optimize pointer operations. r? ``@oli-obk``
Rollup merge of rust-lang#124251 - scottmcm:unop-ptr-metadata, r=oli-obk Add an intrinsic for `ptr::metadata` The follow-up to rust-lang#123840, so we can remove `PtrComponents` and `PtrRepr` from libcore entirely (well, after a bootstrap update). As discussed in <https://rust-lang.zulipchat.com/#narrow/stream/189540-t-compiler.2Fwg-mir-opt/topic/.60ptr_metadata.60.20in.20MIR/near/435637808>, this introduces `UnOp::PtrMetadata` taking a raw pointer and returning the associated metadata value. By no longer going through a `union`, this should also help future PRs better optimize pointer operations. r? ``@oli-obk``
Add an intrinsic for `ptr::metadata` The follow-up to rust-lang#123840, so we can remove `PtrComponents` and `PtrRepr` from libcore entirely (well, after a bootstrap update). As discussed in <https://rust-lang.zulipchat.com/#narrow/stream/189540-t-compiler.2Fwg-mir-opt/topic/.60ptr_metadata.60.20in.20MIR/near/435637808>, this introduces `UnOp::PtrMetadata` taking a raw pointer and returning the associated metadata value. By no longer going through a `union`, this should also help future PRs better optimize pointer operations. r? ``@oli-obk``
Fixes #123174
cc @CAD97 @saethlin
r? @cjgillot
As suggested in #123190 (comment), this adds a new
AggregateKind::RawPtr
for creating a pointer from its data pointer and its metadata.That means that
slice::from_raw_parts
and friends no longer need to hard-code pointer layout intolibcore
, and because it no longer does union hacks the MIR is shorter and more amenable to optimizations.