Skip to content

Commit 39f2ecb

Browse files
committedFeb 27, 2025
Fail gracefully when ALooper_forThread returns nil
1 parent 8fa528e commit 39f2ecb

File tree

2 files changed

+16
-9
lines changed

2 files changed

+16
-9
lines changed
 

‎Sources/AndroidLooper/AndroidLooper.swift

+16-7
Original file line numberDiff line numberDiff line change
@@ -50,15 +50,23 @@ public struct AndroidLooper: ~Copyable, @unchecked Sendable {
5050
}
5151

5252
// Called from applicaton entry point
53-
public static func setupMainLooper() {
53+
public static func setupMainLooper() -> Bool {
54+
logger.debug("setupMainLooper")
5455
if _mainLooper != nil {
5556
ALooper_release(_mainLooper)
57+
_mainLooper = nil
5658
}
59+
5760
_mainLooper = ALooper_forThread()
61+
if _mainLooper == nil {
62+
// this happens sometimes when running in test cases
63+
logger.debug("setupMainLooper: ALooper_forThread: no looper found for thread")
64+
return false
65+
}
5866
ALooper_acquire(_mainLooper)
5967

6068
// override the global executors to wake the main looper to drain the queue whenever something is scheduled
61-
AndroidMainActor.installGlobalExecutor()
69+
return AndroidMainActor.installGlobalExecutor()
6270
}
6371

6472
func deinitMainLooper() {
@@ -134,7 +142,7 @@ public struct AndroidLooper: ~Copyable, @unchecked Sendable {
134142

135143
public typealias LooperCallback = @convention(c) (CInt, CInt, UnsafeMutableRawPointer?) -> CInt
136144

137-
private var _mainLooper: OpaquePointer?
145+
private var _mainLooper: OpaquePointer? = nil
138146

139147
public extension AndroidLooper {
140148
static var main: Self {
@@ -226,7 +234,6 @@ open class AndroidLooperExecutor: SerialExecutor, @unchecked Sendable {
226234

227235
/// Enqueue a single job
228236
public func enqueue(_ job: UnownedJob) {
229-
//logger.info("### AndroidLooperExecutor.enqueue(\(job)")
230237
_queue.withLock { queue in
231238
queue.append(job)
232239
}
@@ -261,11 +268,12 @@ private extension AndroidMainActor {
261268
/// https://github.com/swiftlang/swift-evolution/blob/main/proposals/0392-custom-actor-executors.md#overriding-the-mainactor-executor
262269
///
263270
/// See also [a draft proposal for custom executors](https://github.com/rjmccall/swift-evolution/blob/custom-executors/proposals/0000-custom-executors.md#the-default-global-concurrent-executor)
264-
static func installGlobalExecutor() {
265-
guard !didInstallGlobalExecutor else { return }
271+
static func installGlobalExecutor() -> Bool {
272+
if didInstallGlobalExecutor {
273+
return false
274+
}
266275
didInstallGlobalExecutor = true
267276

268-
269277
let looperCallback: LooperCallback = { ft, event, data in
270278
while true {
271279
switch CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0.0, true) {
@@ -290,6 +298,7 @@ private extension AndroidMainActor {
290298

291299
let dispatchPort = _dispatch_get_main_queue_port_4CF()
292300
let result = ALooper_addFd(_mainLooper, dispatchPort, 0, CInt(ALOOPER_EVENT_INPUT), looperCallback, nil)
301+
return result == 1 // Returns 1 if the file descriptor was added or -1 if an error occurred.
293302
}
294303
}
295304

‎Tests/AndroidNativeTests/AndroidNativeTests.swift

-2
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@ class AndroidNativeTests : XCTestCase {
1111

1212
for i in 0..<100 {
1313
tasks.append(Task(priority: [.low, .medium, .high].randomElement()!) {
14-
//print("### Task: \(Thread.current)")
1514
assert(!Thread.isMainThread)
1615
return await actorDemo.add(n1: i, n2: i)
1716
})
@@ -32,7 +31,6 @@ class AndroidNativeTests : XCTestCase {
3231
}
3332

3433
func add(n1: Int, n2: Int) -> Int {
35-
//print("### MainActorDemo: \(Thread.current)")
3634
assert(Thread.isMainThread)
3735
return n1 + n2
3836
}

0 commit comments

Comments
 (0)