Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 3510f36

Browse files
committedJun 22, 2023
Audit pass on Sendable conformances and requirements
1 parent b339466 commit 3510f36

7 files changed

+58
-13
lines changed
 

‎Package.swift

+7-2
Original file line numberDiff line numberDiff line change
@@ -16,11 +16,16 @@ let package = Package(
1616
.library(name: "_CAsyncSequenceValidationSupport", type: .static, targets: ["AsyncSequenceValidation"]),
1717
.library(name: "AsyncAlgorithms_XCTest", targets: ["AsyncAlgorithms_XCTest"]),
1818
],
19-
dependencies: [.package(url: "https://github.com/apple/swift-collections.git", .upToNextMajor(from: "1.0.3"))],
19+
dependencies: [.package(url: "https://github.com/apple/swift-collections.git", .upToNextMajor(from: "1.0.4"))],
2020
targets: [
2121
.target(
2222
name: "AsyncAlgorithms",
23-
dependencies: [.product(name: "Collections", package: "swift-collections")]
23+
dependencies: [.product(name: "Collections", package: "swift-collections"),],
24+
swiftSettings: [
25+
.unsafeFlags([
26+
"-strict-concurrency=complete"
27+
], .when(configuration: .debug))
28+
]
2429
),
2530
.target(
2631
name: "AsyncSequenceValidation",

‎Sources/AsyncAlgorithms/Buffer/BoundedBufferStateMachine.swift

+4-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ struct BoundedBufferStateMachine<Base: AsyncSequence> {
1616
typealias SuspendedProducer = UnsafeContinuation<Void, Never>
1717
typealias SuspendedConsumer = UnsafeContinuation<Result<Base.Element, Error>?, Never>
1818

19-
private enum State {
19+
fileprivate enum State {
2020
case initial(base: Base)
2121
case buffering(
2222
task: Task<Void, Never>,
@@ -308,3 +308,6 @@ struct BoundedBufferStateMachine<Base: AsyncSequence> {
308308
}
309309
}
310310
}
311+
312+
extension BoundedBufferStateMachine: Sendable where Base: Sendable { }
313+
extension BoundedBufferStateMachine.State: Sendable where Base: Sendable { }

‎Sources/AsyncAlgorithms/Buffer/UnboundedBufferStateMachine.swift

+4-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ struct UnboundedBufferStateMachine<Base: AsyncSequence> {
2121
case bufferingOldest(Int)
2222
}
2323

24-
private enum State {
24+
fileprivate enum State {
2525
case initial(base: Base)
2626
case buffering(
2727
task: Task<Void, Never>,
@@ -248,3 +248,6 @@ struct UnboundedBufferStateMachine<Base: AsyncSequence> {
248248
}
249249
}
250250
}
251+
252+
extension UnboundedBufferStateMachine: Sendable where Base: Sendable { }
253+
extension UnboundedBufferStateMachine.State: Sendable where Base: Sendable { }

‎Sources/AsyncAlgorithms/Channels/AsyncChannel.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919
/// on the `Iterator` is made, or when `finish()` is called from another Task.
2020
/// As `finish()` induces a terminal state, there is no more need for a back pressure management.
2121
/// This function does not suspend and will finish all the pending iterations.
22-
public final class AsyncChannel<Element>: AsyncSequence, @unchecked Sendable {
22+
public final class AsyncChannel<Element: Sendable>: AsyncSequence, @unchecked Sendable {
2323
public typealias Element = Element
2424
public typealias AsyncIterator = Iterator
2525

‎Sources/AsyncAlgorithms/Channels/AsyncThrowingChannel.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@
1818
/// and is resumed when the next call to `next()` on the `Iterator` is made, or when `finish()`/`fail(_:)` is called
1919
/// from another Task. As `finish()` and `fail(_:)` induce a terminal state, there is no more need for a back pressure management.
2020
/// Those functions do not suspend and will finish all the pending iterations.
21-
public final class AsyncThrowingChannel<Element, Failure: Error>: AsyncSequence, @unchecked Sendable {
21+
public final class AsyncThrowingChannel<Element: Sendable, Failure: Error>: AsyncSequence, @unchecked Sendable {
2222
public typealias Element = Element
2323
public typealias AsyncIterator = Iterator
2424

‎Sources/AsyncAlgorithms/Channels/ChannelStateMachine.swift

+40-6
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,42 @@
1010
//===----------------------------------------------------------------------===//
1111
@_implementationOnly import OrderedCollections
1212

13-
struct ChannelStateMachine<Element, Failure: Error>: Sendable {
14-
private struct SuspendedProducer: Hashable {
13+
struct OrderedSetContainer<Element: Hashable> {
14+
var contents: OrderedSet<Element>
15+
16+
var isEmpty: Bool { contents.isEmpty }
17+
18+
mutating func removeFirst() -> Element {
19+
contents.removeFirst()
20+
}
21+
22+
mutating func remove(_ element: Element) -> Element? {
23+
contents.remove(element)
24+
}
25+
26+
@discardableResult
27+
mutating func append(_ element: Element) -> (inserted: Bool, index: Int) {
28+
contents.append(element)
29+
}
30+
31+
func map<T>(_ apply: (Element) throws -> T) rethrows -> [T] {
32+
try contents.map(apply)
33+
}
34+
}
35+
36+
extension OrderedSetContainer: ExpressibleByArrayLiteral {
37+
typealias ArrayLiteralElement = OrderedSet<Element>.ArrayLiteralElement
38+
39+
init(arrayLiteral elements: ArrayLiteralElement...) {
40+
contents = OrderedSet(elements)
41+
}
42+
}
43+
44+
extension OrderedSetContainer: @unchecked Sendable where Element: Sendable { }
45+
46+
47+
struct ChannelStateMachine<Element: Sendable, Failure: Error>: Sendable {
48+
private struct SuspendedProducer: Hashable, Sendable {
1549
let id: UInt64
1650
let continuation: UnsafeContinuation<Void, Never>?
1751
let element: Element?
@@ -29,7 +63,7 @@ struct ChannelStateMachine<Element, Failure: Error>: Sendable {
2963
}
3064
}
3165

32-
private struct SuspendedConsumer: Hashable {
66+
private struct SuspendedConsumer: Hashable, Sendable {
3367
let id: UInt64
3468
let continuation: UnsafeContinuation<Element?, any Error>?
3569

@@ -51,11 +85,11 @@ struct ChannelStateMachine<Element, Failure: Error>: Sendable {
5185
case failed(Error)
5286
}
5387

54-
private enum State {
88+
private enum State: Sendable {
5589
case channeling(
56-
suspendedProducers: OrderedSet<SuspendedProducer>,
90+
suspendedProducers: OrderedSetContainer<SuspendedProducer>,
5791
cancelledProducers: Set<SuspendedProducer>,
58-
suspendedConsumers: OrderedSet<SuspendedConsumer>,
92+
suspendedConsumers: OrderedSetContainer<SuspendedConsumer>,
5993
cancelledConsumers: Set<SuspendedConsumer>
6094
)
6195
case terminated(Termination)

‎Sources/AsyncAlgorithms/Channels/ChannelStorage.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
// See https://swift.org/LICENSE.txt for license information
99
//
1010
//===----------------------------------------------------------------------===//
11-
struct ChannelStorage<Element, Failure: Error>: Sendable {
11+
struct ChannelStorage<Element: Sendable, Failure: Error>: Sendable {
1212
private let stateMachine: ManagedCriticalState<ChannelStateMachine<Element, Failure>>
1313
private let ids = ManagedCriticalState<UInt64>(0)
1414

0 commit comments

Comments
 (0)
Please sign in to comment.