Skip to content

Commit 87c9b3f

Browse files
Rollup merge of #110126 - compiler-errors:new-solver-safe-transmute, r=oli-obk
Support safe transmute in new solver Basically copies the same implementation as the old solver, but instead of looking for param types, we look for type or const placeholders.
2 parents 2f5440c + 4adee91 commit 87c9b3f

17 files changed

+1184
-121
lines changed

compiler/rustc_middle/src/ty/visit.rs

+3
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,9 @@ pub trait TypeVisitableExt<'tcx>: TypeVisitable<TyCtxt<'tcx>> {
8383
| TypeFlags::HAS_CT_PLACEHOLDER,
8484
)
8585
}
86+
fn has_non_region_placeholders(&self) -> bool {
87+
self.has_type_flags(TypeFlags::HAS_TY_PLACEHOLDER | TypeFlags::HAS_CT_PLACEHOLDER)
88+
}
8689
fn needs_subst(&self) -> bool {
8790
self.has_type_flags(TypeFlags::NEEDS_SUBST)
8891
}

compiler/rustc_trait_selection/src/solve/assembly/mod.rs

+7
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,11 @@ pub(super) trait GoalKind<'tcx>: TypeFoldable<TyCtxt<'tcx>> + Copy + Eq {
225225
ecx: &mut EvalCtxt<'_, 'tcx>,
226226
goal: Goal<'tcx, Self>,
227227
) -> QueryResult<'tcx>;
228+
229+
fn consider_builtin_transmute_candidate(
230+
ecx: &mut EvalCtxt<'_, 'tcx>,
231+
goal: Goal<'tcx, Self>,
232+
) -> QueryResult<'tcx>;
228233
}
229234

230235
impl<'tcx> EvalCtxt<'_, 'tcx> {
@@ -373,6 +378,8 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
373378
G::consider_builtin_discriminant_kind_candidate(self, goal)
374379
} else if lang_items.destruct_trait() == Some(trait_def_id) {
375380
G::consider_builtin_destruct_candidate(self, goal)
381+
} else if lang_items.transmute_trait() == Some(trait_def_id) {
382+
G::consider_builtin_transmute_candidate(self, goal)
376383
} else {
377384
Err(NoSolution)
378385
};

compiler/rustc_trait_selection/src/solve/eval_ctxt.rs

+21
Original file line numberDiff line numberDiff line change
@@ -639,4 +639,25 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
639639
crate::traits::wf::unnormalized_obligations(self.infcx, param_env, arg)
640640
.map(|obligations| obligations.into_iter().map(|obligation| obligation.into()))
641641
}
642+
643+
pub(super) fn is_transmutable(
644+
&self,
645+
src_and_dst: rustc_transmute::Types<'tcx>,
646+
scope: Ty<'tcx>,
647+
assume: rustc_transmute::Assume,
648+
) -> Result<Certainty, NoSolution> {
649+
// FIXME(transmutability): This really should be returning nested goals for `Answer::If*`
650+
match rustc_transmute::TransmuteTypeEnv::new(self.infcx).is_transmutable(
651+
ObligationCause::dummy(),
652+
ty::Binder::dummy(src_and_dst),
653+
scope,
654+
assume,
655+
) {
656+
rustc_transmute::Answer::Yes => Ok(Certainty::Yes),
657+
rustc_transmute::Answer::No(_)
658+
| rustc_transmute::Answer::IfTransmutable { .. }
659+
| rustc_transmute::Answer::IfAll(_)
660+
| rustc_transmute::Answer::IfAny(_) => Err(NoSolution),
661+
}
662+
}
642663
}

compiler/rustc_trait_selection/src/solve/project_goals.rs

+7
Original file line numberDiff line numberDiff line change
@@ -524,6 +524,13 @@ impl<'tcx> assembly::GoalKind<'tcx> for ProjectionPredicate<'tcx> {
524524
) -> QueryResult<'tcx> {
525525
bug!("`Destruct` does not have an associated type: {:?}", goal);
526526
}
527+
528+
fn consider_builtin_transmute_candidate(
529+
_ecx: &mut EvalCtxt<'_, 'tcx>,
530+
goal: Goal<'tcx, Self>,
531+
) -> QueryResult<'tcx> {
532+
bug!("`BikeshedIntrinsicFrom` does not have an associated type: {:?}", goal)
533+
}
527534
}
528535

