1 /*****************************************************************************
2 * caopengllayer.m: CAOpenGLLayer (Mac OS X) video output
3 *****************************************************************************
4 * Copyright (C) 2014-2017 VLC authors and VideoLAN
7 * Authors: David Fuhrmann <david dot fuhrmann at googlemail dot com>
8 * Felix Paul Kühne <fkuehne at videolan dot org>
9 * Pierre d'Herbemont <pdherbemont at videolan dot org>
11 * This program is free software; you can redistribute it and/or modify it
12 * under the terms of the GNU Lesser General Public License as published by
13 * the Free Software Foundation; either version 2.1 of the License, or
14 * (at your option) any later version.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU Lesser General Public License for more details.
21 * You should have received a copy of the GNU Lesser General Public License
22 * along with this program; if not, write to the Free Software Foundation,
23 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24 *****************************************************************************/
26 /*****************************************************************************
28 *****************************************************************************/
34 #include <vlc_common.h>
35 #include <vlc_plugin.h>
36 #include <vlc_vout_display.h>
37 #include <vlc_opengl.h>
39 #import <QuartzCore/QuartzCore.h>
40 #import <Cocoa/Cocoa.h>
41 #import <OpenGL/OpenGL.h>
42 #import <dlfcn.h> /* dlsym */
44 #include "opengl/vout_helper.h"
46 #define OSX_SIERRA_AND_HIGHER (NSAppKitVersionNumber >= 1485)
48 /*****************************************************************************
50 *****************************************************************************/
51 static int Open (vlc_object_t *);
52 static void Close (vlc_object_t *);
55 set_description(N_("Core Animation OpenGL Layer (Mac OS X)"))
56 set_capability("vout display", 0)
57 set_category(CAT_VIDEO)
58 set_subcategory(SUBCAT_VIDEO_VOUT)
59 set_callbacks(Open, Close)
62 static picture_pool_t *Pool (vout_display_t *vd, unsigned requested_count);
63 static void PictureRender (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture);
64 static void PictureDisplay (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture);
65 static int Control (vout_display_t *vd, int query, va_list ap);
67 static void *OurGetProcAddress (vlc_gl_t *gl, const char *name);
68 static int OpenglLock (vlc_gl_t *gl);
69 static void OpenglUnlock (vlc_gl_t *gl);
70 static void OpenglSwap (vlc_gl_t *gl);
72 @protocol VLCCoreAnimationVideoLayerEmbedding <NSObject>
73 - (void)addVoutLayer:(CALayer *)aLayer;
74 - (void)removeVoutLayer:(CALayer *)aLayer;
75 - (CGSize)currentOutputSize;
78 @interface VLCCAOpenGLLayer : CAOpenGLLayer
80 @property (nonatomic, readwrite) vout_display_t* voutDisplay;
81 @property (nonatomic, readwrite) CGLContextObj glContext;
86 struct vout_display_sys_t {
89 picture_resource_t resource;
91 CALayer <VLCCoreAnimationVideoLayerEmbedding> *container;
93 VLCCAOpenGLLayer *cgLayer;
96 vout_display_opengl_t *vgl;
98 vout_display_place_t place;
100 bool b_frame_available;
105 CGLContextObj locked_ctx;
106 VLCCAOpenGLLayer *cgLayer;
109 /*****************************************************************************
110 * Open: This function allocates and initializes the OpenGL vout method.
111 *****************************************************************************/
112 static int Open (vlc_object_t *p_this)
114 vout_display_t *vd = (vout_display_t *)p_this;
115 vout_display_sys_t *sys;
117 /* Allocate structure */
118 vd->sys = sys = calloc(1, sizeof(vout_display_sys_t));
123 id container = var_CreateGetAddress(vd, "drawable-nsobject");
125 vout_display_DeleteWindow(vd, NULL);
127 sys->embed = vout_display_NewWindow(vd, VOUT_WINDOW_TYPE_NSOBJECT);
129 container = sys->embed->handle.nsobject;
132 msg_Err(vd, "No drawable-nsobject found!");
137 /* store for later, released in Close() */
138 sys->container = [container retain];
140 [CATransaction begin];
141 sys->cgLayer = [[VLCCAOpenGLLayer alloc] init];
142 [sys->cgLayer setVoutDisplay:vd];
144 [sys->cgLayer performSelectorOnMainThread:@selector(display)
148 if ([container respondsToSelector:@selector(addVoutLayer:)]) {
149 msg_Dbg(vd, "container implements implicit protocol");
150 [container addVoutLayer:sys->cgLayer];
151 } else if ([container respondsToSelector:@selector(addSublayer:)] ||
152 [container isKindOfClass:[CALayer class]]) {
153 msg_Dbg(vd, "container doesn't implement implicit protocol, fallback mode used");
154 [container addSublayer:sys->cgLayer];
156 msg_Err(vd, "Provided NSObject container isn't compatible");
157 [sys->cgLayer release];
159 [CATransaction commit];
162 [CATransaction commit];
167 if (![sys->cgLayer glContext])
168 msg_Warn(vd, "we might not have an OpenGL context yet");
170 /* Initialize common OpenGL video display */
171 sys->gl = vlc_object_create(vd, sizeof(*sys->gl));
172 if (unlikely(!sys->gl))
174 sys->gl->makeCurrent = OpenglLock;
175 sys->gl->releaseCurrent = OpenglUnlock;
176 sys->gl->swap = OpenglSwap;
177 sys->gl->getProcAddress = OurGetProcAddress;
179 struct gl_sys *glsys = sys->gl->sys = malloc(sizeof(*glsys));
182 glsys->locked_ctx = NULL;
183 glsys->cgLayer = sys->cgLayer;
185 const vlc_fourcc_t *subpicture_chromas;
186 video_format_t fmt = vd->fmt;
187 if (!OpenglLock(sys->gl)) {
188 sys->vgl = vout_display_opengl_New(&vd->fmt, &subpicture_chromas,
189 sys->gl, &vd->cfg->viewpoint);
190 OpenglUnlock(sys->gl);
194 msg_Err(vd, "Error while initializing opengl display.");
198 /* setup vout display */
199 vout_display_info_t info = vd->info;
200 info.subpicture_chromas = subpicture_chromas;
204 vd->prepare = PictureRender;
205 vd->display = PictureDisplay;
206 vd->control = Control;
208 if (OSX_SIERRA_AND_HIGHER) {
209 /* request our screen's HDR mode (introduced in OS X 10.11, but correctly supported in 10.12 only) */
210 if ([sys->cgLayer respondsToSelector:@selector(setWantsExtendedDynamicRangeContent:)]) {
211 [sys->cgLayer setWantsExtendedDynamicRangeContent:YES];
215 /* setup initial state */
217 if ([container respondsToSelector:@selector(currentOutputSize)])
218 outputSize = [container currentOutputSize];
220 outputSize = [sys->container visibleRect].size;
221 vout_display_SendEventDisplaySize(vd, (int)outputSize.width, (int)outputSize.height);
231 static void Close (vlc_object_t *p_this)
233 vout_display_t *vd = (vout_display_t *)p_this;
234 vout_display_sys_t *sys = vd->sys;
237 if ([sys->container respondsToSelector:@selector(removeVoutLayer:)])
238 [sys->container removeVoutLayer:sys->cgLayer];
240 [sys->cgLayer removeFromSuperlayer];
241 [sys->cgLayer release];
245 [sys->container release];
248 vout_display_DeleteWindow(vd, sys->embed);
250 if (sys->vgl != NULL && !OpenglLock(sys->gl)) {
251 vout_display_opengl_Delete(sys->vgl);
252 OpenglUnlock(sys->gl);
257 if (sys->gl->sys != NULL)
259 assert(((struct gl_sys *)sys->gl->sys)->locked_ctx == NULL);
262 vlc_object_release(sys->gl);
265 if ([sys->cgLayer glContext])
266 CGLReleaseContext([sys->cgLayer glContext]);
271 static picture_pool_t *Pool (vout_display_t *vd, unsigned count)
273 vout_display_sys_t *sys = vd->sys;
275 if (!sys->pool && !OpenglLock(sys->gl)) {
276 sys->pool = vout_display_opengl_GetPool(sys->vgl, count);
277 OpenglUnlock(sys->gl);
283 static void PictureRender (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture)
285 vout_display_sys_t *sys = vd->sys;
288 msg_Warn(vd, "invalid pic, skipping frame");
292 @synchronized (sys->cgLayer) {
293 if (!OpenglLock(sys->gl)) {
294 vout_display_opengl_Prepare(sys->vgl, pic, subpicture);
295 OpenglUnlock(sys->gl);
300 static void PictureDisplay (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture)
302 vout_display_sys_t *sys = vd->sys;
304 @synchronized (sys->cgLayer) {
305 sys->b_frame_available = YES;
307 /* Calling display on the non-main thread is not officially supported, but
308 * its suggested at several places and works fine here. Flush is thread-safe
309 * and makes sure the picture is actually displayed. */
310 [sys->cgLayer display];
311 [CATransaction flush];
314 picture_Release(pic);
317 subpicture_Delete(subpicture);
320 static int Control (vout_display_t *vd, int query, va_list ap)
322 vout_display_sys_t *sys = vd->sys;
329 case VOUT_DISPLAY_CHANGE_DISPLAY_SIZE:
330 case VOUT_DISPLAY_CHANGE_DISPLAY_FILLED:
331 case VOUT_DISPLAY_CHANGE_ZOOM:
332 case VOUT_DISPLAY_CHANGE_SOURCE_ASPECT:
333 case VOUT_DISPLAY_CHANGE_SOURCE_CROP:
335 const vout_display_cfg_t *cfg;
337 if (query == VOUT_DISPLAY_CHANGE_SOURCE_ASPECT || query == VOUT_DISPLAY_CHANGE_SOURCE_CROP) {
340 cfg = (const vout_display_cfg_t*)va_arg (ap, const vout_display_cfg_t *);
343 /* we always use our current frame here */
344 vout_display_cfg_t cfg_tmp = *cfg;
345 [CATransaction lock];
346 CGRect bounds = [sys->cgLayer visibleRect];
347 [CATransaction unlock];
348 cfg_tmp.display.width = bounds.size.width;
349 cfg_tmp.display.height = bounds.size.height;
351 /* Reverse vertical alignment as the GL tex are Y inverted */
352 if (cfg_tmp.align.vertical == VOUT_DISPLAY_ALIGN_TOP)
353 cfg_tmp.align.vertical = VOUT_DISPLAY_ALIGN_BOTTOM;
354 else if (cfg_tmp.align.vertical == VOUT_DISPLAY_ALIGN_BOTTOM)
355 cfg_tmp.align.vertical = VOUT_DISPLAY_ALIGN_TOP;
357 vout_display_place_t place;
358 vout_display_PlacePicture (&place, &vd->source, &cfg_tmp, false);
359 if (OpenglLock(sys->gl))
362 vout_display_opengl_SetWindowAspectRatio(sys->vgl, (float)place.width / place.height);
363 OpenglUnlock(sys->gl);
370 case VOUT_DISPLAY_CHANGE_VIEWPOINT:
374 if (OpenglLock(sys->gl))
377 ret = vout_display_opengl_SetViewpoint(sys->vgl,
378 &va_arg (ap, const vout_display_cfg_t* )->viewpoint);
379 OpenglUnlock(sys->gl);
383 case VOUT_DISPLAY_RESET_PICTURES:
384 vlc_assert_unreachable ();
386 msg_Err (vd, "Unhandled request %d", query);
394 #pragma mark OpenGL callbacks
396 static int OpenglLock (vlc_gl_t *gl)
398 struct gl_sys *sys = gl->sys;
399 assert(sys->locked_ctx == NULL);
401 CGLContextObj ctx = [sys->cgLayer glContext];
406 CGLError err = CGLLockContext(ctx);
407 if (kCGLNoError == err) {
408 sys->locked_ctx = ctx;
409 CGLSetCurrentContext(ctx);
415 static void OpenglUnlock (vlc_gl_t *gl)
417 struct gl_sys *sys = gl->sys;
418 CGLUnlockContext(sys->locked_ctx);
419 sys->locked_ctx = NULL;
422 static void OpenglSwap (vlc_gl_t *gl)
427 static void *OurGetProcAddress (vlc_gl_t *gl, const char *name)
431 return dlsym(RTLD_DEFAULT, name);
435 #pragma mark CA layer
437 /*****************************************************************************
438 * @implementation VLCCAOpenGLLayer
439 *****************************************************************************/
440 @implementation VLCCAOpenGLLayer
446 [CATransaction lock];
447 self.needsDisplayOnBoundsChange = YES;
448 self.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable;
449 self.asynchronous = NO;
450 [CATransaction unlock];
456 - (void)setVoutDisplay:(vout_display_t *)aVd
461 - (void)resizeWithOldSuperlayerSize:(CGSize)size
463 [super resizeWithOldSuperlayerSize: size];
465 CGSize boundsSize = self.visibleRect.size;
468 vout_display_SendEventDisplaySize(_voutDisplay, boundsSize.width, boundsSize.height);
471 - (BOOL)canDrawInCGLContext:(CGLContextObj)glContext pixelFormat:(CGLPixelFormatObj)pixelFormat forLayerTime:(CFTimeInterval)timeInterval displayTime:(const CVTimeStamp *)timeStamp
473 /* Only draw the frame if we have a frame that was previously rendered */
477 return _voutDisplay->sys->b_frame_available;
480 - (void)drawInCGLContext:(CGLContextObj)glContext pixelFormat:(CGLPixelFormatObj)pixelFormat forLayerTime:(CFTimeInterval)timeInterval displayTime:(const CVTimeStamp *)timeStamp
484 vout_display_sys_t *sys = _voutDisplay->sys;
489 CGRect bounds = [self visibleRect];
491 // x / y are top left corner, but we need the lower left one
492 glViewport (sys->place.x, bounds.size.height - (sys->place.y + sys->place.height), sys->place.width, sys->place.height);
494 // flush is also done by this method, no need to call super
495 vout_display_opengl_Display (sys->vgl, &_voutDisplay->source);
496 sys->b_frame_available = NO;
499 -(CGLPixelFormatObj)copyCGLPixelFormatForDisplayMask:(uint32_t)mask
501 // The default is fine for this demonstration.
502 return [super copyCGLPixelFormatForDisplayMask:mask];
505 - (CGLContextObj)copyCGLContextForPixelFormat:(CGLPixelFormatObj)pixelFormat
507 // Only one opengl context is allowed for the module lifetime
509 msg_Dbg(_voutDisplay, "Return existing context: %p", _glContext);
513 CGLContextObj context = [super copyCGLContextForPixelFormat:pixelFormat];
515 // Swap buffers only during the vertical retrace of the monitor.
516 //http://developer.apple.com/documentation/GraphicsImaging/
517 //Conceptual/OpenGL/chap5/chapter_5_section_44.html /
520 CGLSetParameter( CGLGetCurrentContext(), kCGLCPSwapInterval,
523 @synchronized (self) {
524 _glContext = context;
530 - (void)releaseCGLContext:(CGLContextObj)glContext
532 // do not release anything here, we do that when closing the module
535 - (void)mouseButtonDown:(int)buttonNumber
537 @synchronized (self) {
539 if (buttonNumber == 0)
540 vout_display_SendEventMousePressed (_voutDisplay, MOUSE_BUTTON_LEFT);
541 else if (buttonNumber == 1)
542 vout_display_SendEventMousePressed (_voutDisplay, MOUSE_BUTTON_RIGHT);
544 vout_display_SendEventMousePressed (_voutDisplay, MOUSE_BUTTON_CENTER);
549 - (void)mouseButtonUp:(int)buttonNumber
551 @synchronized (self) {
553 if (buttonNumber == 0)
554 vout_display_SendEventMouseReleased (_voutDisplay, MOUSE_BUTTON_LEFT);
555 else if (buttonNumber == 1)
556 vout_display_SendEventMouseReleased (_voutDisplay, MOUSE_BUTTON_RIGHT);
558 vout_display_SendEventMouseReleased (_voutDisplay, MOUSE_BUTTON_CENTER);
563 - (void)mouseMovedToX:(double)xValue Y:(double)yValue
565 @synchronized (self) {
567 vout_display_SendMouseMovedDisplayCoordinates (_voutDisplay,
571 &_voutDisplay->sys->place);