|
1 | 1 | use super::potentially_plural_count;
|
2 | 2 | use crate::errors::LifetimesOrBoundsMismatchOnTrait;
|
3 | 3 | use hir::def_id::{DefId, LocalDefId};
|
4 |
| -use rustc_data_structures::fx::{FxHashMap, FxIndexSet}; |
| 4 | +use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexSet}; |
5 | 5 | use rustc_errors::{
|
6 | 6 | pluralize, struct_span_err, Applicability, DiagnosticId, ErrorGuaranteed, MultiSpan,
|
7 | 7 | };
|
@@ -265,7 +265,6 @@ fn compare_method_predicate_entailment<'tcx>(
|
265 | 265 | infer::HigherRankedType,
|
266 | 266 | tcx.fn_sig(impl_m.def_id).instantiate_identity(),
|
267 | 267 | );
|
268 |
| - let unnormalized_impl_fty = Ty::new_fn_ptr(tcx, ty::Binder::dummy(unnormalized_impl_sig)); |
269 | 268 |
|
270 | 269 | let norm_cause = ObligationCause::misc(impl_m_span, impl_m_def_id);
|
271 | 270 | let impl_sig = ocx.normalize(&norm_cause, param_env, unnormalized_impl_sig);
|
@@ -309,16 +308,44 @@ fn compare_method_predicate_entailment<'tcx>(
|
309 | 308 | }
|
310 | 309 |
|
311 | 310 | if check_implied_wf == CheckImpliedWfMode::Check && !(impl_sig, trait_sig).references_error() {
|
312 |
| - // We need to check that the impl's args are well-formed given |
313 |
| - // the hybrid param-env (impl + trait method where-clauses). |
314 |
| - ocx.register_obligation(traits::Obligation::new( |
315 |
| - infcx.tcx, |
316 |
| - ObligationCause::dummy(), |
317 |
| - param_env, |
318 |
| - ty::Binder::dummy(ty::PredicateKind::Clause(ty::ClauseKind::WellFormed( |
319 |
| - unnormalized_impl_fty.into(), |
320 |
| - ))), |
321 |
| - )); |
| 311 | + // See #108544. Annoying, we can end up in cases where, because of winnowing, |
| 312 | + // we pick param env candidates over a more general impl, leading to more |
| 313 | + // stricter lifetime requirements than we would otherwise need. This can |
| 314 | + // trigger the lint. Instead, let's only consider type outlives and |
| 315 | + // region outlives obligations. |
| 316 | + // |
| 317 | + // FIXME(-Ztrait-solver=next): Try removing this hack again once |
| 318 | + // the new solver is stable. |
| 319 | + let mut wf_args: smallvec::SmallVec<[_; 4]> = |
| 320 | + unnormalized_impl_sig.inputs_and_output.iter().map(|ty| ty.into()).collect(); |
| 321 | + // Annoyingly, asking for the WF predicates of an array (with an unevaluated const (only?)) |
| 322 | + // will give back the well-formed predicate of the same array. |
| 323 | + let mut wf_args_seen: FxHashSet<_> = wf_args.iter().copied().collect(); |
| 324 | + while let Some(arg) = wf_args.pop() { |
| 325 | + let Some(obligations) = rustc_trait_selection::traits::wf::obligations( |
| 326 | + infcx, |
| 327 | + param_env, |
| 328 | + impl_m_def_id, |
| 329 | + 0, |
| 330 | + arg, |
| 331 | + impl_m_span, |
| 332 | + ) else { |
| 333 | + continue; |
| 334 | + }; |
| 335 | + for obligation in obligations { |
| 336 | + match obligation.predicate.kind().skip_binder() { |
| 337 | + ty::PredicateKind::Clause( |
| 338 | + ty::ClauseKind::RegionOutlives(..) | ty::ClauseKind::TypeOutlives(..), |
| 339 | + ) => ocx.register_obligation(obligation), |
| 340 | + ty::PredicateKind::Clause(ty::ClauseKind::WellFormed(arg)) => { |
| 341 | + if wf_args_seen.insert(arg) { |
| 342 | + wf_args.push(arg) |
| 343 | + } |
| 344 | + } |
| 345 | + _ => {} |
| 346 | + } |
| 347 | + } |
| 348 | + } |
322 | 349 | }
|
323 | 350 |
|
324 | 351 | // Check that all obligations are satisfied by the implementation's
|
|
0 commit comments