about summary refs log tree commit diff
path: root/tests/ui/traits/assignability-trait.rs
diff options
context:
space:
mode:
Diffstat (limited to 'tests/ui/traits/assignability-trait.rs')
-rw-r--r--tests/ui/traits/assignability-trait.rs47
1 files changed, 47 insertions, 0 deletions
diff --git a/tests/ui/traits/assignability-trait.rs b/tests/ui/traits/assignability-trait.rs
new file mode 100644
index 00000000000..a8547c1d271
--- /dev/null
+++ b/tests/ui/traits/assignability-trait.rs
@@ -0,0 +1,47 @@
+// run-pass
+#![allow(non_camel_case_types)]
+
+// Tests that type assignability is used to search for instances when
+// making method calls, but only if there aren't any matches without
+// it.
+
+trait iterable<A> {
+    fn iterate<F>(&self, blk: F) -> bool where F: FnMut(&A) -> bool;
+}
+
+impl<'a,A> iterable<A> for &'a [A] {
+    fn iterate<F>(&self, f: F) -> bool where F: FnMut(&A) -> bool {
+        self.iter().all(f)
+    }
+}
+
+impl<A> iterable<A> for Vec<A> {
+    fn iterate<F>(&self, f: F) -> bool where F: FnMut(&A) -> bool {
+        self.iter().all(f)
+    }
+}
+
+fn length<A, T: iterable<A>>(x: T) -> usize {
+    let mut len = 0;
+    x.iterate(|_y| {
+        len += 1;
+        true
+    });
+    return len;
+}
+
+pub fn main() {
+    let x: Vec<isize> = vec![0,1,2,3];
+    // Call a method
+    x.iterate(|y| { assert_eq!(x[*y as usize], *y); true });
+    // Call a parameterized function
+    assert_eq!(length(x.clone()), x.len());
+    // Call a parameterized function, with type arguments that require
+    // a borrow
+    assert_eq!(length::<isize, &[isize]>(&*x), x.len());
+
+    // Now try it with a type that *needs* to be borrowed
+    let z = [0,1,2,3];
+    // Call a parameterized function
+    assert_eq!(length::<isize, &[isize]>(&z), z.len());
+}