Skip to content

Commit 3028dc4

Browse files
committed
Only check outlives goals on impl compared to trait
1 parent a6f8aa5 commit 3028dc4

File tree

3 files changed

+77
-12
lines changed

3 files changed

+77
-12
lines changed

compiler/rustc_hir_analysis/src/check/compare_impl_item.rs

+39-12
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use super::potentially_plural_count;
22
use crate::errors::LifetimesOrBoundsMismatchOnTrait;
33
use hir::def_id::{DefId, LocalDefId};
4-
use rustc_data_structures::fx::{FxHashMap, FxIndexSet};
4+
use rustc_data_structures::fx::{FxHashMap, FxHashSet, FxIndexSet};
55
use rustc_errors::{
66
pluralize, struct_span_err, Applicability, DiagnosticId, ErrorGuaranteed, MultiSpan,
77
};
@@ -265,7 +265,6 @@ fn compare_method_predicate_entailment<'tcx>(
265265
infer::HigherRankedType,
266266
tcx.fn_sig(impl_m.def_id).instantiate_identity(),
267267
);
268-
let unnormalized_impl_fty = Ty::new_fn_ptr(tcx, ty::Binder::dummy(unnormalized_impl_sig));
269268

270269
let norm_cause = ObligationCause::misc(impl_m_span, impl_m_def_id);
271270
let impl_sig = ocx.normalize(&norm_cause, param_env, unnormalized_impl_sig);
@@ -309,16 +308,44 @@ fn compare_method_predicate_entailment<'tcx>(
309308
}
310309

311310
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+
}
322349
}
323350

324351
// Check that all obligations are satisfied by the implementation's
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// check-pass
2+
// See issue #109356. We don't want a false positive to the `implied_bounds_entailment` lint.
3+
4+
use std::borrow::Cow;
5+
6+
pub trait Trait {
7+
fn method(self) -> Option<Cow<'static, str>>
8+
where
9+
Self: Sized;
10+
}
11+
12+
impl<'a> Trait for Cow<'a, str> {
13+
// If we're not careful here, we'll check `WF(return-type)` using the trait
14+
// and impl where clauses, requiring that `Cow<'a, str>: Sized`. This is
15+
// obviously true, but if we pick the `Self: Sized` clause from the trait
16+
// over the "inherent impl", we will require `'a == 'static`, which triggers
17+
// the `implied_bounds_entailment` lint.
18+
fn method(self) -> Option<Cow<'static, str>> {
19+
None
20+
}
21+
}
22+
23+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// check-pass
2+
3+
pub trait Trait<'a, 'b> {
4+
fn method(self, _: &'static &'static ())
5+
where
6+
'a: 'b;
7+
}
8+
9+
impl<'a> Trait<'a, 'static> for () {
10+
// On first glance, this seems like we have the extra implied bound that
11+
// `'a: 'static`, but we know this from the trait method where clause.
12+
fn method(self, _: &'static &'a ()) {}
13+
}
14+
15+
fn main() {}

0 commit comments

Comments
 (0)