529536
/// This behavior is also implemented in `rustc_ty_utils` and in the old `project` code.

compiler/rustc_trait_selection/src/solve/trait_goals.rs

+29
Original file line numberDiff line numberDiff line change
@@ -598,6 +598,35 @@ impl<'tcx> assembly::GoalKind<'tcx> for TraitPredicate<'tcx> {
598598
Err(NoSolution)
599599
}
600600
}
601+
602+
fn consider_builtin_transmute_candidate(
603+
ecx: &mut EvalCtxt<'_, 'tcx>,
604+
goal: Goal<'tcx, Self>,
605+
) -> QueryResult<'tcx> {
606+
// `rustc_transmute` does not have support for type or const params
607+
if goal.has_non_region_placeholders() {
608+
return Err(NoSolution);
609+
}
610+
611+
// Erase regions because we compute layouts in `rustc_transmute`,
612+
// which will ICE for region vars.
613+
let substs = ecx.tcx().erase_regions(goal.predicate.trait_ref.substs);
614+
615+
let Some(assume) = rustc_transmute::Assume::from_const(
616+
ecx.tcx(),
617+
goal.param_env,
618+
substs.const_at(3),
619+
) else {
620+
return Err(NoSolution);
621+
};
622+
623+
let certainty = ecx.is_transmutable(
624+
rustc_transmute::Types { dst: substs.type_at(0), src: substs.type_at(1) },
625+
substs.type_at(2),
626+
assume,
627+
)?;
628+
ecx.evaluate_added_goals_and_make_canonical_response(certainty)
629+
}
601630
}
602631

603632
impl<'tcx> EvalCtxt<'_, 'tcx> {

tests/ui/transmutability/primitives/bool.stderr tests/ui/transmutability/primitives/bool.current.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
11
error[E0277]: `u8` cannot be safely transmuted into `bool` in the defining scope of `assert::Context`.
2-
--> $DIR/bool.rs:22:35
2+
--> $DIR/bool.rs:24:35
33
|
44
LL | assert::is_transmutable::<u8, bool>();
55
| ^^^^ `u8` cannot be safely transmuted into `bool` in the defining scope of `assert::Context`.
66
|
77
= help: the trait `BikeshedIntrinsicFrom<u8, assert::Context, Assume { alignment: false, lifetimes: false, safety: true, validity: false }>` is not implemented for `bool`
88
note: required by a bound in `is_transmutable`
9-
--> $DIR/bool.rs:12:14
9+
--> $DIR/bool.rs:14:14
1010
|
1111
LL | pub fn is_transmutable<Src, Dst>()
1212
| --------------- required by a bound in this function
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
error[E0277]: `u8` cannot be safely transmuted into `bool` in the defining scope of `assert::Context`.
2+
--> $DIR/bool.rs:24:35
3+
|
4+
LL | assert::is_transmutable::<u8, bool>();
5+
| ^^^^ `u8` cannot be safely transmuted into `bool` in the defining scope of `assert::Context`.
6+
|
7+
= help: the trait `BikeshedIntrinsicFrom<u8, assert::Context, Assume { alignment: false, lifetimes: false, safety: true, validity: false }>` is not implemented for `bool`
8+
note: required by a bound in `is_transmutable`
9+
--> $DIR/bool.rs:14:14
10+
|
11+
LL | pub fn is_transmutable<Src, Dst>()
12+
| --------------- required by a bound in this function
13+
LL | where
14+
LL | Dst: BikeshedIntrinsicFrom<Src, Context, { Assume::SAFETY }>
15+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ required by this bound in `is_transmutable`
16+
17+
error: aborting due to previous error
18+
19+
For more information about this error, try `rustc --explain E0277`.

tests/ui/transmutability/primitives/bool.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
1+
// revisions: current next
2+
//[next] compile-flags: -Ztrait-solver=next
3+
14
#![crate_type = "lib"]
25
#![feature(transmutability)]
36
#![allow(dead_code)]
47
#![allow(incomplete_features)]
5-
68
mod assert {
79
use std::mem::{Assume, BikeshedIntrinsicFrom};
810
pub struct Context;

0 commit comments

Comments
 (0)