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 6018f07

Browse files
committedJun 4, 2024·
Code review
1 parent 88e9535 commit 6018f07

File tree

4 files changed

+108
-17
lines changed

4 files changed

+108
-17
lines changed
 

‎Benchmarks/Benchmarks/NIOPosixBenchmarks/Benchmarks.swift

+28-7
Original file line numberDiff line numberDiff line change
@@ -20,18 +20,18 @@ private let eventLoop = MultiThreadedEventLoopGroup.singleton.next()
2020
let benchmarks = {
2121
let defaultMetrics: [BenchmarkMetric] = [
2222
.mallocCountTotal,
23+
.cpuTotal
2324
]
2425

2526
Benchmark(
2627
"TCPEcho",
2728
configuration: .init(
2829
metrics: defaultMetrics,
29-
timeUnits: .milliseconds,
30-
scalingFactor: .mega
30+
scalingFactor: .one
3131
)
3232
) { benchmark in
3333
try runTCPEcho(
34-
numberOfWrites: benchmark.scaledIterations.upperBound,
34+
numberOfWrites: 1_000_000,
3535
eventLoop: eventLoop
3636
)
3737
}
@@ -40,11 +40,10 @@ let benchmarks = {
4040
// to serial executor is also gated behind 5.9.
4141
#if compiler(>=5.9)
4242
Benchmark(
43-
"TCPEchoAsyncChannel",
43+
"TCPEchoAsyncChannel using globalHook 1M times",
4444
configuration: .init(
4545
metrics: defaultMetrics,
46-
timeUnits: .milliseconds,
47-
scalingFactor: .mega,
46+
scalingFactor: .one,
4847
// We are expecting a bit of allocation variance due to an allocation
4948
// in the Concurrency runtime which happens when resuming a continuation.
5049
thresholds: [.mallocCountTotal: .init(absolute: [.p90: 2000])],
@@ -59,9 +58,31 @@ let benchmarks = {
5958
)
6059
) { benchmark in
6160
try await runTCPEchoAsyncChannel(
62-
numberOfWrites: benchmark.scaledIterations.upperBound,
61+
numberOfWrites: 1_000_000,
6362
eventLoop: eventLoop
6463
)
6564
}
6665
#endif
66+
67+
#if compiler(>=6.0)
68+
if #available(macOS 15.0, *) {
69+
Benchmark(
70+
"TCPEchoAsyncChannel using task executor preference",
71+
configuration: .init(
72+
metrics: defaultMetrics,
73+
scalingFactor: .one
74+
// We are expecting a bit of allocation variance due to an allocation
75+
// in the Concurrency runtime which happens when resuming a continuation.
76+
// thresholds: [.mallocCountTotal: .init(absolute: [.p90: 2000])]
77+
)
78+
) { benchmark in
79+
try await withTaskExecutorPreference(eventLoop.taskExecutor) {
80+
try await runTCPEchoAsyncChannel(
81+
numberOfWrites: 1_000_000,
82+
eventLoop: eventLoop
83+
)
84+
}
85+
}
86+
}
87+
#endif
6788
}

‎Sources/NIOEmbedded/AsyncTestingEventLoop.swift

+6
Original file line numberDiff line numberDiff line change
@@ -355,6 +355,12 @@ public final class NIOAsyncTestingEventLoop: EventLoop, @unchecked Sendable {
355355
extension NIOAsyncTestingEventLoop: NIOSerialEventLoopExecutor { }
356356
#endif
357357

358+
// MARK: TaskExecutor conformance
359+
#if compiler(>=6.0)
360+
@available(macOS 9999.0, iOS 9999.0, watchOS 9999.0, tvOS 9999.0, *)
361+
extension NIOAsyncTestingEventLoop: NIOTaskEventLoopExecutor { }
362+
#endif
363+
358364
/// This is a thread-safe promise creation store.
359365
///
360366
/// We use this to keep track of where promises come from in the `NIOAsyncTestingEventLoop`.

‎Sources/NIOEmbedded/Embedded.swift

+7
Original file line numberDiff line numberDiff line change
@@ -240,6 +240,13 @@ public final class EmbeddedEventLoop: EventLoop {
240240
fatalError("EmbeddedEventLoop is not thread safe and cannot be used as a SerialExecutor. Use NIOAsyncTestingEventLoop instead.")
241241
}
242242
#endif
243+
244+
#if compiler(>=6.0)
245+
@available(macOS 9999.0, iOS 9999.0, watchOS 9999.0, tvOS 9999.0, *)
246+
public var taskExecutor: any TaskExecutor {
247+
fatalError("EmbeddedEventLoop is not thread safe and cannot be used as a TaskExecutor. Use NIOAsyncTestingEventLoop instead.")
248+
}
249+
#endif
243250
}
244251

245252
@usableFromInline

‎Tests/NIOPosixTests/TaskExecutorTests.swift

+67-10
Original file line numberDiff line numberDiff line change
@@ -17,32 +17,89 @@ import NIOPosix
1717
import XCTest
1818

1919
final class TaskExecutorTests: XCTestCase {
20+
21+
#if compiler(>=6.0)
2022
@available(macOS 9999.0, iOS 9999.0, watchOS 9999.0, tvOS 9999.0, *)
21-
func testBasicExecutorFitsOnEventLoop_MTELG() async throws {
23+
func _runTests(loop1: some EventLoop, loop2: some EventLoop) async {
24+
await withTaskGroup(of: Void.self) { taskGroup in
25+
taskGroup.addTask(executorPreference: loop1.taskExecutor) {
26+
loop1.assertInEventLoop()
27+
loop2.assertNotInEventLoop()
28+
29+
withUnsafeCurrentTask { task in
30+
// this currently fails on macOS
31+
XCTAssertEqual(task?.unownedTaskExecutor, loop1.taskExecutor.asUnownedTaskExecutor())
32+
}
33+
}
34+
35+
taskGroup.addTask(executorPreference: loop2.taskExecutor) {
36+
loop1.assertNotInEventLoop()
37+
loop2.assertInEventLoop()
38+
39+
withUnsafeCurrentTask { task in
40+
// this currently fails on macOS
41+
XCTAssertEqual(task?.unownedTaskExecutor, loop2.taskExecutor.asUnownedTaskExecutor())
42+
}
43+
}
44+
}
45+
46+
let task = Task(executorPreference: loop1.taskExecutor) {
47+
loop1.assertInEventLoop()
48+
loop2.assertNotInEventLoop()
49+
50+
withUnsafeCurrentTask { task in
51+
// this currently fails on macOS
52+
XCTAssertEqual(task?.unownedTaskExecutor, loop1.taskExecutor.asUnownedTaskExecutor())
53+
}
54+
}
55+
56+
await task.value
57+
}
58+
#endif
59+
60+
@available(macOS 9999.0, iOS 9999.0, watchOS 9999.0, tvOS 9999.0, *)
61+
func testSelectableEventLoopAsTaskExecutor() async throws {
2262
#if compiler(>=6.0)
2363
let group = MultiThreadedEventLoopGroup(numberOfThreads: 2)
2464
defer {
2565
try! group.syncShutdownGracefully()
2666
}
27-
let loops = Array(group.makeIterator())
67+
var iterator = group.makeIterator()
68+
let loop1 = iterator.next()!
69+
let loop2 = iterator.next()!
70+
71+
await self._runTests(loop1: loop1, loop2: loop2)
72+
#endif
73+
}
74+
75+
@available(macOS 9999.0, iOS 9999.0, watchOS 9999.0, tvOS 9999.0, *)
76+
func testAsyncTestingEventLoopAsTaskExecutor() async throws {
77+
#if compiler(>=6.0)
78+
let loop1 = NIOAsyncTestingEventLoop()
79+
let loop2 = NIOAsyncTestingEventLoop()
80+
defer {
81+
try? loop1.syncShutdownGracefully()
82+
try? loop2.syncShutdownGracefully()
83+
}
84+
2885
await withTaskGroup(of: Void.self) { taskGroup in
29-
taskGroup.addTask(executorPreference: loops[0].taskExecutor) {
30-
loops[0].assertInEventLoop()
31-
loops[1].assertNotInEventLoop()
86+
taskGroup.addTask(executorPreference: loop1.taskExecutor) {
87+
loop1.assertInEventLoop()
88+
loop2.assertNotInEventLoop()
3289

3390
withUnsafeCurrentTask { task in
3491
// this currently fails on macOS
35-
XCTAssertEqual(task?.unownedTaskExecutor, loops[0].taskExecutor.asUnownedTaskExecutor())
92+
XCTAssertEqual(task?.unownedTaskExecutor, loop1.taskExecutor.asUnownedTaskExecutor())
3693
}
3794
}
3895

39-
taskGroup.addTask(executorPreference: loops[1].taskExecutor) {
40-
loops[0].assertNotInEventLoop()
41-
loops[1].assertInEventLoop()
96+
taskGroup.addTask(executorPreference: loop2) {
97+
loop1.assertNotInEventLoop()
98+
loop2.assertInEventLoop()
4299

43100
withUnsafeCurrentTask { task in
44101
// this currently fails on macOS
45-
XCTAssertEqual(task?.unownedTaskExecutor, loops[1].taskExecutor.asUnownedTaskExecutor())
102+
XCTAssertEqual(task?.unownedTaskExecutor, loop2.taskExecutor.asUnownedTaskExecutor())
46103
}
47104
}
48105
}

0 commit comments

Comments
 (0)
Please sign in to comment.