Skip to content

Commit 71ef9ec

Browse files
committed
Auto merge of rust-lang#110311 - matthiaskrgr:rollup-kn2k5bq, r=matthiaskrgr
Rollup of 9 pull requests Successful merges: - rust-lang#109225 (Clarify that RUST_MIN_STACK may be internally cached) - rust-lang#109800 (Improve safe transmute error reporting) - rust-lang#110158 (Remove obsolete test case) - rust-lang#110180 (don't uniquify regions when canonicalizing) - rust-lang#110207 (Assemble `Unpin` candidates specially for generators in new solver) - rust-lang#110276 (Remove all but one of the spans in `BoundRegionKind::BrAnon`) - rust-lang#110279 (rustdoc: Correctly handle built-in compiler proc-macros as proc-macro and not macro) - rust-lang#110298 (Cover edge cases for {f32, f64}.hypot() docs) - rust-lang#110299 (Switch to `EarlyBinder` for `impl_subject` query) Failed merges: r? `@ghost` `@rustbot` modify labels: rollup
2 parents edcbb29 + 610bc68 commit 71ef9ec

File tree

52 files changed

+758
-1090
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

52 files changed

+758
-1090
lines changed

compiler/rustc_hir_typeck/src/generator_interior/mod.rs

+10-15
Original file line numberDiff line numberDiff line change
@@ -239,8 +239,7 @@ pub fn resolve_interior<'a, 'tcx>(
239239
// typeck had previously found constraints that would cause them to be related.
240240

241241
let mut counter = 0;
242-
let mut mk_bound_region = |span| {
243-
let kind = ty::BrAnon(span);
242+
let mut mk_bound_region = |kind| {
244243
let var = ty::BoundVar::from_u32(counter);
245244
counter += 1;
246245
ty::BoundRegion { var, kind }
@@ -252,24 +251,23 @@ pub fn resolve_interior<'a, 'tcx>(
252251
let origin = fcx.region_var_origin(vid);
253252
match origin {
254253
RegionVariableOrigin::EarlyBoundRegion(span, _) => {
255-
mk_bound_region(Some(span))
254+
mk_bound_region(ty::BrAnon(Some(span)))
256255
}
257-
_ => mk_bound_region(None),
256+
_ => mk_bound_region(ty::BrAnon(None)),
258257
}
259258
}
260-
// FIXME: these should use `BrNamed`
261259
ty::ReEarlyBound(region) => {
262-
mk_bound_region(Some(fcx.tcx.def_span(region.def_id)))
260+
mk_bound_region(ty::BrNamed(region.def_id, region.name))
263261
}
264262
ty::ReLateBound(_, ty::BoundRegion { kind, .. })
265263
| ty::ReFree(ty::FreeRegion { bound_region: kind, .. }) => match kind {
266-
ty::BoundRegionKind::BrAnon(span) => mk_bound_region(span),
267-
ty::BoundRegionKind::BrNamed(def_id, _) => {
268-
mk_bound_region(Some(fcx.tcx.def_span(def_id)))
264+
ty::BoundRegionKind::BrAnon(span) => mk_bound_region(ty::BrAnon(span)),
265+
ty::BoundRegionKind::BrNamed(def_id, sym) => {
266+
mk_bound_region(ty::BrNamed(def_id, sym))
269267
}
270-
ty::BoundRegionKind::BrEnv => mk_bound_region(None),
268+
ty::BoundRegionKind::BrEnv => mk_bound_region(ty::BrAnon(None)),
271269
},
272-
_ => mk_bound_region(None),
270+
_ => mk_bound_region(ty::BrAnon(None)),
273271
};
274272
let r = fcx.tcx.mk_re_late_bound(current_depth, br);
275273
r
@@ -293,10 +291,7 @@ pub fn resolve_interior<'a, 'tcx>(
293291
type_causes,
294292
FnMutDelegate {
295293
regions: &mut |br| {
296-
let kind = match br.kind {
297-
ty::BrAnon(span) => ty::BrAnon(span),
298-
_ => br.kind,
299-
};
294+
let kind = br.kind;
300295
let var = ty::BoundVar::from_usize(bound_vars.len());
301296
bound_vars.push(ty::BoundVariableKind::Region(kind));
302297
counter += 1;

compiler/rustc_middle/src/hir/mod.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ pub mod nested_filter;
77
pub mod place;
88

99
use crate::ty::query::Providers;
10-
use crate::ty::{ImplSubject, TyCtxt};
10+
use crate::ty::{EarlyBinder, ImplSubject, TyCtxt};
1111
use rustc_data_structures::stable_hasher::{HashStable, StableHasher};
1212
use rustc_data_structures::sync::{par_for_each_in, Send, Sync};
1313
use rustc_hir::def_id::{DefId, LocalDefId};
@@ -104,11 +104,11 @@ impl<'tcx> TyCtxt<'tcx> {
104104
self.parent_module_from_def_id(id.owner.def_id)
105105
}
106106

107-
pub fn impl_subject(self, def_id: DefId) -> ImplSubject<'tcx> {
108-
self.impl_trait_ref(def_id)
109-
.map(|t| t.subst_identity())
110-
.map(ImplSubject::Trait)
111-
.unwrap_or_else(|| ImplSubject::Inherent(self.type_of(def_id).subst_identity()))
107+
pub fn impl_subject(self, def_id: DefId) -> EarlyBinder<ImplSubject<'tcx>> {
108+
match self.impl_trait_ref(def_id) {
109+
Some(t) => t.map_bound(ImplSubject::Trait),
110+
None => self.type_of(def_id).map_bound(ImplSubject::Inherent),
111+
}
112112
}
113113
}
114114

compiler/rustc_middle/src/ty/util.rs

-4
Original file line numberDiff line numberDiff line change
@@ -708,10 +708,6 @@ impl<'tcx> TyCtxt<'tcx> {
708708
ty::EarlyBinder(self.explicit_item_bounds(def_id))
709709
}
710710

711-
pub fn bound_impl_subject(self, def_id: DefId) -> ty::EarlyBinder<ty::ImplSubject<'tcx>> {
712-
ty::EarlyBinder(self.impl_subject(def_id))
713-
}
714-
715711
/// Returns names of captured upvars for closures and generators.
716712
///
717713
/// Here are some examples:

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

+8
Original file line numberDiff line numberDiff line change
@@ -348,6 +348,14 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
348348
) {
349349
let lang_items = self.tcx().lang_items();
350350
let trait_def_id = goal.predicate.trait_def_id(self.tcx());
351+
352+
// N.B. When assembling built-in candidates for lang items that are also
353+
// `auto` traits, then the auto trait candidate that is assembled in
354+
// `consider_auto_trait_candidate` MUST be disqualified to remain sound.
355+
//
356+
// Instead of adding the logic here, it's a better idea to add it in
357+
// `EvalCtxt::disqualify_auto_trait_candidate_due_to_possible_impl` in
358+
// `solve::trait_goals` instead.
351359
let result = if self.tcx().trait_is_auto(trait_def_id) {
352360
G::consider_auto_trait_candidate(self, goal)
353361
} else if self.tcx().trait_is_alias(trait_def_id) {

compiler/rustc_trait_selection/src/solve/canonicalize.rs

+11-14
Original file line numberDiff line numberDiff line change
@@ -125,8 +125,9 @@ impl<'a, 'tcx> Canonicalizer<'a, 'tcx> {
125125
// - var_infos: [E0, U1, E1, U1, E1, E6, U6], curr_compressed_uv: 1, next_orig_uv: 6
126126
// - var_infos: [E0, U1, E1, U1, E1, E2, U2], curr_compressed_uv: 2, next_orig_uv: -
127127
//
128-
// This algorithm runs in `O(n²)` where `n` is the number of different universe
129-
// indices in the input. This should be fine as `n` is expected to be small.
128+
// This algorithm runs in `O(nm)` where `n` is the number of different universe
129+
// indices in the input and `m` is the number of canonical variables.
130+
// This should be fine as both `n` and `m` are expected to be small.
130131
let mut curr_compressed_uv = ty::UniverseIndex::ROOT;
131132
let mut existential_in_new_uv = false;
132133
let mut next_orig_uv = Some(ty::UniverseIndex::ROOT);
@@ -245,18 +246,14 @@ impl<'tcx> TypeFolder<TyCtxt<'tcx>> for Canonicalizer<'_, 'tcx> {
245246
ty::ReError(_) => return r,
246247
};
247248

248-
let existing_bound_var = match self.canonicalize_mode {
249-
CanonicalizeMode::Input => None,
250-
CanonicalizeMode::Response { .. } => {
251-
self.variables.iter().position(|&v| v == r.into()).map(ty::BoundVar::from)
252-
}
253-
};
254-
let var = existing_bound_var.unwrap_or_else(|| {
255-
let var = ty::BoundVar::from(self.variables.len());
256-
self.variables.push(r.into());
257-
self.primitive_var_infos.push(CanonicalVarInfo { kind });
258-
var
259-
});
249+
let var = ty::BoundVar::from(
250+
self.variables.iter().position(|&v| v == r.into()).unwrap_or_else(|| {
251+
let var = self.variables.len();
252+
self.variables.push(r.into());
253+
self.primitive_var_infos.push(CanonicalVarInfo { kind });
254+
var
255+
}),
256+
);
260257
let br = ty::BoundRegion { var, kind: BrAnon(None) };
261258
self.interner().mk_re_late_bound(self.binder_index, br)
262259
}

compiler/rustc_trait_selection/src/solve/trait_goals.rs

+94-61
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
use super::assembly::{self, structural_traits};
44
use super::{EvalCtxt, SolverMode};
55
use rustc_hir::def_id::DefId;
6-
use rustc_hir::LangItem;
6+
use rustc_hir::{LangItem, Movability};
77
use rustc_infer::traits::query::NoSolution;
88
use rustc_infer::traits::util::supertraits;
99
use rustc_middle::traits::solve::{CanonicalResponse, Certainty, Goal, QueryResult};
@@ -147,66 +147,8 @@ impl<'tcx> assembly::GoalKind<'tcx> for TraitPredicate<'tcx> {
147147
ecx: &mut EvalCtxt<'_, 'tcx>,
148148
goal: Goal<'tcx, Self>,
149149
) -> QueryResult<'tcx> {
150-
let self_ty = goal.predicate.self_ty();
151-
match *self_ty.kind() {
152-
// Stall int and float vars until they are resolved to a concrete
153-
// numerical type. That's because the check for impls below treats
154-
// int vars as matching any impl. Even if we filtered such impls,
155-
// we probably don't want to treat an `impl !AutoTrait for i32` as
156-
// disqualifying the built-in auto impl for `i64: AutoTrait` either.
157-
ty::Infer(ty::IntVar(_) | ty::FloatVar(_)) => {
158-
return ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS);
159-
}
160-
161-
// These types cannot be structurally decomposed into constitutent
162-
// types, and therefore have no builtin impl.
163-
ty::Dynamic(..)
164-
| ty::Param(..)
165-
| ty::Foreign(..)
166-
| ty::Alias(ty::Projection, ..)
167-
| ty::Placeholder(..) => return Err(NoSolution),
168-
169-
ty::Infer(_) | ty::Bound(_, _) => bug!("unexpected type `{self_ty}`"),
170-
171-
// For rigid types, we only register a builtin auto implementation
172-
// if there is no implementation that could ever apply to the self
173-
// type.
174-
//
175-
// This differs from the current stable behavior and fixes #84857.
176-
// Due to breakage found via crater, we currently instead lint
177-
// patterns which can be used to exploit this unsoundness on stable,
178-
// see #93367 for more details.
179-
ty::Bool
180-
| ty::Char
181-
| ty::Int(_)
182-
| ty::Uint(_)
183-
| ty::Float(_)
184-
| ty::Str
185-
| ty::Array(_, _)
186-
| ty::Slice(_)
187-
| ty::RawPtr(_)
188-
| ty::Ref(_, _, _)
189-
| ty::FnDef(_, _)
190-
| ty::FnPtr(_)
191-
| ty::Closure(_, _)
192-
| ty::Generator(_, _, _)
193-
| ty::GeneratorWitness(_)
194-
| ty::GeneratorWitnessMIR(_, _)
195-
| ty::Never
196-
| ty::Tuple(_)
197-
| ty::Error(_)
198-
| ty::Adt(_, _)
199-
| ty::Alias(ty::Opaque, _) => {
200-
if let Some(def_id) = ecx.tcx().find_map_relevant_impl(
201-
goal.predicate.def_id(),
202-
goal.predicate.self_ty(),
203-
TreatProjections::NextSolverLookup,
204-
Some,
205-
) {
206-
debug!(?def_id, ?goal, "disqualified auto-trait implementation");
207-
return Err(NoSolution);
208-
}
209-
}
150+
if let Some(result) = ecx.disqualify_auto_trait_candidate_due_to_possible_impl(goal) {
151+
return result;
210152
}
211153

212154
ecx.probe_and_evaluate_goal_for_constituent_tys(
@@ -630,6 +572,97 @@ impl<'tcx> assembly::GoalKind<'tcx> for TraitPredicate<'tcx> {
630572
}
631573

632574
impl<'tcx> EvalCtxt<'_, 'tcx> {
575+
// Return `Some` if there is an impl (built-in or user provided) that may
576+
// hold for the self type of the goal, which for coherence and soundness
577+
// purposes must disqualify the built-in auto impl assembled by considering
578+
// the type's constituent types.
579+
fn disqualify_auto_trait_candidate_due_to_possible_impl(
580+
&mut self,
581+
goal: Goal<'tcx, TraitPredicate<'tcx>>,
582+
) -> Option<QueryResult<'tcx>> {
583+
let self_ty = goal.predicate.self_ty();
584+
match *self_ty.kind() {
585+
// Stall int and float vars until they are resolved to a concrete
586+
// numerical type. That's because the check for impls below treats
587+
// int vars as matching any impl. Even if we filtered such impls,
588+
// we probably don't want to treat an `impl !AutoTrait for i32` as
589+
// disqualifying the built-in auto impl for `i64: AutoTrait` either.
590+
ty::Infer(ty::IntVar(_) | ty::FloatVar(_)) => {
591+
Some(self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS))
592+
}
593+
594+
// These types cannot be structurally decomposed into constitutent
595+
// types, and therefore have no built-in auto impl.
596+
ty::Dynamic(..)
597+
| ty::Param(..)
598+
| ty::Foreign(..)
599+
| ty::Alias(ty::Projection, ..)
600+
| ty::Placeholder(..) => Some(Err(NoSolution)),
601+
602+
ty::Infer(_) | ty::Bound(_, _) => bug!("unexpected type `{self_ty}`"),
603+
604+
// Generators have one special built-in candidate, `Unpin`, which
605+
// takes precedence over the structural auto trait candidate being
606+
// assembled.
607+
ty::Generator(_, _, movability)
608+
if Some(goal.predicate.def_id()) == self.tcx().lang_items().unpin_trait() =>
609+
{
610+
match movability {
611+
Movability::Static => Some(Err(NoSolution)),
612+
Movability::Movable => {
613+
Some(self.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
614+
}
615+
}
616+
}
617+
618+
// For rigid types, any possible implementation that could apply to
619+
// the type (even if after unification and processing nested goals
620+
// it does not hold) will disqualify the built-in auto impl.
621+
//
622+
// This differs from the current stable behavior and fixes #84857.
623+
// Due to breakage found via crater, we currently instead lint
624+
// patterns which can be used to exploit this unsoundness on stable,
625+
// see #93367 for more details.
626+
ty::Bool
627+
| ty::Char
628+
| ty::Int(_)
629+
| ty::Uint(_)
630+
| ty::Float(_)
631+
| ty::Str
632+
| ty::Array(_, _)
633+
| ty::Slice(_)
634+
| ty::RawPtr(_)
635+
| ty::Ref(_, _, _)
636+
| ty::FnDef(_, _)
637+
| ty::FnPtr(_)
638+
| ty::Closure(_, _)
639+
| ty::Generator(_, _, _)
640+
| ty::GeneratorWitness(_)
641+
| ty::GeneratorWitnessMIR(_, _)
642+
| ty::Never
643+
| ty::Tuple(_)
644+
| ty::Adt(_, _)
645+
// FIXME: Handling opaques here is kinda sus. Especially because we
646+
// simplify them to PlaceholderSimplifiedType.
647+
| ty::Alias(ty::Opaque, _) => {
648+
if let Some(def_id) = self.tcx().find_map_relevant_impl(
649+
goal.predicate.def_id(),
650+
goal.predicate.self_ty(),
651+
TreatProjections::NextSolverLookup,
652+
Some,
653+
) {
654+
debug!(?def_id, ?goal, "disqualified auto-trait implementation");
655+
// No need to actually consider the candidate here,
656+
// since we do that in `consider_impl_candidate`.
657+
return Some(Err(NoSolution));
658+
} else {
659+
None
660+
}
661+
}
662+
ty::Error(_) => None,
663+
}
664+
}
665+
633666
/// Convenience function for traits that are structural, i.e. that only
634667
/// have nested subgoals that only change the self type. Unlike other
635668
/// evaluate-like helpers, this does a probe, so it doesn't need to be

compiler/rustc_trait_selection/src/traits/coherence.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -306,7 +306,7 @@ fn negative_impl(tcx: TyCtxt<'_>, impl1_def_id: DefId, impl2_def_id: DefId) -> b
306306
&infcx,
307307
ObligationCause::dummy(),
308308
impl_env,
309-
tcx.impl_subject(impl1_def_id),
309+
tcx.impl_subject(impl1_def_id).subst_identity(),
310310
) {
311311
Ok(s) => s,
312312
Err(err) => {

0 commit comments

Comments
 (0)