Re: [PATCH 1/2] ui/cocoa: Add cursor composition

2024-06-25 Thread Phil Dennis-Jordan
On Mon, 18 Mar 2024 at 08:59, Akihiko Odaki  wrote:
> The common pattern to implement accelerated cursor composition is to
> replace the cursor and warp it so that the replaced cursor is shown at
> the correct position on the guest display. Unfortunately, ui/cocoa
> cannot do the same because warping the cursor position interfers with
> the mouse input so it uses CALayer instead; although it is not
> specialized for cursor composition, it still can compose images with
> hardware acceleration.

As discussed in another thread, this limitation doesn't apply when
using absolute pointer input. I much prefer "native" cursor rendering,
especially in the absolute pointer case, but this is definitely an
improvement over no cursor support at all.

Perhaps some others can chip in with preferences on the way forward.

> @@ -313,6 +318,11 @@ @interface QemuCocoaView : NSView
>  BOOL isMouseGrabbed;
>  BOOL isAbsoluteEnabled;
>  CFMachPortRef eventsTap;
> +CALayer *cursorLayer;
> +QEMUCursor *cursor;

As far as I can see, these can leak on dealloc.

> +int mouseX;
> +int mouseY;
> +int mouseOn;

Any reason not to use a boolean type (bool or BOOL) for this?

> +- (void)setCursor:(QEMUCursor *)given_cursor
> +{
> […]
> +image = CGImageCreate(
> +cursor->width, //width
> +cursor->height, //height
> +8, //bitsPerComponent
> +32, //bitsPerPixel
> +cursor->width * 4, //bytesPerRow
> +CGColorSpaceCreateWithName(kCGColorSpaceSRGB), //colorspace

This leaks the colour space object; CGImageCreate unfortunately does
not consume the reference.

> +kCGBitmapByteOrder32Little | kCGImageAlphaFirst, //bitmapInfo
> +provider, //provider
> +NULL, //decode
> +0, //interpolate
> +kCGRenderingIntentDefault //intent
> +);
> +
> +CGDataProviderRelease(provider);
> +[CATransaction begin];
> +[CATransaction setDisableActions:YES];
> +[cursorLayer setBounds:bounds];
> +[cursorLayer setContents:(id)image];
> +[CATransaction commit];
> +CGImageRelease(image);
> +}

This method also runs on for quite a bit. I'd prefer some parts broken
out into a helper function or two.



[PATCH 1/2] ui/cocoa: Add cursor composition

2024-03-18 Thread Akihiko Odaki
Add accelerated cursor composition to ui/cocoa. This does not only
improve performance for display devices that exposes the capability to
the guest according to dpy_cursor_define_supported(), but fixes the
cursor display for devices that unconditionally expects the availability
of the capability (e.g., virtio-gpu).

The common pattern to implement accelerated cursor composition is to
replace the cursor and warp it so that the replaced cursor is shown at
the correct position on the guest display. Unfortunately, ui/cocoa
cannot do the same because warping the cursor position interfers with
the mouse input so it uses CALayer instead; although it is not
specialized for cursor composition, it still can compose images with
hardware acceleration.

Signed-off-by: Akihiko Odaki 
---
 meson.build |  3 +-
 ui/cocoa.m  | 97 +
 2 files changed, 99 insertions(+), 1 deletion(-)

diff --git a/meson.build b/meson.build
index b375248a7614..b198ca2972ed 100644
--- a/meson.build
+++ b/meson.build
@@ -1044,7 +1044,8 @@ if get_option('attr').allowed()
   endif
 endif
 
