ui/cocoa: Perform UI operations only on the main thread

The OSX Mojave release is more picky about enforcing the Cocoa API
restriction that only the main thread may perform UI calls. To
accommodate this we need to restructure the Cocoa code:
 * the special OSX main() creates a second thread and uses
   that to call the vl.c qemu_main(); the original main
   thread goes into the OSX event loop
 * the refresh, switch and update callbacks asynchronously
   tell the main thread to do the necessary work
 * the refresh callback no longer does the "get events from the
   UI event queue and handle them" loop, since we now use
   the stock OSX event loop. Instead our NSApplication sendEvent
   method will either deal with them or pass them on to OSX

All these things have to be changed in one commit, to avoid
breaking bisection.

Note that since we use dispatch_get_main_queue(), this bumps
our minimum version requirement to OSX 10.10 Yosemite (released
in 2014, unsupported by Apple since 2017).

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Roman Bolshakov <r.bolshakov@yadro.com>
Tested-by: Roman Bolshakov <r.bolshakov@yadro.com>
Message-id: 20190225102433.22401-8-peter.maydell@linaro.org
Message-id: 20190214102816.3393-8-peter.maydell@linaro.org
This commit is contained in:
Peter Maydell 2019-02-25 10:24:33 +00:00
parent 61a2ed447e
commit 5588840ff7
1 changed files with 116 additions and 79 deletions

View File

