mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
285 lines
10 KiB
Swift
285 lines
10 KiB
Swift
// RUN: %target-typecheck-verify-swift
|
|
|
|
func f0(_ x: Float) -> Float {}
|
|
func f1(_ x: Float) -> Float {}
|
|
func f2(_ x: @autoclosure () -> Float) {}
|
|
|
|
var f : Float
|
|
|
|
_ = f0(f0(f))
|
|
_ = f0(1)
|
|
_ = f1(f1(f))
|
|
f2(f)
|
|
f2(1.0)
|
|
|
|
func call_lvalue(_ rhs: @autoclosure () -> Bool) -> Bool {
|
|
return rhs()
|
|
}
|
|
|
|
// Function returns
|
|
func weirdCast<T, U>(_ x: T) -> U {}
|
|
|
|
func ff() -> (Int) -> (Float) { return weirdCast }
|
|
|
|
// Block <-> function conversions
|
|
|
|
var funct: (Int) -> Int = { $0 }
|
|
var block: @convention(block) (Int) -> Int = funct
|
|
funct = block
|
|
block = funct
|
|
|
|
// Application of implicitly unwrapped optional functions
|
|
|
|
var optFunc: ((String) -> String)! = { $0 }
|
|
var s: String = optFunc("hi")
|
|
|
|
// <rdar://problem/17652759> Default arguments cause crash with tuple permutation
|
|
func testArgumentShuffle(_ first: Int = 7, third: Int = 9) {
|
|
}
|
|
testArgumentShuffle(third: 1, 2) // expected-error {{unnamed argument #2 must precede argument 'third'}} {{21-21=2, }} {{29-32=}}
|
|
|
|
|
|
|
|
func rejectsAssertStringLiteral() {
|
|
assert("foo") // expected-error {{cannot convert value of type 'String' to expected argument type 'Bool'}}
|
|
precondition("foo") // expected-error {{cannot convert value of type 'String' to expected argument type 'Bool'}}
|
|
}
|
|
|
|
|
|
|
|
// <rdar://problem/22243469> QoI: Poor error message with throws, default arguments, & overloads
|
|
func process(_ line: UInt = #line, _ fn: () -> Void) {}
|
|
func process(_ line: UInt = #line) -> Int { return 0 }
|
|
func dangerous() throws {}
|
|
|
|
func test() {
|
|
process { // expected-error {{invalid conversion from throwing function of type '() throws -> Void' to non-throwing function type '() -> Void'}}
|
|
try dangerous()
|
|
test()
|
|
}
|
|
}
|
|
|
|
|
|
// <rdar://problem/19962010> QoI: argument label mismatches produce not-great diagnostic
|
|
class A {
|
|
func a(_ text:String) { // expected-note {{incorrect labels for candidate (have: '(text:)', expected: '(_:)')}}
|
|
}
|
|
func a(_ text:String, something:Int?=nil) { // expected-note {{incorrect labels for candidate (have: '(text:)', expected: '(_:)')}}
|
|
}
|
|
}
|
|
A().a(text:"sometext") // expected-error{{no exact matches in call to instance method 'a'}}
|
|
|
|
|
|
// <rdar://problem/22451001> QoI: incorrect diagnostic when argument to print has the wrong type
|
|
func r22451001() -> AnyObject {}
|
|
print(r22451001(5)) // expected-error {{argument passed to call that takes no arguments}}
|
|
|
|
|
|
/// https://github.com/apple/swift/issues/43207
|
|
/// Passing two parameters to a function that takes one argument of type `Any`
|
|
/// crashes the compiler
|
|
do {
|
|
func f(_ x: Any) {} // expected-note {{'f' declared here}}
|
|
|
|
f(3,4) // expected-error {{extra argument in call}}
|
|
f() // expected-error {{missing argument for parameter #1 in call}}
|
|
// Make sure calling with structural tuples still works.
|
|
f(())
|
|
f((1, 2))
|
|
}
|
|
|
|
/// https://github.com/apple/swift/issues/45262
|
|
/// Poor diagnostics when function arguments should be `@escaping`
|
|
func f_45262(block: () -> (), other: () -> Int) {
|
|
|
|
class C<T> { // expected-note 4 {{generic parameters are always considered '@escaping'}}
|
|
let f: T
|
|
init(f: T) { self.f = f }
|
|
}
|
|
|
|
func takesAny(_: Any) {}
|
|
|
|
let _ = C(f: block)
|
|
// expected-error@-1 {{converting non-escaping parameter 'block' to generic parameter 'T' may allow it to escape}}
|
|
let _ = C<()->()>(f: block)
|
|
// expected-error@-1 {{converting non-escaping parameter 'block' to generic parameter 'T' may allow it to escape}}
|
|
let _: C<()->()> = C(f: block)
|
|
// expected-error@-1 {{converting non-escaping parameter 'block' to generic parameter 'T' may allow it to escape}}
|
|
let _: C<()->()> = C<()->()>(f: block)
|
|
// expected-error@-1 {{converting non-escaping parameter 'block' to generic parameter 'T' may allow it to escape}}
|
|
_ = C<Any>(f: block) // expected-error {{converting non-escaping value to 'Any' may allow it to escape}}
|
|
_ = C<Any>(f: other) // expected-error {{converting non-escaping value to 'Any' may allow it to escape}}
|
|
takesAny(block) // expected-error {{converting non-escaping value to 'Any' may allow it to escape}}
|
|
takesAny(other) // expected-error {{converting non-escaping value to 'Any' may allow it to escape}}
|
|
}
|
|
|
|
struct S {
|
|
init<T>(_ x: T, _ y: T) {} // expected-note {{generic parameters are always considered '@escaping'}}
|
|
subscript<T>() -> (T, T) -> Void { { _, _ in } } // expected-note {{generic parameters are always considered '@escaping'}}
|
|
|
|
init(fn: () -> Int) {
|
|
self.init({ 0 }, fn) // expected-error {{converting non-escaping parameter 'fn' to generic parameter 'T' may allow it to escape}}
|
|
_ = self[]({ 0 }, fn) // expected-error {{converting non-escaping parameter 'fn' to generic parameter 'T' may allow it to escape}}
|
|
}
|
|
}
|
|
|
|
protocol P {
|
|
associatedtype U
|
|
}
|
|
|
|
func test_passing_noescape_function_to_dependent_member() {
|
|
struct S<T : P> { // expected-note {{generic parameters are always considered '@escaping'}}
|
|
func foo(_: T.U) {}
|
|
}
|
|
|
|
struct Q : P {
|
|
typealias U = () -> Int
|
|
}
|
|
|
|
func test(_ s: S<Q>, fn: () -> Int) {
|
|
s.foo(fn)
|
|
// expected-error@-1 {{converting non-escaping parameter 'fn' to generic parameter 'T.U' may allow it to escape}}
|
|
}
|
|
}
|
|
|
|
protocol Q {
|
|
associatedtype U : P
|
|
}
|
|
|
|
// https://github.com/apple/swift/issues/53201
|
|
func f_53201(_ fn: () -> Int) {
|
|
struct S1 : P {
|
|
typealias U = () -> Int
|
|
}
|
|
|
|
struct S2 : Q {
|
|
typealias U = S1
|
|
}
|
|
|
|
struct S<T : Q> { // expected-note {{generic parameters are always considered '@escaping'}}
|
|
func foo(_ x: T.U.U) {}
|
|
}
|
|
|
|
S<S2>().foo(fn) // expected-error {{converting non-escaping parameter 'fn' to generic parameter 'T.U.U' may allow it to escape}}
|
|
}
|
|
|
|
struct Wrapper<U> {
|
|
var value: U
|
|
init(_ value: U) { self.value = value }
|
|
}
|
|
|
|
func with<T>(_ x: T, body: (T) -> Void) {}
|
|
func takesGeneric<T>(_ x: T) {}
|
|
func takesEscapingFn(_ fn: @escaping () -> Int) {}
|
|
func returnsTakesEscapingFn() -> (@escaping () -> Int) -> Void { takesEscapingFn }
|
|
|
|
prefix operator ^^^
|
|
prefix func ^^^(_ x: Int) -> (@escaping () -> Int) -> Void { takesEscapingFn }
|
|
|
|
func testWeirdFnExprs<T>(_ fn: () -> Int, _ cond: Bool, _ any: Any, genericArg: T) { // expected-note 12{{parameter 'fn' is implicitly non-escaping}}
|
|
(any as! (@escaping () -> Int) -> Void)(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
let wrapped = Wrapper<(@escaping () -> Int) -> Void>({ x in })
|
|
(wrapped[keyPath: \.value] as (@escaping () -> Int) -> Void)(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
(cond ? returnsTakesEscapingFn() : returnsTakesEscapingFn())(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
(^^^5)(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
(try! takesEscapingFn)(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
var optFn: Optional = takesEscapingFn
|
|
optFn?(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
[takesEscapingFn][0](fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
(takesEscapingFn, "").0(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
with({ (x: @escaping () -> Int) in }) { y in
|
|
Wrapper(y).value(fn)
|
|
// expected-error @-1{{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
}
|
|
|
|
_ = { x in (x({ 0 }), x(fn)) }(takesGeneric)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
_ = { (a: (@escaping () -> Int), b) in () }(fn, genericArg)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
|
|
func returnsVeryCurried() -> () throws -> (@escaping () -> Int) -> Void { { { x in } } }
|
|
(try? returnsVeryCurried()())?(fn)
|
|
// expected-error@-1 {{passing non-escaping parameter 'fn' to function expecting an @escaping closure}}
|
|
}
|
|
|
|
// rdar://problem/59066040 - Confusing error message about argument mismatch where the problem is escapiness
|
|
func test_passing_nonescaping_to_escaping_function() {
|
|
struct S {}
|
|
typealias Handler = (S) -> ()
|
|
|
|
func bar(_ handler: Handler?) {}
|
|
|
|
func foo(_ handler: Handler) { // expected-note {{parameter 'handler' is implicitly non-escaping}}
|
|
bar(handler) // expected-error {{passing non-escaping parameter 'handler' to function expecting an @escaping closure}}
|
|
}
|
|
}
|
|
|
|
func test_passing_noescape_function_ref_to_generic_parameter() {
|
|
func cast<T, U>(_ t: T) -> U {
|
|
return t as! U
|
|
}
|
|
|
|
class A {
|
|
required init(factory: () -> Self) {
|
|
fatalError()
|
|
}
|
|
}
|
|
|
|
struct S {
|
|
func converter() -> B { fatalError() }
|
|
}
|
|
|
|
class B : A {
|
|
class func test(value: S) {
|
|
_ = self.init(factory: cast(value.converter)) // Ok
|
|
}
|
|
}
|
|
}
|
|
|
|
// https://github.com/apple/swift/issues/57133
|
|
func f_57133<T>(_ fs: () -> T..., a _ : Int) -> T {
|
|
fs.first! // expected-error{{function produces expected type 'T'; did you mean to call it with '()'?}} {{11-11=()}}
|
|
}
|
|
|
|
func tuplify<Ts>(_ fn: (Ts) -> Void) {}
|
|
|
|
func testInvalidTupleImplosions() {
|
|
func takesVargs(_ x: Int, _ y: String...) {}
|
|
tuplify(takesVargs) // expected-error {{cannot convert value of type '(Int, String...) -> ()' to expected argument type '(Int) -> Void'}}
|
|
|
|
func takesAutoclosure(_ x: @autoclosure () -> Int, y: String) {}
|
|
tuplify(takesAutoclosure) // expected-error {{cannot convert value of type '(@autoclosure () -> Int, String) -> ()' to expected argument type '(@escaping () -> Int) -> Void'}}
|
|
|
|
func takesInout(_ x: Int, _ y: inout String) {}
|
|
tuplify(takesInout) // expected-error {{cannot convert value of type '(Int, inout String) -> ()' to expected argument type '(Int) -> Void'}}
|
|
}
|
|
|
|
// https://github.com/apple/swift/issues/57502
|
|
do {
|
|
func f<Ts>(_ fn: @escaping (Ts) -> Void) {} // expected-note {{in call to function 'f'}}
|
|
|
|
func g1(x: Int..., y: Int...) {}
|
|
f(g1) // expected-error {{cannot convert value of type '(Int..., Int...) -> ()' to expected argument type '(Ts) -> Void'}}
|
|
// expected-error@-1{{generic parameter 'Ts' could not be inferred}}
|
|
|
|
func g2(_ x: inout Int, _ y: inout Int) {}
|
|
f(g2) // expected-error {{cannot convert value of type '(inout Int, inout Int) -> ()' to expected argument type '(Int) -> Void'}}
|
|
}
|