summaryrefslogtreecommitdiffstats
path: root/tests/ui/methods/method-normalize-bounds-issue-20604.rs
diff options
context:
space:
mode:
Diffstat (limited to 'tests/ui/methods/method-normalize-bounds-issue-20604.rs')
-rw-r--r--tests/ui/methods/method-normalize-bounds-issue-20604.rs61
1 files changed, 61 insertions, 0 deletions
diff --git a/tests/ui/methods/method-normalize-bounds-issue-20604.rs b/tests/ui/methods/method-normalize-bounds-issue-20604.rs
new file mode 100644
index 000000000..9c0b95284
--- /dev/null
+++ b/tests/ui/methods/method-normalize-bounds-issue-20604.rs
@@ -0,0 +1,61 @@
+// run-pass
+#![allow(dead_code)]
+#![allow(unused_variables)]
+#![allow(stable_features)]
+
+// Test that we handle projection types which wind up important for
+// resolving methods. This test was reduced from a larger example; the
+// call to `foo()` at the end was failing to resolve because the
+// winnowing stage of method resolution failed to handle an associated
+// type projection.
+
+// pretty-expanded FIXME #23616
+
+#![feature(associated_types)]
+
+trait Hasher {
+ type Output;
+ fn finish(&self) -> Self::Output;
+}
+
+trait Hash<H: Hasher> {
+ fn hash(&self, h: &mut H);
+}
+
+trait HashState {
+ type Wut: Hasher;
+ fn hasher(&self) -> Self::Wut;
+}
+
+struct SipHasher;
+impl Hasher for SipHasher {
+ type Output = u64;
+ fn finish(&self) -> u64 { 4 }
+}
+
+impl Hash<SipHasher> for isize {
+ fn hash(&self, h: &mut SipHasher) {}
+}
+
+struct SipState;
+impl HashState for SipState {
+ type Wut = SipHasher;
+ fn hasher(&self) -> SipHasher { SipHasher }
+}
+
+struct Map<S> {
+ s: S,
+}
+
+impl<S> Map<S>
+ where S: HashState,
+ <S as HashState>::Wut: Hasher<Output=u64>,
+{
+ fn foo<K>(&self, k: K) where K: Hash< <S as HashState>::Wut> {}
+}
+
+fn foo<K: Hash<SipHasher>>(map: &Map<SipState>) {
+ map.foo(22);
+}
+
+fn main() {}