@ -129,6 +129,9 @@ bool stretch_video;
NSTextField *pauseLabel; NSTextField *pauseLabel;
NSArray * supportedImageFileTypes; NSArray * supportedImageFileTypes;
static QemuSemaphore display_init_sem;
static QemuSemaphore app_started_sem;
// Utility functions to run specified code block with iothread lock held // Utility functions to run specified code block with iothread lock held
typedef void (^CodeBlock)(void); typedef void (^CodeBlock)(void);
typedef bool (^BoolCodeBlock)(void); typedef bool (^BoolCodeBlock)(void);
@ -325,6 +328,7 @@ static void handleAnyDeviceErrors(Error * err)
NSWindow *fullScreenWindow; NSWindow *fullScreenWindow;
float cx,cy,cw,ch,cdx,cdy; float cx,cy,cw,ch,cdx,cdy;
CGDataProviderRef dataProviderRef; CGDataProviderRef dataProviderRef;
pixman_image_t *pixman_image;
BOOL modifiers_state[256]; BOOL modifiers_state[256];
BOOL isMouseGrabbed; BOOL isMouseGrabbed;
BOOL isFullscreen; BOOL isFullscreen;
@ -383,8 +387,10 @@ QemuCocoaView *cocoaView;
{ {
COCOA_DEBUG("QemuCocoaView: dealloc\n"); COCOA_DEBUG("QemuCocoaView: dealloc\n");
if (dataProviderRef) if (dataProviderRef) {
CGDataProviderRelease(dataProviderRef); CGDataProviderRelease(dataProviderRef);
pixman_image_unref(pixman_image);
}
[super dealloc]; [super dealloc];
} }
@ -535,13 +541,16 @@ QemuCocoaView *cocoaView;
} }
// update screenBuffer // update screenBuffer
if (dataProviderRef) if (dataProviderRef) {
CGDataProviderRelease(dataProviderRef); CGDataProviderRelease(dataProviderRef);
pixman_image_unref(pixman_image);
}
//sync host window color space with guests //sync host window color space with guests
screen.bitsPerPixel = PIXMAN_FORMAT_BPP(image_format); screen.bitsPerPixel = PIXMAN_FORMAT_BPP(image_format);
screen.bitsPerComponent = DIV_ROUND_UP(screen.bitsPerPixel, 8) * 2; screen.bitsPerComponent = DIV_ROUND_UP(screen.bitsPerPixel, 8) * 2;
pixman_image = image;
dataProviderRef = CGDataProviderCreateWithData(NULL, pixman_image_get_data(image), w * 4 * h, NULL); dataProviderRef = CGDataProviderCreateWithData(NULL, pixman_image_get_data(image), w * 4 * h, NULL);
// update windows // update windows
@ -1013,7 +1022,6 @@ QemuCocoaView *cocoaView;
#endif #endif
{ {
} }
- (void)startEmulationWithArgc:(int)argc argv:(char**)argv;
- (void)doToggleFullScreen:(id)sender; - (void)doToggleFullScreen:(id)sender;
- (void)toggleFullScreen:(id)sender; - (void)toggleFullScreen:(id)sender;
- (void)showQEMUDoc:(id)sender; - (void)showQEMUDoc:(id)sender;
@ -1101,8 +1109,8 @@ QemuCocoaView *cocoaView;
- (void)applicationDidFinishLaunching: (NSNotification *) note - (void)applicationDidFinishLaunching: (NSNotification *) note
{ {
COCOA_DEBUG("QemuCocoaAppController: applicationDidFinishLaunching\n"); COCOA_DEBUG("QemuCocoaAppController: applicationDidFinishLaunching\n");
// launch QEMU, with the global args /* Tell cocoa_display_init to proceed */
[self startEmulationWithArgc:gArgc argv:(char **)gArgv]; qemu_sem_post(&app_started_sem);
} }
- (void)applicationWillTerminate:(NSNotification *)aNotification - (void)applicationWillTerminate:(NSNotification *)aNotification
@ -1145,15 +1153,6 @@ QemuCocoaView *cocoaView;
[cocoaView raiseAllKeys]; [cocoaView raiseAllKeys];
} }
- (void)startEmulationWithArgc:(int)argc argv:(char**)argv
{
COCOA_DEBUG("QemuCocoaAppController: startEmulationWithArgc\n");
int status;
status = qemu_main(argc, argv, *_NSGetEnviron());
exit(status);
}
/* We abstract the method called by the Enter Fullscreen menu item /* We abstract the method called by the Enter Fullscreen menu item
* because Mac OS 10.7 and higher disables it. This is because of the * because Mac OS 10.7 and higher disables it. This is because of the
* menu item's old selector's name toggleFullScreen: * menu item's old selector's name toggleFullScreen:
@ -1489,7 +1488,9 @@ QemuCocoaView *cocoaView;
- (void)sendEvent:(NSEvent *)event - (void)sendEvent:(NSEvent *)event
{ {
COCOA_DEBUG("QemuApplication: sendEvent\n"); COCOA_DEBUG("QemuApplication: sendEvent\n");
[super sendEvent: event]; if (![cocoaView handleEvent:event]) {
[super sendEvent: event];
}
} }
@end @end
@ -1672,32 +1673,59 @@ static void addRemovableDevicesMenuItems(void)
qapi_free_BlockInfoList(pointerToFree); qapi_free_BlockInfoList(pointerToFree);
} }
int main (int argc, const char * argv[]) { /*
* The startup process for the OSX/Cocoa UI is complicated, because
* OSX insists that the UI runs on the initial main thread, and so we
* need to start a second thread which runs the vl.c qemu_main():
*
* Initial thread: 2nd thread:
* in main():
* create qemu-main thread
* wait on display_init semaphore
* call qemu_main()
* ...
* in cocoa_display_init():
* post the display_init semaphore
* wait on app_started semaphore
* create application, menus, etc
* enter OSX run loop
* in applicationDidFinishLaunching:
* post app_started semaphore
* tell main thread to fullscreen if needed
* [...]
* run qemu main-loop
*
* We do this in two stages so that we don't do the creation of the
* GUI application menus and so on for command line options like --help
* where we want to just print text to stdout and exit immediately.
*/
static void *call_qemu_main(void *opaque)
{
int status;
COCOA_DEBUG("Second thread: calling qemu_main()\n");
status = qemu_main(gArgc, gArgv, *_NSGetEnviron());
COCOA_DEBUG("Second thread: qemu_main() returned, exiting\n");
exit(status);
}
int main (int argc, const char * argv[]) {
QemuThread thread;
COCOA_DEBUG("Entered main()\n");
gArgc = argc; gArgc = argc;
gArgv = (char **)argv; gArgv = (char **)argv;
int i;
/* In case we don't need to display a window, let's not do that */ qemu_sem_init(&display_init_sem, 0);
for (i = 1; i < argc; i++) { qemu_sem_init(&app_started_sem, 0);
const char *opt = argv[i];
if (opt[0] == '-') { qemu_thread_create(&thread, "qemu_main", call_qemu_main,
/* Treat --foo the same as -foo. */ NULL, QEMU_THREAD_DETACHED);
if (opt[1] == '-') {
opt++; COCOA_DEBUG("Main thread: waiting for display_init_sem\n");
} qemu_sem_wait(&display_init_sem);
if (!strcmp(opt, "-h") || !strcmp(opt, "-help") || COCOA_DEBUG("Main thread: initializing app\n");
!strcmp(opt, "-vnc") ||
!strcmp(opt, "-nographic") ||
!strcmp(opt, "-version") ||
!strcmp(opt, "-curses") ||
!strcmp(opt, "-display") ||
!strcmp(opt, "-qtest")) {
return qemu_main(gArgc, gArgv, *_NSGetEnviron());
}
}
}
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
@ -1710,12 +1738,24 @@ int main (int argc, const char * argv[]) {
create_initial_menus(); create_initial_menus();
/*
* Create the menu entries which depend on QEMU state (for consoles
* and removeable devices). These make calls back into QEMU functions,
* which is OK because at this point we know that the second thread
* holds the iothread lock and is synchronously waiting for us to
* finish.
*/
add_console_menu_entries();
addRemovableDevicesMenuItems();
// Create an Application controller // Create an Application controller
QemuCocoaAppController *appController = [[QemuCocoaAppController alloc] init]; QemuCocoaAppController *appController = [[QemuCocoaAppController alloc] init];
[NSApp setDelegate:appController]; [NSApp setDelegate:appController];
// Start the main event loop // Start the main event loop
COCOA_DEBUG("Main thread: entering OSX run loop\n");
[NSApp run]; [NSApp run];
COCOA_DEBUG("Main thread: left OSX run loop, exiting\n");
[appController release]; [appController release];
[pool release]; [pool release];
@ -1733,17 +1773,19 @@ static void cocoa_update(DisplayChangeListener *dcl,
COCOA_DEBUG("qemu_cocoa: cocoa_update\n"); COCOA_DEBUG("qemu_cocoa: cocoa_update\n");
NSRect rect; dispatch_async(dispatch_get_main_queue(), ^{
if ([cocoaView cdx] == 1.0) { NSRect rect;
rect = NSMakeRect(x, [cocoaView gscreen].height - y - h, w, h); if ([cocoaView cdx] == 1.0) {
} else { rect = NSMakeRect(x, [cocoaView gscreen].height - y - h, w, h);
rect = NSMakeRect( } else {
x * [cocoaView cdx], rect = NSMakeRect(
([cocoaView gscreen].height - y - h) * [cocoaView cdy], x * [cocoaView cdx],
w * [cocoaView cdx], ([cocoaView gscreen].height - y - h) * [cocoaView cdy],
h * [cocoaView cdy]); w * [cocoaView cdx],
} h * [cocoaView cdy]);
[cocoaView setNeedsDisplayInRect:rect]; }
[cocoaView setNeedsDisplayInRect:rect];
});
[pool release]; [pool release];
} }
@ -1752,9 +1794,19 @@ static void cocoa_switch(DisplayChangeListener *dcl,
DisplaySurface *surface) DisplaySurface *surface)
{ {
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
pixman_image_t *image = surface->image;
COCOA_DEBUG("qemu_cocoa: cocoa_switch\n"); COCOA_DEBUG("qemu_cocoa: cocoa_switch\n");
[cocoaView switchSurface:surface->image];
// The DisplaySurface will be freed as soon as this callback returns.
// We take a reference to the underlying pixman image here so it does
// not disappear from under our feet; the switchSurface method will
// deref the old image when it is done with it.
pixman_image_ref(image);
dispatch_async(dispatch_get_main_queue(), ^{
[cocoaView switchSurface:image];
});
[pool release]; [pool release];
} }
@ -1766,26 +1818,15 @@ static void cocoa_refresh(DisplayChangeListener *dcl)
graphic_hw_update(NULL); graphic_hw_update(NULL);
if (qemu_input_is_absolute()) { if (qemu_input_is_absolute()) {
if (![cocoaView isAbsoluteEnabled]) { dispatch_async(dispatch_get_main_queue(), ^{
if ([cocoaView isMouseGrabbed]) { if (![cocoaView isAbsoluteEnabled]) {
[cocoaView ungrabMouse]; if ([cocoaView isMouseGrabbed]) {
[cocoaView ungrabMouse];
}
} }
} [cocoaView setAbsoluteEnabled:YES];
[cocoaView setAbsoluteEnabled:YES]; });
} }
NSDate *distantPast;
NSEvent *event;
distantPast = [NSDate distantPast];
do {
event = [NSApp nextEventMatchingMask:NSEventMaskAny untilDate:distantPast
inMode: NSDefaultRunLoopMode dequeue:YES];
if (event != nil) {
if (![cocoaView handleEvent:event]) {
[NSApp sendEvent:event];
}
}
} while(event != nil);
[pool release]; [pool release];
} }
@ -1806,10 +1847,17 @@ static void cocoa_display_init(DisplayState *ds, DisplayOptions *opts)
{ {
COCOA_DEBUG("qemu_cocoa: cocoa_display_init\n"); COCOA_DEBUG("qemu_cocoa: cocoa_display_init\n");
/* Tell main thread to go ahead and create the app and enter the run loop */
qemu_sem_post(&display_init_sem);
qemu_sem_wait(&app_started_sem);
COCOA_DEBUG("cocoa_display_init: app start completed\n");
/* if fullscreen mode is to be used */ /* if fullscreen mode is to be used */
if (opts->has_full_screen && opts->full_screen) { if (opts->has_full_screen && opts->full_screen) {
[NSApp activateIgnoringOtherApps: YES]; dispatch_async(dispatch_get_main_queue(), ^{
[(QemuCocoaAppController *)[[NSApplication sharedApplication] delegate] toggleFullScreen: nil]; [NSApp activateIgnoringOtherApps: YES];
[(QemuCocoaAppController *)[[NSApplication sharedApplication] delegate] toggleFullScreen: nil];
});
} }
dcl = g_malloc0(sizeof(DisplayChangeListener)); dcl = g_malloc0(sizeof(DisplayChangeListener));
@ -1820,17 +1868,6 @@ static void cocoa_display_init(DisplayState *ds, DisplayOptions *opts)
// register cleanup function // register cleanup function
atexit(cocoa_cleanup); atexit(cocoa_cleanup);
/* At this point QEMU has created all the consoles, so we can add View
* menu entries for them.
*/
add_console_menu_entries();
/* Give all removable devices a menu item.
* Has to be called after QEMU has started to
* find out what removable devices it has.
*/
addRemovableDevicesMenuItems();
} }
static QemuDisplay qemu_display_cocoa = { static QemuDisplay qemu_display_cocoa = {