-cocoa = dependency('appleframeworks', modules: ['Cocoa', 'CoreVideo'],
+cocoa = dependency('appleframeworks',
+   modules: ['Cocoa', 'CoreVideo', 'QuartzCore'],
required: get_option('cocoa'))
 
 vmnet = dependency('appleframeworks', modules: 'vmnet', required: 
get_option('vmnet'))
diff --git a/ui/cocoa.m b/ui/cocoa.m
index fa879d7dcd4b..c13c2a01e947 100644
--- a/ui/cocoa.m
+++ b/ui/cocoa.m
@@ -25,6 +25,7 @@
 #include "qemu/osdep.h"
 
 #import 
+#import 
 #include 
 
 #include "qemu/help-texts.h"
@@ -92,12 +93,16 @@ static void cocoa_switch(DisplayChangeListener *dcl,
  DisplaySurface *surface);
 
 static void cocoa_refresh(DisplayChangeListener *dcl);
+static void cocoa_mouse_set(DisplayChangeListener *dcl, int x, int y, int on);
+static void cocoa_cursor_define(DisplayChangeListener *dcl, QEMUCursor 
*cursor);
 
 static const DisplayChangeListenerOps dcl_ops = {
 .dpy_name  = "cocoa",
 .dpy_gfx_update = cocoa_update,
 .dpy_gfx_switch = cocoa_switch,
 .dpy_refresh = cocoa_refresh,
+.dpy_mouse_set = cocoa_mouse_set,
+.dpy_cursor_define = cocoa_cursor_define,
 };
 static DisplayChangeListener dcl = {
 .ops = &dcl_ops,
@@ -313,6 +318,11 @@ @interface QemuCocoaView : NSView
 BOOL isMouseGrabbed;
 BOOL isAbsoluteEnabled;
 CFMachPortRef eventsTap;
+CALayer *cursorLayer;
+QEMUCursor *cursor;
+int mouseX;
+int mouseY;
+int mouseOn;
 }
 - (void) switchSurface:(pixman_image_t *)image;
 - (void) grabMouse;
@@ -365,6 +375,12 @@ - (id)initWithFrame:(NSRect)frameRect
 #if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_VERSION_14_0
 [self setClipsToBounds:YES];
 #endif
+[self setWantsLayer:YES];
+cursorLayer = [[CALayer alloc] init];
+[cursorLayer setAnchorPoint:CGPointMake(0, 1)];
+[cursorLayer setAutoresizingMask:kCALayerMaxXMargin |
+ kCALayerMinYMargin];
+[[self layer] addSublayer:cursorLayer];
 
 }
 return self;
@@ -445,6 +461,72 @@ - (void) unhideCursor
 [NSCursor unhide];
 }
 
+- (void)setMouseX:(int)x y:(int)y on:(int)on
+{
+CGPoint position;
+
+mouseX = x;
+mouseY = y;
+mouseOn = on;
+
+position.x = mouseX;
+position.y = screen.height - mouseY;
+
+[CATransaction begin];
+[CATransaction setDisableActions:YES];
+[cursorLayer setPosition:position];
+[cursorLayer setHidden:!mouseOn];
+[CATransaction commit];
+}
+
+- (void)setCursor:(QEMUCursor *)given_cursor
+{
+CGDataProviderRef provider;
+CGImageRef image;
+CGRect bounds = CGRectZero;
+
+cursor_unref(cursor);
+cursor = given_cursor;
+
+if (!cursor) {
+return;
+}
+
+cursor_ref(cursor);
+
+bounds.size.width = cursor->width;
+bounds.size.height = cursor->height;
+
+provider = CGDataProviderCreateWithData(
+NULL,
+cursor->data,
+cursor->width * cursor->height * 4,
+NULL
+);
+
+image = CGImageCreate(
+cursor->width, //width
+cursor->height, //height
+8, //bitsPerComponent
+32, //bitsPerPixel
+cursor->width * 4, //bytesPerRow
+CGColorSpaceCreateWithName(kCGColorSpaceSRGB), //colorspace
+kCGBitmapByteOrder32Little | kCGImageAlphaFirst, //bitmapInfo
+provider, //provider
+NULL, //decode
+0, //interpolate
+kCGRenderingIntentDefault //intent
+);
+
+CGDataProviderRelease(provider);
+[CATransaction begin];
+[CATransaction setDisableActions:YES];
+[cursorLayer setBounds:bounds];
+[cursorLayer setContents:(id)image];
+[CATransaction commit];
+CGImageRelease(image);
+}
+
 - (void) drawRect:(NSRect) rect
 {
 COCOA_DEBUG("QemuCocoaView: drawRect\n");
@@ -1982,6 +2064,21 @@ static void cocoa_refresh(Display