Files
swift-mirror/test/decl/protocol/req/func.swift
Doug Gregor 38c1de69e4 Reinstate "[SR-511][Parse] Add 'associatedtype' keyword and fixit"
This reverts commit ce7b2bcf09, tweaking
a few validation tests appropriately (1 crasher fixed, two -verify
tests that needed updating).
2016-01-14 00:21:48 -08:00

251 lines
7.2 KiB
Swift

// RUN: %target-parse-verify-swift
// Test function requirements within protocols, as well as conformance to
// said protocols.
// Simple function
protocol P1 {
func f0()
}
// Simple match
struct X1a : P1 {
func f0() {}
}
// Simple match selecting among two overloads.
struct X1b : P1 {
func f0() -> Int {}
func f0() {}
}
// Function with an associated type
protocol P2 {
associatedtype Assoc : P1 // expected-note{{ambiguous inference of associated type 'Assoc': 'X1a' vs. 'X1b'}}
// expected-note@-1{{protocol requires nested type 'Assoc'}}
func f1(x: Assoc) // expected-note{{protocol requires function 'f1' with type 'Assoc -> ()'}} expected-note{{protocol requires function 'f1' with type 'Assoc -> ()'}}
}
// Exact match.
struct X2a : P2 {
typealias Assoc = X1a
func f1(x: X1a) {}
}
// Select among overloads.
struct X2d : P2 {
typealias Assoc = X1a
func f1(x: Int) { }
func f1(x: X1a) { }
}
struct X2e : P2 {
typealias Assoc = X1a
func f1(x: X1b) { }
func f1(x: X1a) { }
}
// Select among overloads distinguished by name.
struct X2f : P2 {
typealias Assoc = X1a
func f1(y y: X1a) { }
func f1(x: X1a) { }
}
// Infer associated type from function parameter
struct X2g : P2 {
func f1(x: X1a) { }
}
// Static/non-static mismatch.
struct X2w : P2 { // expected-error{{type 'X2w' does not conform to protocol 'P2'}}
typealias Assoc = X1a
static func f1(x: X1a) { } // expected-note{{candidate operates on a type, not an instance as required}}
}
// Deduction of type that doesn't meet requirements
struct X2x : P2 { // expected-error{{type 'X2x' does not conform to protocol 'P2'}}
func f1(x x: Int) { }
}
// Mismatch in parameter types
struct X2y : P2 { // expected-error{{type 'X2y' does not conform to protocol 'P2'}}
typealias Assoc = X1a
func f1(x x: X1b) { } // expected-note{{candidate has non-matching type '(x: X1b) -> ()'}}
}
// Ambiguous deduction
struct X2z : P2 { // expected-error{{type 'X2z' does not conform to protocol 'P2'}}
func f1(x: X1a) { } // expected-note{{matching requirement 'f1' to this declaration inferred associated type to 'X1a'}}
func f1(x: X1b) { } // expected-note{{matching requirement 'f1' to this declaration inferred associated type to 'X1b'}}
}
// Protocol with prefix unary function
prefix operator ~~ {}
protocol P3 {
associatedtype Assoc : P1
prefix func ~~(_: Self) -> Assoc // expected-note{{protocol requires function '~~' with type 'X3z -> Assoc'}}
}
// Global operator match
struct X3a : P3 {
typealias Assoc = X1a
}
prefix func ~~(_: X3a) -> X1a {} // expected-note{{candidate has non-matching type '(X3a) -> X1a'}} expected-note{{candidate is prefix, not postfix as required}}
// FIXME: Add example with overloaded prefix/postfix
// Prefix/postfix mismatch.
struct X3z : P3 { // expected-error{{type 'X3z' does not conform to protocol 'P3'}}
typealias Assoc = X1a
}
postfix func ~~(_: X3z) -> X1a {} // expected-note{{candidate is postfix, not prefix as required}} expected-note{{candidate has non-matching type '(X3z) -> X1a'}}
// Protocol with postfix unary function
postfix operator ~~ {}
protocol P4 {
associatedtype Assoc : P1
postfix func ~~ (_: Self) -> Assoc // expected-note{{protocol requires function '~~' with type 'X4z -> Assoc'}}
}
// Global operator match
struct X4a : P4 {
typealias Assoc = X1a
}
postfix func ~~(_: X4a) -> X1a {} // expected-note{{candidate has non-matching type '(X4a) -> X1a'}} expected-note{{candidate is postfix, not prefix as required}}
// Prefix/postfix mismatch.
struct X4z : P4 { // expected-error{{type 'X4z' does not conform to protocol 'P4'}}
typealias Assoc = X1a
}
prefix func ~~(_: X4z) -> X1a {} // expected-note{{candidate has non-matching type '(X4z) -> X1a'}} expected-note{{candidate is prefix, not postfix as required}}
// Objective-C protocol
@objc protocol P5 {
func f2(x: Int, withInt a: Int) // expected-note{{protocol requires function 'f2(_:withInt:)' with type '(Int, withInt: Int) -> ()'}}
func f2(x: Int, withOtherInt a: Int) // expected-note{{protocol requires function 'f2(_:withOtherInt:)' with type '(Int, withOtherInt: Int) -> ()'}}
}
// Exact match.
class X5a : P5 {
@objc func f2(x: Int, withInt a: Int) {}
@objc func f2(x: Int, withOtherInt a: Int) {}
}
// Body parameter names can vary.
class X5b : P5 {
@objc func f2(y: Int, withInt a: Int) {}
@objc func f2(y: Int, withOtherInt a: Int) {}
}
class X5c : P5 {
@objc func f2(y: Int, withInt b: Int) {}
@objc func f2(y: Int, withOtherInt b: Int) {}
}
// Names need to match up for an Objective-C protocol as well.
class X5d : P5 { // expected-error{{type 'X5d' does not conform to protocol 'P5'}}
@objc func f2(y y: Int, withInt a: Int) {} // expected-note {{Objective-C method 'f2WithY:withInt:' provided by method 'f2(y:withInt:)' does not match the requirement's selector ('f2:withInt:')}}
// expected-note@-1{{Objective-C method 'f2WithY:withInt:' provided by method 'f2(y:withInt:)' does not match the requirement's selector ('f2:withOtherInt:')}}
@objc func f2(y y: Int, withOtherValue a: Int) {} // expected-note{{Objective-C method 'f2WithY:withOtherValue:' provided by method 'f2(y:withOtherValue:)' does not match the requirement's selector ('f2:withInt:')}}
// expected-note@-1{{Objective-C method 'f2WithY:withOtherValue:' provided by method 'f2(y:withOtherValue:)' does not match the requirement's selector ('f2:withOtherInt:')}}
}
// Distinguish names within tuple arguments.
typealias T0 = (x: Int, y: String)
typealias T1 = (xx: Int, y: String)
func f(args: T0) {
}
func f(args: T1) {
}
f(T0(1, "Hi"))
infix operator ~>> { precedence 255 }
func ~>> (x: Int, args: T0) {}
func ~>> (x: Int, args: T1) {}
3~>>T0(1, "Hi")
3~>>T1(2, "Hi")
protocol Crankable {
func ~>> (x: Self, args: T0)
func ~>> (x: Self, args: T1)
}
extension Int : Crankable {}
// Invalid witnesses.
protocol P6 {
func foo(x: Int)
func bar(x x: Int) // expected-note{{protocol requires function 'bar(x:)' with type '(x: Int) -> ()'}}
}
struct X6 : P6 { // expected-error{{type 'X6' does not conform to protocol 'P6'}}
func foo(x: Missing) { } // expected-error{{use of undeclared type 'Missing'}}
func bar() { } // expected-note{{candidate has non-matching type '() -> ()'}}
}
protocol P7 {
func foo(x: Blarg) // expected-error{{use of undeclared type 'Blarg'}}
}
struct X7 : P7 { }
// Selecting the most specialized witness.
prefix operator %%% {}
protocol P8 {
func foo()
}
prefix func %%% <T : P8>(x: T) -> T { }
protocol P9 : P8 {
prefix func %%% (x: Self) -> Self
}
struct X9 : P9 {
func foo() {}
}
prefix func %%%(x: X9) -> X9 { }
protocol P10 {
associatedtype Assoc
func bar(x: Assoc)
}
struct X10 : P10 {
typealias Assoc = Int
func bar(x: Int) { }
func bar<T>(x: T) { }
}
protocol P11 {
func ==(x: Self, y: Self) -> Bool
}
protocol P12 {
associatedtype Index : P1 // expected-note{{unable to infer associated type 'Index' for protocol 'P12'}}
func getIndex() -> Index
}
struct XIndexType : P11 { }
struct X12 : P12 { // expected-error{{type 'X12' does not conform to protocol 'P12'}}
func getIndex() -> XIndexType { return XIndexType() } // expected-note{{inferred type 'XIndexType' (by matching requirement 'getIndex()') is invalid: does not conform to 'P1'}}
}
func ==(x: X12.Index, y: X12.Index) -> Bool { return true }