Files
swift-mirror/test/Concurrency/sendable_conformance_checking.swift
Michael Gottesman df2844d267 [concurrency] Cleanup verify-additional-prefix usage in tests.
This includes a bunch of fixes. It is not exhaustive but fit my time boxed time
period I set aside to look at this today.

A quick non-exhaustive list:

1. I removed unnecessary verify-additional-prefix lines.
2. Split tests with typechecker error and non-typechecker error components.
3. Removed complete- lines that we used when testing w/without send
non sednable.
4. Translated complete-and-tns- lines to be just complete- since they are just
testing strict-concurrency=complete and we are not testing complete without
send non sendable anymore.
2025-08-12 10:56:02 -07:00

219 lines
8.0 KiB
Swift

// RUN: %target-swift-frontend -emit-sil %s -o /dev/null -verify
// RUN: %target-swift-frontend -emit-sil %s -o /dev/null -verify -strict-concurrency=targeted
// RUN: %target-swift-frontend -emit-sil %s -o /dev/null -verify -verify-additional-prefix complete- -strict-concurrency=complete
// REQUIRES: concurrency
@available(SwiftStdlib 5.1, *)
class NotSendable { // expected-note 9{{class 'NotSendable' does not conform to the 'Sendable' protocol}}
}
@available(SwiftStdlib 5.1, *)
@available(*, unavailable)
extension NotSendable: Sendable { }
@available(SwiftStdlib 5.1, *)
protocol IsolatedWithNotSendableRequirements: Actor {
func f() -> NotSendable
var prop: NotSendable { get }
func fAsync() async -> NotSendable
}
// Okay, everything is isolated the same way
@available(SwiftStdlib 5.1, *)
actor A1: IsolatedWithNotSendableRequirements {
func f() -> NotSendable { NotSendable() }
var prop: NotSendable { NotSendable() }
func fAsync() async -> NotSendable { NotSendable() }
}
// Okay, sendable checking occurs when calling through the protocol
// and also inside the bodies.
@available(SwiftStdlib 5.1, *)
actor A2: IsolatedWithNotSendableRequirements {
nonisolated func f() -> NotSendable { NotSendable() }
nonisolated var prop: NotSendable { NotSendable() }
nonisolated func fAsync() async -> NotSendable { NotSendable() }
// expected-warning@-1{{non-Sendable type 'NotSendable' cannot be returned from nonisolated implementation to caller of protocol requirement 'fAsync()'}}
}
@available(SwiftStdlib 5.1, *)
protocol AsyncProtocolWithNotSendable {
func f() async -> NotSendable
var prop: NotSendable { get async }
}
// Sendable checking required because calls through protocol cross into the
// actor's domain.
@available(SwiftStdlib 5.1, *)
actor A3: AsyncProtocolWithNotSendable {
func f() async -> NotSendable { NotSendable() } // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'f()'}}
var prop: NotSendable { // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'prop'}}
get async {
NotSendable()
}
}
}
// Sendable checking required because calls through protocol cross into the
// actor's domain.
@available(SwiftStdlib 5.1, *)
actor A4: AsyncProtocolWithNotSendable {
func f() -> NotSendable { NotSendable() } // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'f()'}}
var prop: NotSendable { // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'prop'}}
get {
NotSendable()
}
}
}
// Sendable checking not required because we never cross into the actor's
// domain.
@available(SwiftStdlib 5.1, *)
actor A5: AsyncProtocolWithNotSendable {
nonisolated func f() async -> NotSendable { NotSendable() }
nonisolated var prop: NotSendable {
get async {
NotSendable()
}
}
}
// Sendable checking not required because we never cross into the actor's
// domain.
@available(SwiftStdlib 5.1, *)
actor A6: AsyncProtocolWithNotSendable {
nonisolated func f() -> NotSendable { NotSendable() }
nonisolated var prop: NotSendable {
get {
NotSendable()
}
}
}
@available(SwiftStdlib 5.1, *)
protocol AsyncThrowingProtocolWithNotSendable {
func f() async throws -> NotSendable
var prop: NotSendable { get async throws }
}
// Sendable checking required because calls through protocol cross into the
// actor's domain.
@available(SwiftStdlib 5.1, *)
actor A7: AsyncThrowingProtocolWithNotSendable {
func f() async -> NotSendable { NotSendable() } // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'f()'}}
var prop: NotSendable { // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'prop'}}
get async {
NotSendable()
}
}
}
// Sendable checking required because calls through protocol cross into the
// actor's domain.
@available(SwiftStdlib 5.1, *)
actor A8: AsyncThrowingProtocolWithNotSendable {
func f() -> NotSendable { NotSendable() } // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'f()'}}
var prop: NotSendable { // expected-warning{{non-Sendable type 'NotSendable' cannot be returned from actor-isolated implementation to caller of protocol requirement 'prop'}}
get {
NotSendable()
}
}
}
// Sendable checking not required because we never cross into the actor's
// domain.
@available(SwiftStdlib 5.1, *)
actor A9: AsyncThrowingProtocolWithNotSendable {
nonisolated func f() async -> NotSendable { NotSendable() }
nonisolated var prop: NotSendable {
get async {
NotSendable()
}
}
}
// Sendable checking not required because we never cross into the actor's
// domain.
@available(SwiftStdlib 5.1, *)
actor A10: AsyncThrowingProtocolWithNotSendable {
nonisolated func f() -> NotSendable { NotSendable() }
nonisolated var prop: NotSendable {
get {
NotSendable()
}
}
}
// rdar://86653457 - Crash due to missing Sendable conformances.
// expected-warning @+1 {{non-final class 'Klass' cannot conform to the 'Sendable' protocol; this is an error in the Swift 6 language mode}}
class Klass<Output: Sendable>: Sendable {}
// expected-complete-warning @+1 {{type 'S' does not conform to the 'Sendable' protocol}}
final class SubKlass: Klass<[S]> {}
// expected-complete-note @+1 {{consider making struct 'S' conform to the 'Sendable' protocol}}
public struct S {}
// rdar://88700507 - redundant conformance of @MainActor-isolated subclass to 'Sendable'
@available(SwiftStdlib 5.1, *)
@MainActor class MainSuper {}
@available(SwiftStdlib 5.1, *)
class MainSub: MainSuper, @unchecked Sendable {}
class SendableSuper: @unchecked Sendable {}
class SendableSub: SendableSuper, @unchecked Sendable {}
class SendableExtSub: SendableSuper {}
extension SendableExtSub: @unchecked Sendable {}
// Still want to know about same-class redundancy
class MultiConformance: @unchecked Sendable {} // expected-note {{'MultiConformance' declares conformance to protocol 'Sendable' here}}
extension MultiConformance: @unchecked Sendable {} // expected-warning {{redundant conformance of 'MultiConformance' to protocol 'Sendable'}}
@available(SwiftStdlib 5.1, *)
actor MyActor {
// expected-warning@+1 {{non-final class 'Nested' cannot conform to the 'Sendable' protocol; this is an error in the Swift 6 language mode}}
class Nested: Sendable {}
}
@Sendable func globalFn() {}
protocol NoSendableReqs {
var prop: () -> Void { get }
static var staticProp: () -> Void { get }
}
public struct TestSendableWitnesses1 : NoSendableReqs {
var prop: @Sendable () -> Void // Ok (no warnings)
static let staticProp: @Sendable () -> Void = { } // Ok (no warnings)
}
public struct TestSendableWitnesses2 : NoSendableReqs {
var prop = globalFn // Ok (no warnings)
static let staticProp = globalFn // Ok (no warnings)
}
// @preconcurrency attributes to make it akin to an imported Obj-C API
@preconcurrency @MainActor public protocol EscapingSendableProtocol {
// expected-complete-note @+1 {{protocol requires function 'f(handler:)' with type '(@escaping @MainActor @Sendable (Int) -> Void) -> ()'}}
@preconcurrency func f(handler: @escaping @MainActor @Sendable (Int) -> Void)
}
// TODO: The following error should actually be a warning.
// expected-complete-error @+2 {{type 'TestEscapingOnly' does not conform to protocol 'EscapingSendableProtocol'}}
// expected-complete-note @+1 {{add stubs for conformance}}
class TestEscapingOnly: EscapingSendableProtocol {
// expected-complete-note @+1 {{candidate has non-matching type '(@escaping (Int) -> Void) -> ()'}}
func f(handler: @escaping (Int) -> Void) {}
}