about summary refs log tree commit diff
diff options
context:
space:
mode:
authorMatthew Jasper <mjjasper1@gmail.com>2020-07-25 14:00:13 +0100
committerMatthew Jasper <mjjasper1@gmail.com>2020-10-06 11:19:32 +0100
commited324825341dbc763a8044f769eb0544afe87f08 (patch)
treec9bdcaff7454cec592c3e28a5cc6ea43e816bde3
parent0dfa6ff3be2316f91b7f148ff30617287278617a (diff)
downloadrust-ed324825341dbc763a8044f769eb0544afe87f08.tar.gz
rust-ed324825341dbc763a8044f769eb0544afe87f08.zip
Handle multiple trait-def projection candidates
-rw-r--r--compiler/rustc_trait_selection/src/traits/project.rs7
-rw-r--r--src/test/ui/associated-types/associated-types-projection-bound-ambiguity.rs16
2 files changed, 23 insertions, 0 deletions
diff --git a/compiler/rustc_trait_selection/src/traits/project.rs b/compiler/rustc_trait_selection/src/traits/project.rs
index 8f7513cf27c..c33c8a67aa8 100644
--- a/compiler/rustc_trait_selection/src/traits/project.rs
+++ b/compiler/rustc_trait_selection/src/traits/project.rs
@@ -977,6 +977,13 @@ fn assemble_candidates_from_predicates<'cx, 'tcx>(
 
             if is_match {
                 candidate_set.push_candidate(ctor(data));
+
+                if potentially_unnormalized_candidates && !obligation.predicate.needs_infer() {
+                    // HACK: Pick the first trait def candidate for a fully
+                    // inferred predicate. This is to allow duplicates that
+                    // differ only in normalization.
+                    return;
+                }
             }
         }
     }
diff --git a/src/test/ui/associated-types/associated-types-projection-bound-ambiguity.rs b/src/test/ui/associated-types/associated-types-projection-bound-ambiguity.rs
new file mode 100644
index 00000000000..353f82e7c6e
--- /dev/null
+++ b/src/test/ui/associated-types/associated-types-projection-bound-ambiguity.rs
@@ -0,0 +1,16 @@
+// Check that if we have multiple applicable projection bounds we pick one (for
+// backwards compatibility reasons).
+
+// check-pass
+use std::ops::Mul;
+
+trait A {
+    type V;
+    type U: Mul<Self::V, Output = ()> + Mul<(), Output = ()>;
+}
+
+fn g<T: A<V = ()>>() {
+    let y: <T::U as Mul<()>>::Output = ();
+}
+
+fn main() {}