Fix compilation after FFmpeg r22565.
[mplayer/glamo.git] / libvo / vo_xvmc.c
blobfb0266e081df0e8dc587df861c3fd3e7b6a5d026
1 /*
2 * This file is part of MPlayer.
4 * MPlayer is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
9 * MPlayer is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License along
15 * with MPlayer; if not, write to the Free Software Foundation, Inc.,
16 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 #include <stdlib.h>
20 #include <stdio.h>
21 #include <string.h>
22 #include <unistd.h>
24 #include "config.h"
25 #include "mp_msg.h"
26 #include "video_out.h"
27 #include "video_out_internal.h"
28 #include "osdep/timer.h"
30 #include <X11/Xlib.h>
31 #include <X11/Xutil.h>
32 #include <X11/Xatom.h>
34 #ifdef HAVE_SHM
35 #include <sys/ipc.h>
36 #include <sys/shm.h>
37 #include <X11/extensions/XShm.h>
38 #endif
40 #include <X11/extensions/Xv.h>
41 #include <X11/extensions/Xvlib.h>
42 #include <X11/extensions/XvMClib.h>
44 #include "x11_common.h"
45 #include "libavcodec/avcodec.h"
46 #include "libavcodec/xvmc.h"
48 #include "sub.h"
49 #include "aspect.h"
51 #include "subopt-helper.h"
52 #include "gui/interface.h"
54 #include "libavutil/common.h"
56 //no chance for xinerama to be supported in the near future
57 #undef CONFIG_XINERAMA
59 #undef NDEBUG
60 #include <assert.h>
63 #define UNUSED(x) ((void)(x))
65 #if LIBAVCODEC_BUILD < ((51<<16)+(40<<8)+2)
66 #error You need at least libavcodec v51.40.2
67 #endif
70 static int benchmark;
71 static int use_sleep;
72 static int first_frame;//draw colorkey on first frame
73 static int use_queue;
74 static int xv_port_request = 0;
75 static int xv_adaptor = -1;
76 static int bob_deinterlace;
77 static int top_field_first;
79 static int image_width,image_height;
80 static int image_format;
82 #define NO_SUBPICTURE 0
83 #define OVERLAY_SUBPICTURE 1
84 #define BLEND_SUBPICTURE 2
85 #define BACKEND_SUBPICTURE 3
87 static int subpicture_mode;
88 static int subpicture_alloc;
89 static XvMCSubpicture subpicture;
90 static XvImageFormatValues subpicture_info;
91 static int subpicture_clear_color;//transparent color for the subpicture or color key for overlay
93 static XvMCSurfaceInfo surface_info;
94 static XvMCContext ctx;
95 static XvMCBlockArray data_blocks;
96 static XvMCMacroBlockArray mv_blocks;
98 #define MAX_SURFACES 8
99 static int number_of_surfaces=0;
100 static XvMCSurface surface_array[MAX_SURFACES];
101 static struct xvmc_pix_fmt *surface_render;
103 static struct xvmc_pix_fmt *p_render_surface_to_show = NULL;
104 static struct xvmc_pix_fmt *p_render_surface_visible = NULL;
106 //display queue, kinda render ahead
107 static struct xvmc_pix_fmt *show_queue[MAX_SURFACES];
108 static int free_element;
111 static void (*draw_osd_fnc)(int x0,int y0, int w,int h, unsigned char* src, unsigned char *srca, int stride);
112 static void (*clear_osd_fnc)(int x0,int y0, int w,int h);
113 static void (*init_osd_fnc)(void);
115 static void draw_osd_AI44(int x0,int y0, int w,int h, unsigned char* src, unsigned char *srca, int stride);
116 static void draw_osd_IA44(int x0,int y0, int w,int h, unsigned char* src, unsigned char *srca, int stride);
117 static void clear_osd_subpic(int x0,int y0, int w,int h);
118 static void init_osd_yuv_pal(void);
121 static const struct{
122 int id;//id as xvimages or as mplayer RGB|{8,15,16,24,32}
123 void (* init_func_ptr)();
124 void (* draw_func_ptr)();
125 void (* clear_func_ptr)();
126 } osd_render[]={
127 {0x34344149,init_osd_yuv_pal,draw_osd_AI44,clear_osd_subpic},
128 {0x34344941,init_osd_yuv_pal,draw_osd_IA44,clear_osd_subpic},
129 {0,NULL,NULL,NULL}
132 static void xvmc_free(void);
133 static void xvmc_clean_surfaces(void);
134 static int count_free_surfaces(void);
135 static struct xvmc_pix_fmt *find_free_surface(void);
137 static const vo_info_t info = {
138 "XVideo Motion Compensation",
139 "xvmc",
140 "Ivan Kalvachev <iive@users.sf.net>",
144 const LIBVO_EXTERN(xvmc);
146 //shm stuff from vo_xv
147 #ifdef HAVE_SHM
148 static XShmSegmentInfo Shminfo;
149 static int Shmem_Flag;
150 #endif
151 XvImage * xvimage;
154 static void allocate_xvimage(int xvimage_width,int xvimage_height,int xv_format)
157 * allocate XvImages. FIXME: no error checking, without
158 * mit-shm this will bomb... trzing to fix ::atmos
160 #ifdef HAVE_SHM
161 if ( mLocalDisplay && XShmQueryExtension( mDisplay ) ) Shmem_Flag = 1;
162 else
164 Shmem_Flag = 0;
165 mp_msg(MSGT_VO,MSGL_WARN, "Shared memory not supported\nReverting to normal Xv\n" );
167 if ( Shmem_Flag )
169 xvimage = (XvImage *) XvShmCreateImage(mDisplay, xv_port, xv_format,
170 NULL, xvimage_width, xvimage_height, &Shminfo);
172 Shminfo.shmid = shmget(IPC_PRIVATE, xvimage->data_size, IPC_CREAT | 0777);
173 Shminfo.shmaddr = (char *) shmat(Shminfo.shmid, 0, 0);
174 Shminfo.readOnly = False;
176 xvimage->data = Shminfo.shmaddr;
177 XShmAttach(mDisplay, &Shminfo);
178 XSync(mDisplay, False);
179 shmctl(Shminfo.shmid, IPC_RMID, 0);
181 else
182 #endif
184 xvimage = (XvImage *) XvCreateImage(mDisplay, xv_port, xv_format, NULL, xvimage_width, xvimage_height);
185 xvimage->data = malloc(xvimage->data_size);
186 XSync(mDisplay,False);
188 // memset(xvimage->data,128,xvimage->data_size);
189 return;
192 static void deallocate_xvimage(void)
194 #ifdef HAVE_SHM
195 if ( Shmem_Flag )
197 XShmDetach( mDisplay,&Shminfo );
198 shmdt( Shminfo.shmaddr );
200 else
201 #endif
203 free(xvimage->data);
205 XFree(xvimage);
207 XSync(mDisplay, False);
208 return;
210 //end of vo_xv shm/xvimage code
212 static int xvmc_check_surface_format(uint32_t format, XvMCSurfaceInfo * surf_info){
213 if ( format == IMGFMT_XVMC_IDCT_MPEG2 ){
214 if( surf_info->mc_type != (XVMC_IDCT|XVMC_MPEG_2) ) return -1;
215 if( surf_info->chroma_format != XVMC_CHROMA_FORMAT_420 ) return -1;
216 return 0;
218 if ( format == IMGFMT_XVMC_MOCO_MPEG2 ){
219 if(surf_info->mc_type != XVMC_MPEG_2) return -1;
220 if(surf_info->chroma_format != XVMC_CHROMA_FORMAT_420) return -1;
221 return 0;
223 return -1;//fail
226 //print all info needed to add new format
227 static void print_xvimage_format_values(XvImageFormatValues *xifv){
228 int i;
229 const int msgl=MSGL_DBG4;
231 mp_msg(MSGT_VO,msgl,"Format_ID = 0x%X\n",xifv->id);
233 mp_msg(MSGT_VO,msgl," type = ");
234 if(xifv->type == XvRGB) mp_msg(MSGT_VO,msgl,"RGB\n");
235 else if(xifv->type == XvYUV) mp_msg(MSGT_VO,msgl,"YUV\n");
236 else mp_msg(MSGT_VO,msgl,"Unknown\n");
238 mp_msg(MSGT_VO,msgl," byte_order = ");
239 if(xifv->byte_order == LSBFirst) mp_msg(MSGT_VO,msgl,"LSB First\n");
240 else if(xifv->type == MSBFirst) mp_msg(MSGT_VO,msgl,"MSB First\n");
241 else mp_msg(MSGT_VO,msgl,"Unknown\n");//yes Linux support other types too
243 mp_msg(MSGT_VO,msgl," guid = ");
244 for(i=0;i<16;i++)
245 mp_msg(MSGT_VO,msgl,"%02X ",(unsigned char)xifv->guid[i]);
246 mp_msg(MSGT_VO,msgl,"\n");
248 mp_msg(MSGT_VO,msgl," bits_per_pixel = %d\n",xifv->bits_per_pixel);
250 mp_msg(MSGT_VO,msgl," format = ");
251 if(xifv->format == XvPacked) mp_msg(MSGT_VO,msgl,"XvPacked\n");
252 else if(xifv->format == XvPlanar) mp_msg(MSGT_VO,msgl,"XvPlanar\n");
253 else mp_msg(MSGT_VO,msgl,"Unknown\n");
255 mp_msg(MSGT_VO,msgl," num_planes = %d\n",xifv->num_planes);
257 if(xifv->type == XvRGB){
258 mp_msg(MSGT_VO,msgl," red_mask = %0X\n", xifv->red_mask);
259 mp_msg(MSGT_VO,msgl," green_mask = %0X\n",xifv->green_mask);
260 mp_msg(MSGT_VO,msgl," blue_mask = %0X\n", xifv->blue_mask);
262 if(xifv->type == XvYUV){
263 mp_msg(MSGT_VO,msgl," y_sample_bits = %d\n u_sample_bits = %d\n v_sample_bits = %d\n",
264 xifv->y_sample_bits,xifv->u_sample_bits,xifv->v_sample_bits);
265 mp_msg(MSGT_VO,msgl," horz_y_period = %d\n horz_u_period = %d\n horz_v_period = %d\n",
266 xifv->horz_y_period,xifv->horz_u_period,xifv->horz_v_period);
267 mp_msg(MSGT_VO,msgl," vert_y_period = %d\n vert_u_period = %d\n vert_v_period = %d\n",
268 xifv->vert_y_period,xifv->vert_u_period,xifv->vert_v_period);
270 mp_msg(MSGT_VO,msgl," component_order = ");
271 for(i=0;i<32;i++)
272 if(xifv->component_order[i]>=32)
273 mp_msg(MSGT_VO,msgl,"%c",xifv->component_order[i]);
274 mp_msg(MSGT_VO,msgl,"\n");
276 mp_msg(MSGT_VO,msgl," scanline = ");
277 if(xifv->scanline_order == XvTopToBottom) mp_msg(MSGT_VO,msgl,"XvTopToBottom\n");
278 else if(xifv->scanline_order == XvBottomToTop) mp_msg(MSGT_VO,msgl,"XvBottomToTop\n");
279 else mp_msg(MSGT_VO,msgl,"Unknown\n");
281 mp_msg(MSGT_VO,msgl,"\n");
284 // WARNING This function may changes xv_port and surface_info!
285 static int xvmc_find_surface_by_format(int format,int width,int height,
286 XvMCSurfaceInfo * surf_info,int query){
287 int rez;
288 XvAdaptorInfo * ai;
289 int num_adaptors,i;
290 unsigned long p;
291 int s,mc_surf_num;
292 XvMCSurfaceInfo * mc_surf_list;
294 rez = XvQueryAdaptors(mDisplay,DefaultRootWindow(mDisplay),&num_adaptors,&ai);
295 if( rez != Success ) return -1;
296 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: Querying %d adaptors\n",num_adaptors);
297 for(i=0; i<num_adaptors; i++)
299 /* check if adaptor number has been specified */
300 if (xv_adaptor != -1 && xv_adaptor != i)
301 continue;
302 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: Quering adaptor #%d\n",i);
303 if( ai[i].type == 0 ) continue;// we need at least dummy type!
304 //probing ports
305 for(p=ai[i].base_id; p<ai[i].base_id+ai[i].num_ports; p++)
307 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: probing port #%ld\n",p);
308 mc_surf_list = XvMCListSurfaceTypes(mDisplay,p,&mc_surf_num);
309 if( mc_surf_list == NULL || mc_surf_num == 0){
310 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: No XvMC supported. \n");
311 continue;
313 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: XvMC list have %d surfaces\n",mc_surf_num);
314 //we have XvMC list!
315 for(s=0; s<mc_surf_num; s++)
317 if( width > mc_surf_list[s].max_width ) continue;
318 if( height > mc_surf_list[s].max_height ) continue;
319 if( xvmc_check_surface_format(format,&mc_surf_list[s])<0 ) continue;
320 //we have match!
321 /* respect the users wish */
322 if ( xv_port_request != 0 && xv_port_request != p )
324 continue;
327 if(!query){
328 rez = XvGrabPort(mDisplay,p,CurrentTime);
329 if(rez != Success){
330 mp_msg(MSGT_VO,MSGL_DBG3,"vo_xvmc: Fail to grab port %ld\n",p);
331 continue;
333 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Using Xv Adaptor #%d (%s)\n", i, ai[i].name);
334 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Port %ld grabed\n",p);
335 xv_port = p;
337 goto surface_found;
338 }//for mc surf
339 XFree(mc_surf_list);//if mc_surf_num==0 is list==NULL ?
340 }//for ports
341 }//for adaptors
342 XvFreeAdaptorInfo(ai);
344 if(!query) mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: Could not find free matching surface. Sorry.\n");
345 return 0;
347 // somebody know cleaner way to escape from 3 internal loops?
348 surface_found:
349 XvFreeAdaptorInfo(ai);
351 memcpy(surf_info,&mc_surf_list[s],sizeof(XvMCSurfaceInfo));
352 mp_msg(MSGT_VO, query?MSGL_INFO:MSGL_DBG3,
353 "vo_xvmc: Found matching surface with id=%X on %ld port at %d adapter\n",
354 mc_surf_list[s].surface_type_id,p,i);
355 return mc_surf_list[s].surface_type_id;
358 static uint32_t xvmc_draw_image(mp_image_t *mpi){
359 struct xvmc_pix_fmt *rndr;
361 assert(mpi!=NULL);
362 assert(mpi->flags &MP_IMGFLAG_DIRECT);
363 // assert(mpi->flags &MP_IMGFLAGS_DRAWBACK);
365 rndr = (struct xvmc_pix_fmt*)mpi->priv; //there is copy in plane[2]
366 assert( rndr != NULL );
367 assert( rndr->xvmc_id == AV_XVMC_ID );
368 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: draw_image(show rndr=%p)\n",rndr);
369 // the surface have passed vf system without been skiped, it will be displayed
370 rndr->state |= AV_XVMC_STATE_DISPLAY_PENDING;
371 p_render_surface_to_show = rndr;
372 top_field_first = mpi->fields & MP_IMGFIELD_TOP_FIRST;
373 return VO_TRUE;
376 static int preinit(const char *arg){
377 int xv_version,xv_release,xv_request_base,xv_event_base,xv_error_base;
378 int mc_eventBase,mc_errorBase;
379 int mc_ver,mc_rev;
380 strarg_t ck_src_arg = { 0, NULL };
381 strarg_t ck_method_arg = { 0, NULL };
382 const opt_t subopts [] =
384 /* name arg type arg var test */
385 { "port", OPT_ARG_INT, &xv_port_request, int_pos },
386 { "adaptor", OPT_ARG_INT, &xv_adaptor, int_non_neg },
387 { "ck", OPT_ARG_STR, &ck_src_arg, xv_test_ck },
388 { "ck-method", OPT_ARG_STR, &ck_method_arg, xv_test_ckm },
389 { "benchmark", OPT_ARG_BOOL, &benchmark, NULL },
390 { "sleep", OPT_ARG_BOOL, &use_sleep, NULL },
391 { "queue", OPT_ARG_BOOL, &use_queue, NULL },
392 { "bobdeint", OPT_ARG_BOOL, &bob_deinterlace, NULL },
393 { NULL }
396 //Obtain display handler
397 if (!vo_init()) return -1;//vo_xv
399 //XvMC is subdivision of XVideo
400 if (Success != XvQueryExtension(mDisplay,&xv_version,&xv_release,&xv_request_base,
401 &xv_event_base,&xv_error_base) ){
402 mp_msg(MSGT_VO,MSGL_ERR,"Sorry, Xv(MC) not supported by this X11 version/driver\n");
403 mp_msg(MSGT_VO,MSGL_ERR,"********** Try with -vo x11 or -vo sdl ***********\n");
404 return -1;
406 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: X-Video extension %d.%d\n",xv_version,xv_release);
408 if( True != XvMCQueryExtension(mDisplay,&mc_eventBase,&mc_errorBase) ){
409 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: No X-Video MotionCompensation Extension on %s\n",
410 XDisplayName(NULL));
411 return -1;
414 if(Success == XvMCQueryVersion(mDisplay, &mc_ver, &mc_rev) ){
415 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: X-Video MotionCompensation Extension version %i.%i\n",
416 mc_ver,mc_rev);
418 else{
419 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: Error querying version info!\n");
420 return -1;
422 surface_render = NULL;
423 xv_port = 0;
424 number_of_surfaces = 0;
425 subpicture_alloc = 0;
427 benchmark = 0; //disable PutImageto allow faster display than screen refresh
428 use_sleep = 0;
429 use_queue = 0;
430 bob_deinterlace = 0;
432 /* parse suboptions */
433 if ( subopt_parse( arg, subopts ) != 0 )
435 return -1;
438 xv_setup_colorkeyhandling( ck_method_arg.str, ck_src_arg.str );
440 return 0;
443 static int config(uint32_t width, uint32_t height,
444 uint32_t d_width, uint32_t d_height,
445 uint32_t flags, char *title, uint32_t format){
446 int i,mode_id,rez;
447 int numblocks,blocks_per_macroblock;//bpmb we have 6,8,12
449 //from vo_xv
450 XVisualInfo vinfo;
451 XSetWindowAttributes xswa;
452 XWindowAttributes attribs;
453 unsigned long xswamask;
454 int depth;
455 #ifdef CONFIG_XF86VM
456 int vm = flags & VOFLAG_MODESWITCHING;
457 #endif
458 //end of vo_xv
460 if( !IMGFMT_IS_XVMC(format) )
462 assert(0);//should never happen, abort on debug or
463 return 1;//return error on relese
466 // Find free port that supports MC, by querying adaptors
467 if( xv_port != 0 || number_of_surfaces != 0 ){
468 if( height==image_height && width==image_width && image_format==format){
469 xvmc_clean_surfaces();
470 goto skip_surface_allocation;
472 xvmc_free();
474 numblocks=((width+15)/16)*((height+15)/16);
475 // Find Supported Surface Type
476 mode_id = xvmc_find_surface_by_format(format,width,height,&surface_info,0);//false=1 to grab port, not query
477 if ( mode_id == 0 )
479 return -1;
482 rez = XvMCCreateContext(mDisplay, xv_port,mode_id,width,height,XVMC_DIRECT,&ctx);
483 if( rez != Success ){
484 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: XvMCCreateContext failed with error %d\n",rez);
485 return -1;
487 if( ctx.flags & XVMC_DIRECT ){
488 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Allocated Direct Context\n");
489 }else{
490 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Allocated Indirect Context!\n");
494 blocks_per_macroblock = 6;
495 if(surface_info.chroma_format == XVMC_CHROMA_FORMAT_422)
496 blocks_per_macroblock = 8;
497 if(surface_info.chroma_format == XVMC_CHROMA_FORMAT_444)
498 blocks_per_macroblock = 12;
500 rez = XvMCCreateBlocks(mDisplay,&ctx,numblocks*blocks_per_macroblock,&data_blocks);
501 if( rez != Success ){
502 XvMCDestroyContext(mDisplay,&ctx);
503 return -1;
505 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: data_blocks allocated\n");
507 rez = XvMCCreateMacroBlocks(mDisplay,&ctx,numblocks,&mv_blocks);
508 if( rez != Success ){
509 XvMCDestroyBlocks(mDisplay,&data_blocks);
510 XvMCDestroyContext(mDisplay,&ctx);
511 return -1;
513 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: mv_blocks allocated\n");
515 if(surface_render==NULL)
516 surface_render = malloc(MAX_SURFACES * sizeof(struct xvmc_pix_fmt)); //easy mem debug
517 memset(surface_render, 0, MAX_SURFACES * sizeof(struct xvmc_pix_fmt));
519 for(i=0; i<MAX_SURFACES; i++){
520 rez=XvMCCreateSurface(mDisplay,&ctx,&surface_array[i]);
521 if( rez != Success )
522 break;
523 surface_render[i].xvmc_id = AV_XVMC_ID;
524 surface_render[i].data_blocks = data_blocks.blocks;
525 surface_render[i].mv_blocks = mv_blocks.macro_blocks;
526 surface_render[i].allocated_mv_blocks = numblocks;
527 surface_render[i].allocated_data_blocks = numblocks*blocks_per_macroblock;
528 surface_render[i].idct = (surface_info.mc_type & XVMC_IDCT) == XVMC_IDCT;
529 surface_render[i].unsigned_intra = (surface_info.flags & XVMC_INTRA_UNSIGNED) == XVMC_INTRA_UNSIGNED;
530 surface_render[i].p_surface = &surface_array[i];
531 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: surface[%d] = %p .rndr=%p\n",
532 i,&surface_array[i], &surface_render[i]);
534 number_of_surfaces = i;
535 if( number_of_surfaces < 4 ){// +2 I or P and +2 for B (to avoid visible motion drawing)
536 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: Unable to allocate at least 4 Surfaces\n");
537 uninit();
538 return -1;
540 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Motion Compensation context allocated - %d surfaces\n",
541 number_of_surfaces);
543 //debug
544 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: idct=%d unsigned_intra=%d\n",
545 (surface_info.mc_type & XVMC_IDCT) == XVMC_IDCT,
546 (surface_info.flags & XVMC_INTRA_UNSIGNED) == XVMC_INTRA_UNSIGNED);
548 // Find way to display OSD & subtitle
549 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: looking for OSD support\n");
550 subpicture_mode = NO_SUBPICTURE;
551 if(surface_info.flags & XVMC_OVERLAID_SURFACE)
552 subpicture_mode = OVERLAY_SUBPICTURE;
554 if(surface_info.subpicture_max_width != 0 &&
555 surface_info.subpicture_max_height != 0 ){
556 int s,k,num_subpic;
558 XvImageFormatValues * xvfmv;
559 xvfmv = XvMCListSubpictureTypes(mDisplay, xv_port,
560 surface_info.surface_type_id, &num_subpic);
562 if(num_subpic != 0 && xvfmv != NULL){
563 if( mp_msg_test(MSGT_VO,MSGL_DBG4) ){//Print all subpicture types for debug
564 for(s=0;s<num_subpic;s++)
565 print_xvimage_format_values(&xvfmv[s]);
568 for(s=0;s<num_subpic;s++){
569 for(k=0;osd_render[k].draw_func_ptr!=NULL;k++){
570 if(xvfmv[s].id == osd_render[k].id)
572 init_osd_fnc = osd_render[k].init_func_ptr;
573 draw_osd_fnc = osd_render[k].draw_func_ptr;
574 clear_osd_fnc = osd_render[k].clear_func_ptr;
576 subpicture_mode = BLEND_SUBPICTURE;
577 subpicture_info = xvfmv[s];
578 mp_msg(MSGT_VO,MSGL_INFO," Subpicture id 0x%08X\n",subpicture_info.id);
579 goto found_subpic;
583 found_subpic:
584 XFree(xvfmv);
586 //Blend2 supicture is always possible, blend1 only at backend
587 if( (subpicture_mode == BLEND_SUBPICTURE) &&
588 (surface_info.flags & XVMC_BACKEND_SUBPICTURE) )
590 subpicture_mode = BACKEND_SUBPICTURE;
595 switch(subpicture_mode){
596 case NO_SUBPICTURE:
597 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: No OSD support for this mode\n");
598 break;
599 case OVERLAY_SUBPICTURE:
600 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: OSD support via color key tricks\n");
601 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: not yet implemented:(\n");
602 break;
603 case BLEND_SUBPICTURE:
604 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: OSD support by additional frontend rendering\n");
605 break;
606 case BACKEND_SUBPICTURE:
607 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: OSD support by backend rendering (fast)\n");
608 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: Please send feedback to confirm that it works,otherwise send bugreport!\n");
609 break;
612 //take keycolor value and choose method for handling it
613 if ( !vo_xv_init_colorkey() )
615 return -1; // bail out, colorkey setup failed
618 vo_xv_enable_vsync();//it won't break anything
620 //taken from vo_xv
621 image_height = height;
622 image_width = width;
624 skip_surface_allocation:
626 #ifdef CONFIG_GUI
627 if(use_gui)
628 guiGetEvent( guiSetShVideo,0 ); // let the GUI to setup/resize our window
629 else
630 #endif
632 #ifdef CONFIG_XF86VM
633 if ( vm )
635 vo_vm_switch();
637 // else
638 #endif
639 XGetWindowAttributes(mDisplay, DefaultRootWindow(mDisplay), &attribs);
640 depth=attribs.depth;
641 if (depth != 15 && depth != 16 && depth != 24 && depth != 32) depth = 24;
642 XMatchVisualInfo(mDisplay, mScreen, depth, TrueColor, &vinfo);
644 xswa.background_pixel = 0;
645 if (xv_ck_info.method == CK_METHOD_BACKGROUND)
646 xswa.background_pixel = xv_colorkey;
647 xswa.border_pixel = 0;
648 xswamask = CWBackPixel | CWBorderPixel;
650 vo_x11_create_vo_window(&vinfo, vo_dx, vo_dy, d_width, d_height, flags,
651 CopyFromParent, "xvmc", title);
652 XChangeWindowAttributes(mDisplay, vo_window, xswamask, &xswa);
654 #ifdef CONFIG_XF86VM
655 if ( vm )
657 /* Grab the mouse pointer in our window */
658 if(vo_grabpointer)
659 XGrabPointer(mDisplay, vo_window, True, 0,
660 GrabModeAsync, GrabModeAsync,
661 vo_window, None, CurrentTime );
662 XSetInputFocus(mDisplay, vo_window, RevertToNone, CurrentTime);
664 #endif
667 //end vo_xv
669 /* store image dimesions for displaying */
670 p_render_surface_visible = NULL;
671 p_render_surface_to_show = NULL;
673 free_element = 0;
674 first_frame = 1;
676 image_format=format;
677 return 0;
680 static int draw_frame(uint8_t *srcp[]){
681 UNUSED(srcp);
682 assert(0);
685 static void init_osd_yuv_pal(void) {
686 char * palette;
687 int rez;
688 int i,j;
689 int snum,seb;
690 int Y,U,V;
692 subpicture_clear_color = 0;
694 if(subpicture.num_palette_entries > 0){
696 snum = subpicture.num_palette_entries;
697 seb = subpicture.entry_bytes;
698 palette = malloc(snum*seb);//check fail
699 if(palette == NULL) return;
700 for(i=0; i<snum; i++){
701 // 0-black max-white the other are gradients
702 Y = i*(1 << subpicture_info.y_sample_bits)/snum;//snum=2;->(0),(1*(1<<1)/2)
703 U = 1 << (subpicture_info.u_sample_bits - 1);
704 V = 1 << (subpicture_info.v_sample_bits - 1);
705 for(j=0; j<seb; j++)
706 switch(subpicture.component_order[j]){
707 case 'U': palette[i*seb+j] = U; break;
708 case 'V': palette[i*seb+j] = V; break;
709 case 'Y':
710 default:
711 palette[i*seb+j] = Y; break;
714 rez = XvMCSetSubpicturePalette(mDisplay, &subpicture, palette);
715 if(rez!=Success){
716 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: Setting palette failed.\n");
718 free(palette);
722 static void clear_osd_subpic(int x0, int y0, int w, int h){
723 int rez;
725 rez=XvMCClearSubpicture(mDisplay, &subpicture,
726 x0, y0, w,h,
727 subpicture_clear_color);
728 if(rez != Success)
729 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: XvMCClearSubpicture failed!\n");
732 static void OSD_init(void) {
733 unsigned short osd_height, osd_width;
734 int rez;
736 if(subpicture_alloc){
737 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: destroying subpicture\n");
738 XvMCDestroySubpicture(mDisplay,&subpicture);
739 deallocate_xvimage();
740 subpicture_alloc = 0;
743 /* if(surface_info.flags & XVMC_SUBPICTURE_INDEPENDENT_SCALING){
744 osd_width = vo_dwidth;
745 osd_height = vo_dheight;
746 }else*/
748 osd_width = image_width;
749 osd_height = image_height;
752 if(osd_width > surface_info.subpicture_max_width)
753 osd_width = surface_info.subpicture_max_width;
754 if(osd_height > surface_info.subpicture_max_height)
755 osd_height = surface_info.subpicture_max_height;
756 if(osd_width == 0 || osd_height == 0)
757 return;//if called before window size is known
759 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: creating subpicture (%d,%d) format %X\n",
760 osd_width,osd_height,subpicture_info.id);
762 rez = XvMCCreateSubpicture(mDisplay,&ctx,&subpicture,
763 osd_width,osd_height,subpicture_info.id);
764 if(rez != Success){
765 subpicture_mode = NO_SUBPICTURE;
766 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: Create Subpicture failed, OSD disabled\n");
767 return;
769 if( mp_msg_test(MSGT_VO,MSGL_DBG4) ){
770 int i;
771 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: Created Subpicture:\n");
772 mp_msg(MSGT_VO,MSGL_DBG4," xvimage_id=0x%X\n",subpicture.xvimage_id);
773 mp_msg(MSGT_VO,MSGL_DBG4," width=%d\n",subpicture.width);
774 mp_msg(MSGT_VO,MSGL_DBG4," height=%d\n",subpicture.height);
775 mp_msg(MSGT_VO,MSGL_DBG4," num_palette_entries=0x%X\n",subpicture.num_palette_entries);
776 mp_msg(MSGT_VO,MSGL_DBG4," entry_bytes=0x%X\n",subpicture.entry_bytes);
778 mp_msg(MSGT_VO,MSGL_DBG4," component_order=\"");
779 for(i=0; i<4; i++)
780 if(subpicture.component_order[i] >= 32)
781 mp_msg(MSGT_VO,MSGL_DBG4,"%c", subpicture.component_order[i]);
782 mp_msg(MSGT_VO,MSGL_DBG4,"\"\n");
785 //call init for the surface type
786 init_osd_fnc();//init palete,clear color etc ...
787 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: clearing subpicture\n");
788 clear_osd_fnc(0, 0, subpicture.width, subpicture.height);
790 allocate_xvimage(subpicture.width, subpicture.height, subpicture_info.id);
791 subpicture_alloc = 1;
794 static void draw_osd_IA44(int x0,int y0, int w,int h, unsigned char* src, unsigned char *srca, int stride){
795 int ox,oy;
796 int rez;
798 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc:composite AI44 subpicture (%d,%d - %d,%d)\n",x0,y0,w,h);
800 for(ox=0; ox<w; ox++){
801 for(oy=0; oy<h; oy++){
802 xvimage->data[oy*xvimage->width+ox] = (src[oy*stride+ox]>>4) | ((0-srca[oy*stride+ox])&0xf0);
805 rez = XvMCCompositeSubpicture(mDisplay, &subpicture, xvimage, 0, 0,
806 w,h,x0,y0);
807 if(rez != Success){
808 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: composite subpicture failed\n");
809 assert(0);
813 static void draw_osd_AI44(int x0,int y0, int w,int h, unsigned char* src, unsigned char *srca, int stride){
814 int ox,oy;
815 int rez;
817 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc:composite AI44 subpicture (%d,%d - %d,%d)\n",x0,y0,w,h);
819 for(ox=0; ox<w; ox++){
820 for(oy=0; oy<h; oy++){
821 xvimage->data[oy*xvimage->width+ox] = (src[oy*stride+ox]&0xf0) | (((0-srca[oy*stride+ox])>>4)&0xf);
824 rez = XvMCCompositeSubpicture(mDisplay, &subpicture, xvimage, 0, 0,
825 w,h,x0,y0);
826 if(rez != Success){
827 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: composite subpicture failed\n");
828 assert(0);
832 static void draw_osd(void){
833 struct xvmc_pix_fmt *osd_rndr;
834 int osd_has_changed;
835 int have_osd_to_draw;
836 int rez;
838 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: draw_osd ,OSD_mode=%d, surface_to_show=%p\n",
839 subpicture_mode,p_render_surface_to_show);
841 if(subpicture_mode == BLEND_SUBPICTURE ||
842 subpicture_mode == BACKEND_SUBPICTURE ){
844 if(!subpicture_alloc) //allocate subpicture when dimensions are known
845 OSD_init();
846 if(!subpicture_alloc)
847 return;//dimensions still unknown.
849 osd_has_changed = vo_update_osd(subpicture.width, subpicture.height);
850 have_osd_to_draw = vo_osd_check_range_update(0, 0, subpicture.width,
851 subpicture.height);
853 if(!have_osd_to_draw)
854 return;//nothing to draw,no subpic, no blend
856 if(osd_has_changed){
857 //vo_remove_text(subpicture.width, subpicture.height,clear_osd_fnc)
858 clear_osd_fnc(0,0,subpicture.width,subpicture.height);
859 vo_draw_text(subpicture.width, subpicture.height, draw_osd_fnc);
861 XvMCSyncSubpicture(mDisplay,&subpicture);//todo usleeep wait!
863 if(subpicture_mode == BLEND_SUBPICTURE){
864 osd_rndr = find_free_surface();
865 if(osd_rndr == NULL)
866 return;// no free surface to draw OSD in
868 rez = XvMCBlendSubpicture2(mDisplay,
869 p_render_surface_to_show->p_surface, osd_rndr->p_surface,
870 &subpicture,
871 0, 0, subpicture.width, subpicture.height,
872 0, 0, image_width, image_height);
873 if(rez!=Success){
874 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: BlendSubpicture failed rez=%d\n",rez);
875 assert(0);
876 return;
878 // XvMCFlushSurface(mDisplay,osd_rndr->p_surface);//fixme- should I?
880 //When replaceing the surface with osd one, save the flags too!
881 osd_rndr->picture_structure = p_render_surface_to_show->picture_structure;
882 //add more if needed osd_rndr-> = p_render_surface_to_show->;
884 p_render_surface_to_show->state &= ~AV_XVMC_STATE_DISPLAY_PENDING;
885 p_render_surface_to_show->state |= AV_XVMC_STATE_OSD_SOURCE;
886 p_render_surface_to_show->p_osd_target_surface_render = osd_rndr;
888 p_render_surface_to_show = osd_rndr;
889 p_render_surface_to_show->state = AV_XVMC_STATE_DISPLAY_PENDING;
891 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc:draw_osd: surface_to_show changed to %p\n",osd_rndr);
892 }//endof if(BLEND)
893 if(subpicture_mode == BACKEND_SUBPICTURE){
894 rez = XvMCBlendSubpicture(mDisplay,
895 p_render_surface_to_show->p_surface,
896 &subpicture,
897 0, 0, subpicture.width, subpicture.height,
898 0, 0, image_width, image_height);
902 }//if(BLEND||BACKEND)
905 static void xvmc_sync_surface(XvMCSurface * srf){
906 int status,rez;
908 rez = XvMCGetSurfaceStatus(mDisplay,srf,&status);
909 assert(rez==Success);
910 if((status & XVMC_RENDERING) == 0)
911 return;//surface is already complete
912 if(use_sleep){
913 rez = XvMCFlushSurface(mDisplay, srf);
914 assert(rez==Success);
917 usec_sleep(1000);//1ms (may be 20ms on linux)
918 XvMCGetSurfaceStatus(mDisplay,srf,&status);
919 } while (status & XVMC_RENDERING);
920 return;//done
923 XvMCSyncSurface(mDisplay, srf);
926 static void put_xvmc_image(struct xvmc_pix_fmt *p_render_surface,
927 int draw_ck){
928 int rez;
929 struct vo_rect src_rect, dst_rect;
930 int i;
932 if(p_render_surface == NULL)
933 return;
935 calc_src_dst_rects(image_width, image_height, &src_rect, &dst_rect, NULL, NULL);
937 if(draw_ck)
938 vo_xv_draw_colorkey(dst_rect.left, dst_rect.top, dst_rect.width, dst_rect.height);
940 if(benchmark)
941 return;
943 for (i = 1; i <= bob_deinterlace + 1; i++) {
944 int field = top_field_first ? i : i ^ 3;
945 rez = XvMCPutSurface(mDisplay, p_render_surface->p_surface,
946 vo_window,
947 src_rect.left, src_rect.top, src_rect.width, src_rect.height,
948 dst_rect.left, dst_rect.top, dst_rect.width, dst_rect.height,
949 bob_deinterlace ? field : 3);
950 if(rez != Success){
951 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: PutSurface failer, critical error %d!\n",rez);
952 assert(0);
955 XFlush(mDisplay);
958 static void flip_page(void){
959 int i,cfs;
961 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: flip_page show(rndr=%p)\n\n",p_render_surface_to_show);
963 if(p_render_surface_to_show == NULL) return;
964 assert( p_render_surface_to_show->xvmc_id == AV_XVMC_ID );
965 //fixme assert( p_render_surface_to_show != p_render_surface_visible);
967 if(use_queue){
968 // fill the queue until only n free surfaces remain
969 // after that start displaying
970 cfs = count_free_surfaces();
971 show_queue[free_element++] = p_render_surface_to_show;
972 if(cfs > 3){//well have 3 free surfaces after add queue
973 if(free_element > 1)//a little voodoo magic
974 xvmc_sync_surface(show_queue[0]->p_surface);
975 return;
977 p_render_surface_to_show=show_queue[0];
978 mp_msg(MSGT_VO,MSGL_DBG5,"vo_xvmc: flip_queue free_element=%d\n",free_element);
979 free_element--;
980 for(i=0; i<free_element; i++){
981 show_queue[i] = show_queue[i+1];
983 show_queue[free_element] = NULL;
986 // make sure the rendering is done
987 xvmc_sync_surface(p_render_surface_to_show->p_surface);
989 //the visible surface won't be displayed anymore, mark it as free
990 if(p_render_surface_visible != NULL)
991 p_render_surface_visible->state &= ~AV_XVMC_STATE_DISPLAY_PENDING;
993 //!!fixme assert(p_render_surface_to_show->state & AV_XVMC_STATE_DISPLAY_PENDING);
995 //show it, displaying is always vsynced, so skip it for benchmark
996 put_xvmc_image(p_render_surface_to_show,first_frame);
997 first_frame=0;//make sure we won't draw it anymore
999 p_render_surface_visible = p_render_surface_to_show;
1000 p_render_surface_to_show = NULL;
1003 static void check_events(void){
1004 int e=vo_x11_check_events(mDisplay);
1006 if(e&VO_EVENT_RESIZE)
1008 e |= VO_EVENT_EXPOSE;
1010 if ( e & VO_EVENT_EXPOSE )
1012 put_xvmc_image(p_render_surface_visible,1);
1016 static void xvmc_free(void){
1017 int i;
1019 if( subpicture_alloc ){
1021 XvMCDestroySubpicture(mDisplay,&subpicture);
1022 deallocate_xvimage();
1024 subpicture_alloc = 0;
1026 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: subpicture destroyed\n");
1029 if( number_of_surfaces ){
1031 XvMCDestroyMacroBlocks(mDisplay,&mv_blocks);
1032 XvMCDestroyBlocks(mDisplay,&data_blocks);
1034 for(i=0; i<number_of_surfaces; i++)
1036 XvMCHideSurface(mDisplay,&surface_array[i]);//it doesn't hurt, I hope
1037 XvMCDestroySurface(mDisplay,&surface_array[i]);
1039 if( (surface_render[i].state != 0) &&
1040 (p_render_surface_visible != &surface_render[i]) )
1041 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc::uninit surface_render[%d].status=%d\n",i,
1042 surface_render[i].state);
1045 memset(surface_render, 0, MAX_SURFACES * sizeof(struct xvmc_pix_fmt)); //for debugging
1046 free(surface_render);surface_render=NULL;
1048 XvMCDestroyContext(mDisplay,&ctx);
1049 number_of_surfaces = 0;
1051 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: Context sucessfuly freed\n");
1055 if( xv_port !=0 ){
1056 XvUngrabPort(mDisplay,xv_port,CurrentTime);
1057 xv_port = 0;
1058 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: xv_port sucessfuly ungrabed\n");
1062 static void uninit(void){
1063 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: uninit called\n");
1064 xvmc_free();
1065 //from vo_xv
1066 #ifdef CONFIG_XF86VM
1067 vo_vm_close();
1068 #endif
1069 vo_x11_uninit();
1072 static int query_format(uint32_t format){
1073 uint32_t flags;
1074 XvMCSurfaceInfo qsurface_info;
1075 int mode_id;
1077 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: query_format=%X\n",format);
1079 if(!IMGFMT_IS_XVMC(format)) return 0;// no caps supported
1080 mode_id = xvmc_find_surface_by_format(format, 16, 16, &qsurface_info, 1);//true=1 - quering
1082 if( mode_id == 0 ) return 0;
1084 flags = VFCAP_CSP_SUPPORTED |
1085 VFCAP_CSP_SUPPORTED_BY_HW |
1086 VFCAP_ACCEPT_STRIDE;
1088 if( (qsurface_info.subpicture_max_width != 0) &&
1089 (qsurface_info.subpicture_max_height != 0) )
1090 flags|=VFCAP_OSD;
1091 return flags;
1095 static int draw_slice(uint8_t *image[], int stride[],
1096 int w, int h, int x, int y){
1097 struct xvmc_pix_fmt *rndr;
1098 int rez;
1100 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: draw_slice y=%d\n",y);
1102 rndr = (struct xvmc_pix_fmt*)image[2]; //this is copy of priv-ate
1103 assert( rndr != NULL );
1104 assert( rndr->xvmc_id == AV_XVMC_ID );
1106 rez = XvMCRenderSurface(mDisplay,&ctx,rndr->picture_structure,
1107 rndr->p_surface,
1108 rndr->p_past_surface,
1109 rndr->p_future_surface,
1110 rndr->flags,
1111 rndr->filled_mv_blocks_num,rndr->start_mv_blocks_num,
1112 &mv_blocks,&data_blocks);
1113 if(rez != Success)
1115 int i;
1116 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: RenderSirface returned %d\n",rez);
1118 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: pict=%d,flags=%x,start_blocks=%d,num_blocks=%d\n",
1119 rndr->picture_structure,rndr->flags,rndr->start_mv_blocks_num,
1120 rndr->filled_mv_blocks_num);
1121 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: this_surf=%p, past_surf=%p, future_surf=%p\n",
1122 rndr->p_surface,rndr->p_past_surface,rndr->p_future_surface);
1124 for(i=0; i<rndr->filled_mv_blocks_num; i++){
1125 XvMCMacroBlock* testblock;
1126 testblock = &mv_blocks.macro_blocks[i];
1128 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: mv_block - x=%d,y=%d,mb_type=0x%x,mv_type=0x%x,mv_field_select=%d\n",
1129 testblock->x,testblock->y,testblock->macroblock_type,
1130 testblock->motion_type,testblock->motion_vertical_field_select);
1131 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: dct_type=%d,data_index=0x%x,cbp=%d,pad0=%d\n",
1132 testblock->dct_type,testblock->index,testblock->coded_block_pattern,
1133 testblock->pad0);
1134 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc::slice: PMV[0][0][0/1]=(%d,%d)\n",
1135 testblock->PMV[0][0][0],testblock->PMV[0][0][1]);
1138 assert(rez==Success);
1139 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: flush surface\n");
1140 rez = XvMCFlushSurface(mDisplay, rndr->p_surface);
1141 assert(rez==Success);
1143 // rndr->start_mv_blocks_num += rndr->filled_mv_blocks_num;
1144 rndr->start_mv_blocks_num = 0;
1145 rndr->filled_mv_blocks_num = 0;
1147 rndr->next_free_data_block_num = 0;
1149 return VO_TRUE;
1152 //XvMCHide hides the surface on next retrace, so
1153 //check if the surface is not still displaying
1154 static void check_osd_source(struct xvmc_pix_fmt *src_rndr) {
1155 struct xvmc_pix_fmt *osd_rndr;
1156 int stat;
1158 //If this is source surface, check does the OSD rendering is compleate
1159 if(src_rndr->state & AV_XVMC_STATE_OSD_SOURCE){
1160 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: OSD surface=%p quering\n",src_rndr);
1161 osd_rndr = src_rndr->p_osd_target_surface_render;
1162 XvMCGetSurfaceStatus(mDisplay, osd_rndr->p_surface, &stat);
1163 if(!(stat & XVMC_RENDERING))
1164 src_rndr->state &= ~AV_XVMC_STATE_OSD_SOURCE;
1167 static int count_free_surfaces(void) {
1168 int i,num;
1170 num=0;
1171 for(i=0; i<number_of_surfaces; i++){
1172 check_osd_source(&surface_render[i]);
1173 if(surface_render[i].state == 0)
1174 num++;
1176 return num;
1179 static struct xvmc_pix_fmt *find_free_surface(void) {
1180 int i,t;
1181 int stat;
1182 struct xvmc_pix_fmt *visible_rndr;
1184 visible_rndr = NULL;
1185 for(i=0; i<number_of_surfaces; i++){
1187 check_osd_source(&surface_render[i]);
1188 if( surface_render[i].state == 0){
1189 XvMCGetSurfaceStatus(mDisplay, surface_render[i].p_surface,&stat);
1190 if( (stat & XVMC_DISPLAYING) == 0 )
1191 return &surface_render[i];
1192 visible_rndr = &surface_render[i];// remember it, use as last resort
1196 //all surfaces are busy, but there is one that will be free
1197 //on next monitor retrace, we just have to wait
1198 if(visible_rndr != NULL){
1199 mp_msg(MSGT_VO,MSGL_INFO,"vo_xvmc: waiting retrace\n");
1200 for(t=0;t<1000;t++){
1201 usec_sleep(1000);//1ms
1202 XvMCGetSurfaceStatus(mDisplay, visible_rndr->p_surface,&stat);
1203 if( (stat & XVMC_DISPLAYING) == 0 )
1204 return visible_rndr;
1207 //todo remove when stable
1208 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: no free surfaces, this should not happen in g1\n");
1209 for(i=0;i<number_of_surfaces;i++)
1210 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: surface[%d].state=%d\n",i,surface_render[i].state);
1211 return NULL;
1214 static void xvmc_clean_surfaces(void){
1215 int i;
1217 for(i=0; i<number_of_surfaces; i++){
1219 surface_render[i].state&=!( AV_XVMC_STATE_DISPLAY_PENDING |
1220 AV_XVMC_STATE_OSD_SOURCE |
1222 surface_render[i].p_osd_target_surface_render=NULL;
1223 if(surface_render[i].state != 0){
1224 mp_msg(MSGT_VO,MSGL_WARN,"vo_xvmc: surface[%d].state=%d\n",
1225 i,surface_render[i].state);
1228 free_element=0;//clean up the queue
1231 static uint32_t get_image(mp_image_t *mpi){
1232 struct xvmc_pix_fmt *rndr;
1234 rndr = find_free_surface();
1236 if(rndr == NULL){
1237 mp_msg(MSGT_VO,MSGL_ERR,"vo_xvmc: get_image failed\n");
1238 return VO_FALSE;
1241 assert(rndr->start_mv_blocks_num == 0);
1242 assert(rndr->filled_mv_blocks_num == 0);
1243 assert(rndr->next_free_data_block_num == 0);
1245 mpi->flags |= MP_IMGFLAG_DIRECT;
1246 //keep strides 0 to avoid field manipulations
1247 mpi->stride[0] = 0;
1248 mpi->stride[1] = 0;
1249 mpi->stride[2] = 0;
1251 // these are shared!! so watch out
1252 // do call RenderSurface before overwriting
1253 mpi->planes[0] = (char*)data_blocks.blocks;
1254 mpi->planes[1] = (char*)mv_blocks.macro_blocks;
1255 mpi->priv =
1256 mpi->planes[2] = (char*)rndr;
1258 rndr->picture_structure = 0;
1259 rndr->flags = 0;
1260 rndr->state = 0;
1261 rndr->start_mv_blocks_num = 0;
1262 rndr->filled_mv_blocks_num = 0;
1263 rndr->next_free_data_block_num = 0;
1265 mp_msg(MSGT_VO,MSGL_DBG4,"vo_xvmc: get_image: rndr=%p (surface=%p) \n",
1266 rndr,rndr->p_surface);
1267 return VO_TRUE;
1270 static int control(uint32_t request, void *data, ... )
1272 switch (request){
1273 case VOCTRL_GET_DEINTERLACE:
1274 *(int*)data = bob_deinterlace;
1275 return VO_TRUE;
1276 case VOCTRL_SET_DEINTERLACE:
1277 bob_deinterlace = *(int*)data;
1278 return VO_TRUE;
1279 case VOCTRL_QUERY_FORMAT:
1280 return query_format(*((uint32_t*)data));
1281 case VOCTRL_DRAW_IMAGE:
1282 return xvmc_draw_image((mp_image_t *)data);
1283 case VOCTRL_GET_IMAGE:
1284 return get_image((mp_image_t *)data);
1285 //vo_xv
1286 case VOCTRL_GUISUPPORT:
1287 return VO_TRUE;
1288 case VOCTRL_ONTOP:
1289 vo_x11_ontop();
1290 return VO_TRUE;
1291 case VOCTRL_FULLSCREEN:
1292 vo_x11_fullscreen();
1293 // indended, fallthrough to update panscan on fullscreen/windowed switch
1294 case VOCTRL_SET_PANSCAN:
1295 if ( ( vo_fs && ( vo_panscan != vo_panscan_amount ) ) || ( !vo_fs && vo_panscan_amount ) )
1297 int old_y = vo_panscan_y;
1298 panscan_calc();
1300 if(old_y != vo_panscan_y)
1302 //this also draws the colorkey
1303 put_xvmc_image(p_render_surface_visible,1);
1306 return VO_TRUE;
1307 case VOCTRL_GET_PANSCAN:
1308 if ( !vo_config_count || !vo_fs ) return VO_FALSE;
1309 return VO_TRUE;
1310 case VOCTRL_SET_EQUALIZER:
1312 va_list ap;
1313 int value;
1315 va_start(ap, data);
1316 value = va_arg(ap, int);
1317 va_end(ap);
1319 return vo_xv_set_eq(xv_port, data, value);
1322 case VOCTRL_GET_EQUALIZER:
1324 va_list ap;
1325 int *value;
1327 va_start(ap, data);
1328 value = va_arg(ap, int*);
1329 va_end(ap);
1331 return vo_xv_get_eq(xv_port, data, value);
1333 case VOCTRL_UPDATE_SCREENINFO:
1334 update_xinerama_info();
1335 return VO_TRUE;
1337 return VO_NOTIMPL;