]> git.sesse.net Git - vlc/blob - modules/video_output/opengllayer.m
remove bogus code introduced in the previous commit
[vlc] / modules / video_output / opengllayer.m
1 /*****************************************************************************
2  * opengl.c: CAOpenGLLayer (Mac OS X) video output. Display a video output in
3  * a layer. The layer will register itself to the drawable object stored in 
4  * the "drawable" variable. 
5  *****************************************************************************
6  * Copyright (C) 2004 the VideoLAN team
7  * $Id$
8  *
9  * Authors: Cyril Deguet <asmax@videolan.org>
10  *          Gildas Bazin <gbazin@videolan.org>
11  *          Eric Petit <titer@m0k.org>
12  *          Cedric Cocquebert <cedric.cocquebert@supelec.fr>
13  *          Pierre d'Herbemont <pdherbemont # videolan.org>
14  *
15  * This program is free software; you can redistribute it and/or modify
16  * it under the terms of the GNU General Public License as published by
17  * the Free Software Foundation; either version 2 of the License, or
18  * (at your option) any later version.
19  *
20  * This program is distributed in the hope that it will be useful,
21  * but WITHOUT ANY WARRANTY; without even the implied warranty of
22  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
23  * GNU General Public License for more details.
24  *
25  * You should have received a copy of the GNU General Public License
26  * along with this program; if not, write to the Free Software
27  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
28  *****************************************************************************/
29
30 /*****************************************************************************
31  * Preamble
32  *****************************************************************************/
33 #include <errno.h>                                                 /* ENOMEM */
34
35 #ifdef HAVE_CONFIG_H
36 # include "config.h"
37 #endif
38
39 #include <vlc/vlc.h>
40 #include <vlc_vout.h>
41
42 #import <QuartzCore/QuartzCore.h>
43 #import <Cocoa/Cocoa.h>
44 #import <OpenGL/OpenGL.h>
45
46 #ifdef CALayer
47
48 /* On OS X, use GL_TEXTURE_RECTANGLE_EXT instead of GL_TEXTURE_2D.
49    This allows sizes which are not powers of 2 */
50 #define VLCGL_TARGET GL_TEXTURE_RECTANGLE_EXT
51
52 /* OS X OpenGL supports YUV. Hehe. */
53 #define VLCGL_FORMAT GL_YCBCR_422_APPLE
54 #define VLCGL_TYPE   GL_UNSIGNED_SHORT_8_8_APPLE
55
56 /* RV32 */
57 #define VLCGL_RGB_FORMAT GL_RGBA
58 #define VLCGL_RGB_TYPE GL_UNSIGNED_BYTE
59
60 /* YUY2 */
61 #ifndef YCBCR_MESA
62 #define YCBCR_MESA 0x8757
63 #endif
64 #ifndef UNSIGNED_SHORT_8_8_MESA
65 #define UNSIGNED_SHORT_8_8_MESA 0x85BA
66 #endif
67 #define VLCGL_YUV_FORMAT YCBCR_MESA
68 #define VLCGL_YUV_TYPE UNSIGNED_SHORT_8_8_MESA
69
70
71 #ifndef GL_CLAMP_TO_EDGE
72 #   define GL_CLAMP_TO_EDGE 0x812F
73 #endif
74
75 @interface VLCVideoView : NSObject
76 - (void)addVoutLayer:(CALayer *)layer;
77 @end
78
79 /*****************************************************************************
80  * Vout interface
81  *****************************************************************************/
82 static int  CreateVout   ( vlc_object_t * );
83 static void DestroyVout  ( vlc_object_t * );
84 static int  Init         ( vout_thread_t * );
85 static void End          ( vout_thread_t * );
86 static int  Manage       ( vout_thread_t * );
87 static void Render       ( vout_thread_t *, picture_t * );
88 static void DisplayVideo ( vout_thread_t *, picture_t * );
89 static int  Control      ( vout_thread_t *, int, va_list );
90
91 static int InitTextures  ( vout_thread_t * );
92
93 vlc_module_begin();
94     set_shortname( "OpenGLLayer" );
95     set_category( CAT_VIDEO );
96     set_subcategory( SUBCAT_VIDEO_VOUT );
97     set_description( _("Core Animation OpenGL Layer (Mac OS X)") );
98     set_capability( "video output", 20 );
99     add_shortcut( "opengllayer" );
100     set_callbacks( CreateVout, DestroyVout );
101 vlc_module_end();
102
103 @interface VLCVoutLayer : CAOpenGLLayer {
104     vout_thread_t * p_vout;
105 }
106 + (id)layerWithVout:(vout_thread_t*)_p_vout; 
107 @end
108
109 /*****************************************************************************
110  * vout_sys_t: video output method descriptor
111  *****************************************************************************
112  * This structure is part of the video output thread descriptor.
113  * It describes the OpenGL specific properties of the output thread.
114  *****************************************************************************/
115 struct vout_sys_t
116 {
117     vout_thread_t * p_vout;
118
119     uint8_t    *pp_buffer[2]; /* one last rendered, one to be rendered */
120     int         i_index;
121     bool  b_frame_available;
122     
123     CGLContextObj glContext;
124
125     int         i_tex_width;
126     int         i_tex_height;
127     GLuint      p_textures[2];
128
129     NSAutoreleasePool *autorealease_pool;
130     VLCVoutLayer * o_layer;
131     id          o_cocoa_container;
132 };
133
134 /*****************************************************************************
135  * CreateVout: This function allocates and initializes the OpenGL vout method.
136  *****************************************************************************/
137 static int CreateVout( vlc_object_t *p_this )
138 {
139     vout_thread_t *p_vout = (vout_thread_t *)p_this;
140     vout_sys_t *p_sys;
141     char * psz;
142
143     /* Allocate structure */
144     p_vout->p_sys = p_sys = calloc( sizeof( vout_sys_t ), 1 );
145     if( p_sys == NULL )
146     {
147         msg_Err( p_vout, "out of memory" );
148         return VLC_EGENERIC;
149     }
150
151     p_sys->i_tex_width  = p_vout->fmt_in.i_width;
152     p_sys->i_tex_height = p_vout->fmt_in.i_height;
153
154     msg_Dbg( p_vout, "Texture size: %dx%d", p_sys->i_tex_width,
155              p_sys->i_tex_height );
156
157     p_vout->pf_init = Init;
158     p_vout->pf_end = End;
159     p_vout->pf_manage = Manage;
160     p_vout->pf_render = Render;
161     p_vout->pf_display = DisplayVideo;
162     p_vout->pf_control = Control;
163
164     return VLC_SUCCESS;
165 }
166
167 /*****************************************************************************
168  * Init: initialize the OpenGL video thread output method
169  *****************************************************************************/
170 static int Init( vout_thread_t *p_vout )
171 {
172     vout_sys_t *p_sys = p_vout->p_sys;
173     int i_pixel_pitch;
174     vlc_value_t val;
175
176 #if ( defined( WORDS_BIGENDIAN ) && VLCGL_FORMAT == GL_YCBCR_422_APPLE ) || (VLCGL_FORMAT == YCBCR_MESA)
177     p_vout->output.i_chroma = VLC_FOURCC('Y','U','Y','2');
178     i_pixel_pitch = 2;
179 #elif (VLCGL_FORMAT == GL_YCBCR_422_APPLE)
180     p_vout->output.i_chroma = VLC_FOURCC('U','Y','V','Y');
181     i_pixel_pitch = 2;
182 #endif
183
184     /* Since OpenGL can do rescaling for us, stick to the default
185      * coordinates and aspect. */
186     p_vout->output.i_width  = p_vout->render.i_width;
187     p_vout->output.i_height = p_vout->render.i_height;
188     p_vout->output.i_aspect = p_vout->render.i_aspect;
189
190     /* We do need a drawable to work properly */
191     vlc_value_t value_drawable;
192     var_Create( p_vout, "drawable", VLC_VAR_DOINHERIT );
193     var_Get( p_vout, "drawable", &value_drawable );
194
195     p_vout->p_sys->o_cocoa_container = (id) value_drawable.i_int;
196     
197     p_vout->fmt_out = p_vout->fmt_in;
198     p_vout->fmt_out.i_chroma = p_vout->output.i_chroma;
199
200     /* We know the chroma, allocate two buffer which will be used
201      * directly by the decoder */
202     int i;
203     for( i = 0; i < 2; i++ )
204     {
205         p_sys->pp_buffer[i] =
206             malloc( p_sys->i_tex_width * p_sys->i_tex_height * i_pixel_pitch );
207         if( !p_sys->pp_buffer[i] )
208         {
209             msg_Err( p_vout, "out of memory" );
210             return VLC_EGENERIC;
211         }
212     }
213     p_sys->b_frame_available = false;
214     p_sys->i_index = 0;
215
216     p_vout->p_picture[0].i_planes = 1;
217     p_vout->p_picture[0].p->p_pixels = p_sys->pp_buffer[p_sys->i_index];
218     p_vout->p_picture[0].p->i_lines = p_vout->output.i_height;
219     p_vout->p_picture[0].p->i_visible_lines = p_vout->output.i_height;
220     p_vout->p_picture[0].p->i_pixel_pitch = i_pixel_pitch;
221     p_vout->p_picture[0].p->i_pitch = p_vout->output.i_width *
222         p_vout->p_picture[0].p->i_pixel_pitch;
223     p_vout->p_picture[0].p->i_visible_pitch = p_vout->output.i_width *
224         p_vout->p_picture[0].p->i_pixel_pitch;
225
226     p_vout->p_picture[0].i_status = DESTROYED_PICTURE;
227     p_vout->p_picture[0].i_type   = DIRECT_PICTURE;
228
229     PP_OUTPUTPICTURE[ 0 ] = &p_vout->p_picture[0];
230
231     I_OUTPUTPICTURES = 1;
232     p_sys->autorealease_pool = [[NSAutoreleasePool alloc] init];
233
234     [VLCVoutLayer performSelectorOnMainThread:@selector(autoinitInVout:)
235                              withObject:[NSValue valueWithPointer:p_vout]
236                              waitUntilDone:YES];
237
238     return 0;
239 }
240
241 /*****************************************************************************
242  * End: terminate GLX video thread output method
243  *****************************************************************************/
244 static void End( vout_thread_t *p_vout )
245 {
246     vout_sys_t *p_sys = p_vout->p_sys;
247
248     p_vout->p_sys->b_frame_available = false;
249
250     [p_vout->p_sys->o_cocoa_container performSelectorOnMainThread:@selector(removeVoutLayer:) withObject:p_vout->p_sys->o_layer waitUntilDone:YES];
251
252     // Should be done automatically
253     [p_sys->o_layer release];
254     [p_sys->autorealease_pool release];
255
256     /* Free the texture buffer*/
257     free( p_sys->pp_buffer[0] );
258     free( p_sys->pp_buffer[1] );
259 }
260
261 /*****************************************************************************
262  * Destroy: destroy GLX video thread output method
263  *****************************************************************************
264  * Terminate an output method created by CreateVout
265  *****************************************************************************/
266 static void DestroyVout( vlc_object_t *p_this )
267 {
268     vout_thread_t *p_vout = (vout_thread_t *)p_this;
269     vout_sys_t *p_sys = p_vout->p_sys;
270     free( p_sys );
271 }
272
273 /*****************************************************************************
274  * Manage: handle Sys events
275  *****************************************************************************
276  * This function should be called regularly by video output thread. It returns
277  * a non null value if an error occurred.
278  *****************************************************************************/
279 static int Manage( vout_thread_t *p_vout )
280 {
281     vout_sys_t *p_sys = p_vout->p_sys;
282
283     return VLC_SUCCESS;
284 }
285
286 /*****************************************************************************
287  * Render: render previously calculated output
288  *****************************************************************************/
289 static void Render( vout_thread_t *p_vout, picture_t *p_pic )
290 {
291     vout_sys_t *p_sys = p_vout->p_sys;
292
293     @synchronized( p_sys->o_layer ) /* Make sure the p_sys->glContext isn't edited */
294     {
295         if( p_sys->glContext )
296         {
297             CGLLockContext(p_sys->glContext);
298             CGLSetCurrentContext(p_sys->glContext);
299             int i_new_index;
300             i_new_index = ( p_sys->i_index + 1 ) & 1;
301
302
303             /* Update the texture */
304             glBindTexture( VLCGL_TARGET, p_sys->p_textures[i_new_index] );
305             glTexSubImage2D( VLCGL_TARGET, 0, 0, 0,
306                          p_vout->fmt_out.i_width,
307                          p_vout->fmt_out.i_height,
308                          VLCGL_FORMAT, VLCGL_TYPE, p_sys->pp_buffer[i_new_index] );
309
310             /* Bind to the previous texture for drawing */
311             glBindTexture( VLCGL_TARGET, p_sys->p_textures[p_sys->i_index] );
312
313             /* Switch buffers */
314             p_sys->i_index = i_new_index;
315             p_pic->p->p_pixels = p_sys->pp_buffer[p_sys->i_index];
316             CGLUnlockContext(p_sys->glContext);
317             
318             p_sys->b_frame_available = true;
319         }
320     }
321
322     /* Give a buffer where the image will be rendered */
323     p_pic->p->p_pixels = p_sys->pp_buffer[p_sys->i_index];
324 }
325
326 /*****************************************************************************
327  * DisplayVideo: displays previously rendered output
328  *****************************************************************************/
329 static void DisplayVideo( vout_thread_t *p_vout, picture_t *p_pic )
330 {
331     vout_sys_t *p_sys = p_vout->p_sys;
332     
333     [p_sys->o_layer performSelectorOnMainThread:@selector(display)
334                     withObject:nil waitUntilDone:YES];
335 }
336
337 /*****************************************************************************
338  * Control: control facility for the vout
339  *****************************************************************************/
340 static int Control( vout_thread_t *p_vout, int i_query, va_list args )
341 {
342     vout_sys_t *p_sys = p_vout->p_sys;
343
344     switch( i_query )
345     {
346     case VOUT_SNAPSHOT:
347         return vout_vaControlDefault( p_vout, i_query, args );
348
349     default:
350         if( p_sys->p_vout->pf_control )
351             return p_sys->p_vout->pf_control( p_sys->p_vout, i_query, args );
352         else
353             return vout_vaControlDefault( p_vout, i_query, args );
354     }
355 }
356
357 /*****************************************************************************
358  * InitTextures
359  *****************************************************************************/
360 static int InitTextures( vout_thread_t *p_vout )
361 {
362     vout_sys_t *p_sys = p_vout->p_sys;
363     int i_index;
364
365     glDeleteTextures( 2, p_sys->p_textures );
366     glGenTextures( 2, p_sys->p_textures );
367
368     for( i_index = 0; i_index < 2; i_index++ )
369     {
370         glBindTexture( VLCGL_TARGET, p_sys->p_textures[i_index] );
371
372         /* Set the texture parameters */
373         glTexParameterf( VLCGL_TARGET, GL_TEXTURE_PRIORITY, 1.0 );
374
375         glTexParameteri( VLCGL_TARGET, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE );
376         glTexParameteri( VLCGL_TARGET, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE );
377
378         glTexParameteri( VLCGL_TARGET, GL_TEXTURE_MAG_FILTER, GL_LINEAR );
379         glTexParameteri( VLCGL_TARGET, GL_TEXTURE_MIN_FILTER, GL_LINEAR );
380
381         glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE );
382
383         /* Note: It seems that we can't bypass those, and even
384          * disabled they are used. They are the cause of the flickering */
385
386         /* Tell the driver not to make a copy of the texture but to use
387            our buffer */
388         glEnable( GL_UNPACK_CLIENT_STORAGE_APPLE );
389         glPixelStorei( GL_UNPACK_CLIENT_STORAGE_APPLE, GL_TRUE );
390
391         /* Use AGP texturing */
392         glTexParameteri( VLCGL_TARGET, GL_TEXTURE_STORAGE_HINT_APPLE, GL_STORAGE_SHARED_APPLE );
393
394         /* Call glTexImage2D only once, and use glTexSubImage2D later */
395         glTexImage2D( VLCGL_TARGET, 0, 4, p_sys->i_tex_width,
396                       p_sys->i_tex_height, 0, VLCGL_FORMAT, VLCGL_TYPE,
397                       p_sys->pp_buffer[i_index] );
398     }
399
400     return 0;
401 }
402
403 /*****************************************************************************
404  * @implementation VLCVoutLayer
405  */
406 @implementation VLCVoutLayer
407
408 /*****************************************************************************
409  * autoinitInVout: Called from the video thread to create a layer.
410  * The created layer is stored in the p_vout. We do that way because, cocoa
411  * doesn't support layer creation on non-main thread.
412  *****************************************************************************/
413 + (void)autoinitInVout:(NSValue*)arg
414 {
415     vout_thread_t * p_vout = [arg pointerValue];
416     p_vout->p_sys->o_layer = [[VLCVoutLayer layerWithVout:p_vout] retain];
417     [p_vout->p_sys->o_cocoa_container addVoutLayer:p_vout->p_sys->o_layer];
418 }
419
420 + (id)layerWithVout:(vout_thread_t*)_p_vout 
421 {
422     VLCVoutLayer* me = [[[self alloc] init] autorelease];
423     if( me )
424     {
425         me->p_vout = _p_vout;
426         me.asynchronous = NO;
427         me.bounds = CGRectMake( 0.0, 0.0, 
428                                 (float)_p_vout->fmt_in.i_visible_width * _p_vout->fmt_in.i_sar_num,
429                                 (float)_p_vout->fmt_in.i_visible_height * _p_vout->fmt_in.i_sar_den );
430     }
431     return me;
432 }
433
434 - (BOOL)canDrawInCGLContext:(CGLContextObj)glContext pixelFormat:(CGLPixelFormatObj)pixelFormat forLayerTime:(CFTimeInterval)timeInterval displayTime:(const CVTimeStamp *)timeStamp
435 {
436     /* Only draw the frame if we have a frame that was previously rendered */
437         return p_vout->p_sys->b_frame_available; // Flag is cleared by drawInCGLContext:pixelFormat:forLayerTime:displayTime:
438 }
439
440 - (void)drawInCGLContext:(CGLContextObj)glContext pixelFormat:(CGLPixelFormatObj)pixelFormat forLayerTime:(CFTimeInterval)timeInterval displayTime:(const CVTimeStamp *)timeStamp
441 {
442     CGLLockContext( glContext );
443     CGLSetCurrentContext( glContext );
444
445     float f_width, f_height, f_x, f_y;
446
447     f_x = (float)p_vout->fmt_out.i_x_offset;
448     f_y = (float)p_vout->fmt_out.i_y_offset;
449     f_width = (float)p_vout->fmt_out.i_x_offset +
450               (float)p_vout->fmt_out.i_visible_width;
451     f_height = (float)p_vout->fmt_out.i_y_offset +
452                (float)p_vout->fmt_out.i_visible_height;
453
454     glClear( GL_COLOR_BUFFER_BIT );
455
456     glEnable( VLCGL_TARGET );
457     glBegin( GL_POLYGON );
458     glTexCoord2f( f_x, f_y ); glVertex2f( -1.0, 1.0 );
459     glTexCoord2f( f_width, f_y ); glVertex2f( 1.0, 1.0 );
460     glTexCoord2f( f_width, f_height ); glVertex2f( 1.0, -1.0 );
461     glTexCoord2f( f_x, f_height ); glVertex2f( -1.0, -1.0 );
462     glEnd();
463
464     glDisable( VLCGL_TARGET );
465
466     glFlush();
467
468     CGLUnlockContext( glContext );
469 }
470
471 - (CGLContextObj)copyCGLContextForPixelFormat:(CGLPixelFormatObj)pixelFormat
472 {
473     CGLContextObj context = [super copyCGLContextForPixelFormat:pixelFormat];
474
475     CGLLockContext( context );
476
477     CGLSetCurrentContext( context );
478
479     /* Swap buffers only during the vertical retrace of the monitor.
480     http://developer.apple.com/documentation/GraphicsImaging/
481     Conceptual/OpenGL/chap5/chapter_5_section_44.html */
482
483     GLint params = 1;
484     CGLSetParameter( CGLGetCurrentContext(), kCGLCPSwapInterval,
485                      &params );
486
487     InitTextures( p_vout );
488
489     glDisable( GL_BLEND );
490     glDisable( GL_DEPTH_TEST );
491     glDepthMask( GL_FALSE );
492     glDisable( GL_CULL_FACE) ;
493     glClearColor( 0.0f, 0.0f, 0.0f, 1.0f );
494     glClear( GL_COLOR_BUFFER_BIT );
495
496     CGLUnlockContext( context );
497     @synchronized( self )
498     {
499         p_vout->p_sys->glContext = context;
500     }
501
502     return context;
503 }
504
505 - (void)releaseCGLContext:(CGLContextObj)glContext
506 {
507     @synchronized( self )
508     {
509         p_vout->p_sys->glContext = nil;
510     }
511
512     CGLLockContext( glContext );
513     CGLSetCurrentContext( glContext );
514
515     glDeleteTextures( 2, p_vout->p_sys->p_textures );
516
517     CGLUnlockContext( glContext );
518 }
519 @end
520
521 #endif