]> git.lizzy.rs Git - rust.git/blob - compiler/rustc_trait_selection/src/traits/specialize/mod.rs
Stabilize File::options()
[rust.git] / compiler / rustc_trait_selection / src / traits / specialize / mod.rs
1 //! Logic and data structures related to impl specialization, explained in
2 //! greater detail below.
3 //!
4 //! At the moment, this implementation support only the simple "chain" rule:
5 //! If any two impls overlap, one must be a strict subset of the other.
6 //!
7 //! See the [rustc dev guide] for a bit more detail on how specialization
8 //! fits together with the rest of the trait machinery.
9 //!
10 //! [rustc dev guide]: https://rustc-dev-guide.rust-lang.org/traits/specialization.html
11
12 pub mod specialization_graph;
13 use specialization_graph::GraphExt;
14
15 use crate::infer::{InferCtxt, InferOk, TyCtxtInferExt};
16 use crate::traits::select::IntercrateAmbiguityCause;
17 use crate::traits::{self, coherence, FutureCompatOverlapErrorKind, ObligationCause, TraitEngine};
18 use rustc_data_structures::fx::FxHashSet;
19 use rustc_errors::struct_span_err;
20 use rustc_hir::def_id::{DefId, LocalDefId};
21 use rustc_middle::lint::LintDiagnosticBuilder;
22 use rustc_middle::ty::subst::{InternalSubsts, Subst, SubstsRef};
23 use rustc_middle::ty::{self, TyCtxt};
24 use rustc_session::lint::builtin::COHERENCE_LEAK_CHECK;
25 use rustc_session::lint::builtin::ORDER_DEPENDENT_TRAIT_OBJECTS;
26 use rustc_span::DUMMY_SP;
27
28 use super::util::impl_trait_ref_and_oblig;
29 use super::{FulfillmentContext, SelectionContext};
30
31 /// Information pertinent to an overlapping impl error.
32 #[derive(Debug)]
33 pub struct OverlapError {
34     pub with_impl: DefId,
35     pub trait_desc: String,
36     pub self_desc: Option<String>,
37     pub intercrate_ambiguity_causes: Vec<IntercrateAmbiguityCause>,
38     pub involves_placeholder: bool,
39 }
40
41 /// Given a subst for the requested impl, translate it to a subst
42 /// appropriate for the actual item definition (whether it be in that impl,
43 /// a parent impl, or the trait).
44 ///
45 /// When we have selected one impl, but are actually using item definitions from
46 /// a parent impl providing a default, we need a way to translate between the
47 /// type parameters of the two impls. Here the `source_impl` is the one we've
48 /// selected, and `source_substs` is a substitution of its generics.
49 /// And `target_node` is the impl/trait we're actually going to get the
50 /// definition from. The resulting substitution will map from `target_node`'s
51 /// generics to `source_impl`'s generics as instantiated by `source_subst`.
52 ///
53 /// For example, consider the following scenario:
54 ///
55 /// ```rust
56 /// trait Foo { ... }
57 /// impl<T, U> Foo for (T, U) { ... }  // target impl
58 /// impl<V> Foo for (V, V) { ... }     // source impl
59 /// ```
60 ///
61 /// Suppose we have selected "source impl" with `V` instantiated with `u32`.
62 /// This function will produce a substitution with `T` and `U` both mapping to `u32`.
63 ///
64 /// where-clauses add some trickiness here, because they can be used to "define"
65 /// an argument indirectly:
66 ///
67 /// ```rust
68 /// impl<'a, I, T: 'a> Iterator for Cloned<I>
69 ///    where I: Iterator<Item = &'a T>, T: Clone
70 /// ```
71 ///
72 /// In a case like this, the substitution for `T` is determined indirectly,
73 /// through associated type projection. We deal with such cases by using
74 /// *fulfillment* to relate the two impls, requiring that all projections are
75 /// resolved.
76 pub fn translate_substs<'a, 'tcx>(
77     infcx: &InferCtxt<'a, 'tcx>,
78     param_env: ty::ParamEnv<'tcx>,
79     source_impl: DefId,
80     source_substs: SubstsRef<'tcx>,
81     target_node: specialization_graph::Node,
82 ) -> SubstsRef<'tcx> {
83     debug!(
84         "translate_substs({:?}, {:?}, {:?}, {:?})",
85         param_env, source_impl, source_substs, target_node
86     );
87     let source_trait_ref =
88         infcx.tcx.impl_trait_ref(source_impl).unwrap().subst(infcx.tcx, &source_substs);
89
90     // translate the Self and Param parts of the substitution, since those
91     // vary across impls
92     let target_substs = match target_node {
93         specialization_graph::Node::Impl(target_impl) => {
94             // no need to translate if we're targeting the impl we started with
95             if source_impl == target_impl {
96                 return source_substs;
97             }
98
99             fulfill_implication(infcx, param_env, source_trait_ref, target_impl).unwrap_or_else(
100                 |_| {
101                     bug!(
102                         "When translating substitutions for specialization, the expected \
103                          specialization failed to hold"
104                     )
105                 },
106             )
107         }
108         specialization_graph::Node::Trait(..) => source_trait_ref.substs,
109     };
110
111     // directly inherent the method generics, since those do not vary across impls
112     source_substs.rebase_onto(infcx.tcx, source_impl, target_substs)
113 }
114
115 /// Is `impl1` a specialization of `impl2`?
116 ///
117 /// Specialization is determined by the sets of types to which the impls apply;
118 /// `impl1` specializes `impl2` if it applies to a subset of the types `impl2` applies
119 /// to.
120 pub(super) fn specializes(tcx: TyCtxt<'_>, (impl1_def_id, impl2_def_id): (DefId, DefId)) -> bool {
121     debug!("specializes({:?}, {:?})", impl1_def_id, impl2_def_id);
122
123     // The feature gate should prevent introducing new specializations, but not
124     // taking advantage of upstream ones.
125     let features = tcx.features();
126     let specialization_enabled = features.specialization || features.min_specialization;
127     if !specialization_enabled && (impl1_def_id.is_local() || impl2_def_id.is_local()) {
128         return false;
129     }
130
131     // We determine whether there's a subset relationship by:
132     //
133     // - replacing bound vars with placeholders in impl1,
134     // - assuming the where clauses for impl1,
135     // - instantiating impl2 with fresh inference variables,
136     // - unifying,
137     // - attempting to prove the where clauses for impl2
138     //
139     // The last three steps are encapsulated in `fulfill_implication`.
140     //
141     // See RFC 1210 for more details and justification.
142
143     // Currently we do not allow e.g., a negative impl to specialize a positive one
144     if tcx.impl_polarity(impl1_def_id) != tcx.impl_polarity(impl2_def_id) {
145         return false;
146     }
147
148     // create a parameter environment corresponding to a (placeholder) instantiation of impl1
149     let penv = tcx.param_env(impl1_def_id);
150     let impl1_trait_ref = tcx.impl_trait_ref(impl1_def_id).unwrap();
151
152     // Create an infcx, taking the predicates of impl1 as assumptions:
153     tcx.infer_ctxt().enter(|infcx| {
154         // Normalize the trait reference. The WF rules ought to ensure
155         // that this always succeeds.
156         let impl1_trait_ref = match traits::fully_normalize(
157             &infcx,
158             FulfillmentContext::new(),
159             ObligationCause::dummy(),
160             penv,
161             impl1_trait_ref,
162         ) {
163             Ok(impl1_trait_ref) => impl1_trait_ref,
164             Err(err) => {
165                 bug!("failed to fully normalize {:?}: {:?}", impl1_trait_ref, err);
166             }
167         };
168
169         // Attempt to prove that impl2 applies, given all of the above.
170         fulfill_implication(&infcx, penv, impl1_trait_ref, impl2_def_id).is_ok()
171     })
172 }
173
174 /// Attempt to fulfill all obligations of `target_impl` after unification with
175 /// `source_trait_ref`. If successful, returns a substitution for *all* the
176 /// generics of `target_impl`, including both those needed to unify with
177 /// `source_trait_ref` and those whose identity is determined via a where
178 /// clause in the impl.
179 fn fulfill_implication<'a, 'tcx>(
180     infcx: &InferCtxt<'a, 'tcx>,
181     param_env: ty::ParamEnv<'tcx>,
182     source_trait_ref: ty::TraitRef<'tcx>,
183     target_impl: DefId,
184 ) -> Result<SubstsRef<'tcx>, ()> {
185     debug!(
186         "fulfill_implication({:?}, trait_ref={:?} |- {:?} applies)",
187         param_env, source_trait_ref, target_impl
188     );
189
190     let selcx = &mut SelectionContext::new(&infcx);
191     let target_substs = infcx.fresh_substs_for_item(DUMMY_SP, target_impl);
192     let (target_trait_ref, obligations) =
193         impl_trait_ref_and_oblig(selcx, param_env, target_impl, target_substs);
194
195     // do the impls unify? If not, no specialization.
196     let more_obligations =
197         match infcx.at(&ObligationCause::dummy(), param_env).eq(source_trait_ref, target_trait_ref)
198         {
199             Ok(InferOk { obligations, .. }) => obligations,
200             Err(_) => {
201                 debug!(
202                     "fulfill_implication: {:?} does not unify with {:?}",
203                     source_trait_ref, target_trait_ref
204                 );
205                 return Err(());
206             }
207         };
208
209     // attempt to prove all of the predicates for impl2 given those for impl1
210     // (which are packed up in penv)
211
212     infcx.save_and_restore_in_snapshot_flag(|infcx| {
213         // If we came from `translate_substs`, we already know that the
214         // predicates for our impl hold (after all, we know that a more
215         // specialized impl holds, so our impl must hold too), and
216         // we only want to process the projections to determine the
217         // the types in our substs using RFC 447, so we can safely
218         // ignore region obligations, which allows us to avoid threading
219         // a node-id to assign them with.
220         //
221         // If we came from specialization graph construction, then
222         // we already make a mockery out of the region system, so
223         // why not ignore them a bit earlier?
224         let mut fulfill_cx = FulfillmentContext::new_ignoring_regions();
225         for oblig in obligations.chain(more_obligations) {
226             fulfill_cx.register_predicate_obligation(&infcx, oblig);
227         }
228         match fulfill_cx.select_all_or_error(infcx).as_slice() {
229             [] => {
230                 debug!(
231                     "fulfill_implication: an impl for {:?} specializes {:?}",
232                     source_trait_ref, target_trait_ref
233                 );
234
235                 // Now resolve the *substitution* we built for the target earlier, replacing
236                 // the inference variables inside with whatever we got from fulfillment.
237                 Ok(infcx.resolve_vars_if_possible(target_substs))
238             }
239             errors => {
240                 // no dice!
241                 debug!(
242                     "fulfill_implication: for impls on {:?} and {:?}, \
243                      could not fulfill: {:?} given {:?}",
244                     source_trait_ref,
245                     target_trait_ref,
246                     errors,
247                     param_env.caller_bounds()
248                 );
249                 Err(())
250             }
251         }
252     })
253 }
254
255 // Query provider for `specialization_graph_of`.
256 pub(super) fn specialization_graph_provider(
257     tcx: TyCtxt<'_>,
258     trait_id: DefId,
259 ) -> specialization_graph::Graph {
260     let mut sg = specialization_graph::Graph::new();
261
262     let mut trait_impls: Vec<_> = tcx.all_impls(trait_id).collect();
263
264     // The coherence checking implementation seems to rely on impls being
265     // iterated over (roughly) in definition order, so we are sorting by
266     // negated `CrateNum` (so remote definitions are visited first) and then
267     // by a flattened version of the `DefIndex`.
268     trait_impls
269         .sort_unstable_by_key(|def_id| (-(def_id.krate.as_u32() as i64), def_id.index.index()));
270
271     for impl_def_id in trait_impls {
272         if let Some(impl_def_id) = impl_def_id.as_local() {
273             // This is where impl overlap checking happens:
274             let insert_result = sg.insert(tcx, impl_def_id.to_def_id());
275             // Report error if there was one.
276             let (overlap, used_to_be_allowed) = match insert_result {
277                 Err(overlap) => (Some(overlap), None),
278                 Ok(Some(overlap)) => (Some(overlap.error), Some(overlap.kind)),
279                 Ok(None) => (None, None),
280             };
281
282             if let Some(overlap) = overlap {
283                 report_overlap_conflict(tcx, overlap, impl_def_id, used_to_be_allowed, &mut sg);
284             }
285         } else {
286             let parent = tcx.impl_parent(impl_def_id).unwrap_or(trait_id);
287             sg.record_impl_from_cstore(tcx, parent, impl_def_id)
288         }
289     }
290
291     sg
292 }
293
294 fn report_overlap_conflict(
295     tcx: TyCtxt<'_>,
296     overlap: OverlapError,
297     impl_def_id: LocalDefId,
298     used_to_be_allowed: Option<FutureCompatOverlapErrorKind>,
299     sg: &mut specialization_graph::Graph,
300 ) {
301     let impl_polarity = tcx.impl_polarity(impl_def_id.to_def_id());
302     let other_polarity = tcx.impl_polarity(overlap.with_impl);
303     match (impl_polarity, other_polarity) {
304         (ty::ImplPolarity::Negative, ty::ImplPolarity::Positive) => {
305             report_negative_positive_conflict(
306                 tcx,
307                 &overlap,
308                 impl_def_id,
309                 impl_def_id.to_def_id(),
310                 overlap.with_impl,
311                 sg,
312             );
313         }
314
315         (ty::ImplPolarity::Positive, ty::ImplPolarity::Negative) => {
316             report_negative_positive_conflict(
317                 tcx,
318                 &overlap,
319                 impl_def_id,
320                 overlap.with_impl,
321                 impl_def_id.to_def_id(),
322                 sg,
323             );
324         }
325
326         _ => {
327             report_conflicting_impls(tcx, overlap, impl_def_id, used_to_be_allowed, sg);
328         }
329     }
330 }
331
332 fn report_negative_positive_conflict(
333     tcx: TyCtxt<'_>,
334     overlap: &OverlapError,
335     local_impl_def_id: LocalDefId,
336     negative_impl_def_id: DefId,
337     positive_impl_def_id: DefId,
338     sg: &mut specialization_graph::Graph,
339 ) {
340     let impl_span = tcx
341         .sess
342         .source_map()
343         .guess_head_span(tcx.span_of_impl(local_impl_def_id.to_def_id()).unwrap());
344
345     let mut err = struct_span_err!(
346         tcx.sess,
347         impl_span,
348         E0751,
349         "found both positive and negative implementation of trait `{}`{}:",
350         overlap.trait_desc,
351         overlap.self_desc.clone().map_or_else(String::new, |ty| format!(" for type `{}`", ty))
352     );
353
354     match tcx.span_of_impl(negative_impl_def_id) {
355         Ok(span) => {
356             err.span_label(
357                 tcx.sess.source_map().guess_head_span(span),
358                 "negative implementation here".to_string(),
359             );
360         }
361         Err(cname) => {
362             err.note(&format!("negative implementation in crate `{}`", cname));
363         }
364     }
365
366     match tcx.span_of_impl(positive_impl_def_id) {
367         Ok(span) => {
368             err.span_label(
369                 tcx.sess.source_map().guess_head_span(span),
370                 "positive implementation here".to_string(),
371             );
372         }
373         Err(cname) => {
374             err.note(&format!("positive implementation in crate `{}`", cname));
375         }
376     }
377
378     sg.has_errored = true;
379     err.emit();
380 }
381
382 fn report_conflicting_impls(
383     tcx: TyCtxt<'_>,
384     overlap: OverlapError,
385     impl_def_id: LocalDefId,
386     used_to_be_allowed: Option<FutureCompatOverlapErrorKind>,
387     sg: &mut specialization_graph::Graph,
388 ) {
389     let impl_span =
390         tcx.sess.source_map().guess_head_span(tcx.span_of_impl(impl_def_id.to_def_id()).unwrap());
391
392     // Work to be done after we've built the DiagnosticBuilder. We have to define it
393     // now because the struct_lint methods don't return back the DiagnosticBuilder
394     // that's passed in.
395     let decorate = |err: LintDiagnosticBuilder<'_>| {
396         let msg = format!(
397             "conflicting implementations of trait `{}`{}{}",
398             overlap.trait_desc,
399             overlap
400                 .self_desc
401                 .clone()
402                 .map_or_else(String::new, |ty| { format!(" for type `{}`", ty) }),
403             match used_to_be_allowed {
404                 Some(FutureCompatOverlapErrorKind::Issue33140) => ": (E0119)",
405                 _ => "",
406             }
407         );
408         let mut err = err.build(&msg);
409         match tcx.span_of_impl(overlap.with_impl) {
410             Ok(span) => {
411                 err.span_label(
412                     tcx.sess.source_map().guess_head_span(span),
413                     "first implementation here".to_string(),
414                 );
415
416                 err.span_label(
417                     impl_span,
418                     format!(
419                         "conflicting implementation{}",
420                         overlap.self_desc.map_or_else(String::new, |ty| format!(" for `{}`", ty))
421                     ),
422                 );
423             }
424             Err(cname) => {
425                 let msg = match to_pretty_impl_header(tcx, overlap.with_impl) {
426                     Some(s) => format!("conflicting implementation in crate `{}`:\n- {}", cname, s),
427                     None => format!("conflicting implementation in crate `{}`", cname),
428                 };
429                 err.note(&msg);
430             }
431         }
432
433         for cause in &overlap.intercrate_ambiguity_causes {
434             cause.add_intercrate_ambiguity_hint(&mut err);
435         }
436
437         if overlap.involves_placeholder {
438             coherence::add_placeholder_note(&mut err);
439         }
440         err.emit()
441     };
442
443     match used_to_be_allowed {
444         None => {
445             sg.has_errored = true;
446             let err = struct_span_err!(tcx.sess, impl_span, E0119, "");
447             decorate(LintDiagnosticBuilder::new(err));
448         }
449         Some(kind) => {
450             let lint = match kind {
451                 FutureCompatOverlapErrorKind::Issue33140 => ORDER_DEPENDENT_TRAIT_OBJECTS,
452                 FutureCompatOverlapErrorKind::LeakCheck => COHERENCE_LEAK_CHECK,
453             };
454             tcx.struct_span_lint_hir(
455                 lint,
456                 tcx.hir().local_def_id_to_hir_id(impl_def_id),
457                 impl_span,
458                 decorate,
459             )
460         }
461     };
462 }
463
464 /// Recovers the "impl X for Y" signature from `impl_def_id` and returns it as a
465 /// string.
466 crate fn to_pretty_impl_header(tcx: TyCtxt<'_>, impl_def_id: DefId) -> Option<String> {
467     use std::fmt::Write;
468
469     let trait_ref = tcx.impl_trait_ref(impl_def_id)?;
470     let mut w = "impl".to_owned();
471
472     let substs = InternalSubsts::identity_for_item(tcx, impl_def_id);
473
474     // FIXME: Currently only handles ?Sized.
475     //        Needs to support ?Move and ?DynSized when they are implemented.
476     let mut types_without_default_bounds = FxHashSet::default();
477     let sized_trait = tcx.lang_items().sized_trait();
478
479     if !substs.is_noop() {
480         types_without_default_bounds.extend(substs.types());
481         w.push('<');
482         w.push_str(
483             &substs
484                 .iter()
485                 .map(|k| k.to_string())
486                 .filter(|k| k != "'_")
487                 .collect::<Vec<_>>()
488                 .join(", "),
489         );
490         w.push('>');
491     }
492
493     write!(w, " {} for {}", trait_ref.print_only_trait_path(), tcx.type_of(impl_def_id)).unwrap();
494
495     // The predicates will contain default bounds like `T: Sized`. We need to
496     // remove these bounds, and add `T: ?Sized` to any untouched type parameters.
497     let predicates = tcx.predicates_of(impl_def_id).predicates;
498     let mut pretty_predicates =
499         Vec::with_capacity(predicates.len() + types_without_default_bounds.len());
500
501     for (p, _) in predicates {
502         if let Some(poly_trait_ref) = p.to_opt_poly_trait_ref() {
503             if Some(poly_trait_ref.value.def_id()) == sized_trait {
504                 types_without_default_bounds.remove(poly_trait_ref.value.self_ty().skip_binder());
505                 continue;
506             }
507         }
508         pretty_predicates.push(p.to_string());
509     }
510
511     pretty_predicates
512         .extend(types_without_default_bounds.iter().map(|ty| format!("{}: ?Sized", ty)));
513
514     if !pretty_predicates.is_empty() {
515         write!(w, "\n  where {}", pretty_predicates.join(", ")).unwrap();
516     }
517
518     w.push(';');
519     Some(w)
520 }