Oh, you’re running this within an XPC Service! That’s a critical tibit.
Consider this snippet from your crash report:
Crashed Thread: 4 Dispatch queue: com.apple.main-thread
What’s weird about this is that the main queue is being run by thread 4. In normal application code, the main queue is always run by the main thread, that is, thread 0.
The reason this is happening is that you’re inside an XPC Service. By default an XPC Service’s main thread calls
dispatch_main
(see its
man page), and that triggers a very nice but very unusual behaviour: It terminates the main thread but leaves the process running!
This is nice because it saves resources. If the XPC Service is dormant, there’s no point leaving a thread lying around doing nothing.
It’s weird because of its impact on main queue operations. When you dispatch something to the main queue, you expect it to run on the main thread, but here there’s no main thread to run the main queue. Instead, Dispatch temporarily assigns some other worker thread to run your main queue work. Main queue work is still serialised, it’s just not running on the main thread.
The problem comes when you call AppKit, which has all sorts of ‘must run on the main thread’ checks. In situation like this these checks will fail, throwing an exception that terminates your XPC Service process.
Fortunately there’s an easy fix: Configure your XPC Service to not call
dispatch_main
. You do this by setting the
RunLoopType
property to
NSRunLoop
. See the
xpcservice.plist
man page for details.
Share and Enjoy
—
Quinn “The Eskimo!”
Apple Developer Relations, Developer Technical Support, Core OS/Hardware
let myEmail = "eskimo" + "1" + "@apple.com"