1 /*****************************************************************************
2 * xcommon.c: Functions common to the X11 and XVideo plugins
3 *****************************************************************************
4 * Copyright (C) 1998-2006 the VideoLAN team
7 * Authors: Vincent Seguin <seguin@via.ecp.fr>
8 * Sam Hocevar <sam@zoy.org>
9 * David Kennedy <dkennedy@tinytoad.com>
10 * Gildas Bazin <gbazin@videolan.org>
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
25 *****************************************************************************/
27 /*****************************************************************************
29 *****************************************************************************/
34 #include <vlc_common.h>
35 #include <vlc_playlist.h>
37 #include <vlc_vout_window.h>
40 #ifdef HAVE_MACHINE_PARAM_H
42 # include <machine/param.h>
43 # include <sys/types.h> /* typedef ushort */
48 #include <X11/extensions/Xsp.h>
52 # include <sys/shm.h> /* shmget(), shmctl() */
56 #include <X11/Xproto.h>
58 #include <X11/Xutil.h>
60 # include <X11/extensions/XShm.h>
62 #ifdef DPMSINFO_IN_DPMS_H
63 # include <X11/extensions/dpms.h>
66 #ifdef MODULE_NAME_IS_glx
70 #ifdef MODULE_NAME_IS_xvmc
71 # include <X11/extensions/Xv.h>
72 # include <X11/extensions/Xvlib.h>
73 # include <X11/extensions/vldXvMC.h>
74 # include "../../codec/xvmc/accel_xvmc.h"
79 /*****************************************************************************
81 *****************************************************************************/
82 int Activate ( vlc_object_t * );
83 void Deactivate ( vlc_object_t * );
85 #ifndef MODULE_NAME_IS_glx
86 static int InitVideo ( vout_thread_t * );
87 static void EndVideo ( vout_thread_t * );
88 static void DisplayVideo ( vout_thread_t *, picture_t * );
90 static int ManageVideo ( vout_thread_t * );
91 static int Control ( vout_thread_t *, int, va_list );
93 static int InitDisplay ( vout_thread_t * );
95 static int CreateWindow ( vout_thread_t *, x11_window_t * );
96 static void DestroyWindow ( vout_thread_t *, x11_window_t * );
98 #ifndef MODULE_NAME_IS_glx
99 static int NewPicture ( vout_thread_t *, picture_t * );
100 static void FreePicture ( vout_thread_t *, picture_t * );
103 #ifdef HAVE_SYS_SHM_H
104 static int i_shm_major = 0;
107 static void ToggleFullScreen ( vout_thread_t * );
109 static void EnableXScreenSaver ( vout_thread_t * );
110 static void DisableXScreenSaver ( vout_thread_t * );
112 static void CreateCursor ( vout_thread_t * );
113 static void DestroyCursor ( vout_thread_t * );
114 static void ToggleCursor ( vout_thread_t * );
116 #if defined(MODULE_NAME_IS_xvmc)
117 static int XVideoGetPort ( vout_thread_t *, vlc_fourcc_t, picture_heap_t * );
118 static void RenderVideo ( vout_thread_t *, picture_t * );
119 static int xvmc_check_yv12( Display *display, XvPortID port );
120 static void xvmc_update_XV_DOUBLE_BUFFER( vout_thread_t *p_vout );
123 static int X11ErrorHandler( Display *, XErrorEvent * );
126 static void EnablePixelDoubling( vout_thread_t *p_vout );
127 static void DisablePixelDoubling( vout_thread_t *p_vout );
131 static const int i_backlight_on_interval = 300;
136 /*****************************************************************************
137 * Activate: allocate X11 video thread output method
138 *****************************************************************************
139 * This function allocate and initialize a X11 vout method. It uses some of the
140 * vout properties to choose the window size, and change them according to the
141 * actual properties of the display.
142 *****************************************************************************/
143 int Activate ( vlc_object_t *p_this )
145 vout_thread_t *p_vout = (vout_thread_t *)p_this;
147 #if defined(MODULE_NAME_IS_xvmc)
150 vlc_fourcc_t i_chroma = 0;
154 #ifndef MODULE_NAME_IS_glx
155 p_vout->pf_init = InitVideo;
156 p_vout->pf_end = EndVideo;
157 p_vout->pf_display = DisplayVideo;
159 #ifdef MODULE_NAME_IS_xvmc
160 p_vout->pf_render = RenderVideo;
162 p_vout->pf_render = NULL;
164 p_vout->pf_manage = ManageVideo;
165 p_vout->pf_control = Control;
167 /* Allocate structure */
168 p_vout->p_sys = malloc( sizeof( vout_sys_t ) );
169 if( p_vout->p_sys == NULL )
172 /* key and mouse event handling */
173 p_vout->p_sys->i_vout_event = var_CreateGetInteger( p_vout, "vout-event" );
175 /* Open display, using the "display" config variable or the DISPLAY
176 * environment variable */
177 psz_display = config_GetPsz( p_vout, MODULE_STRING "-display" );
179 p_vout->p_sys->p_display = XOpenDisplay( psz_display );
181 if( p_vout->p_sys->p_display == NULL ) /* error */
183 msg_Err( p_vout, "cannot open display %s",
184 XDisplayName( psz_display ) );
185 free( p_vout->p_sys );
191 /* Replace error handler so we can intercept some non-fatal errors */
192 XSetErrorHandler( X11ErrorHandler );
194 /* Get a screen ID matching the XOpenDisplay return value */
195 p_vout->p_sys->i_screen = DefaultScreen( p_vout->p_sys->p_display );
197 #if defined(MODULE_NAME_IS_xvmc)
198 psz_chroma = config_GetPsz( p_vout, "xvideo-chroma" );
201 if( strlen( psz_chroma ) >= 4 )
203 /* Do not use direct assignment because we are not sure of the
205 memcpy(&i_chroma, psz_chroma, 4);
214 msg_Dbg( p_vout, "forcing chroma 0x%.8x (%4.4s)",
215 i_chroma, (char*)&i_chroma );
219 i_chroma = p_vout->render.i_chroma;
222 /* Check that we have access to an XVideo port providing this chroma */
223 p_vout->p_sys->i_xvport = XVideoGetPort( p_vout, VLC2X11_FOURCC(i_chroma),
225 if( p_vout->p_sys->i_xvport < 0 )
227 /* If a specific chroma format was requested, then we don't try to
228 * be cleverer than the user. He knew pretty well what he wanted. */
231 XCloseDisplay( p_vout->p_sys->p_display );
232 free( p_vout->p_sys );
236 /* It failed, but it's not completely lost ! We try to open an
237 * XVideo port for an YUY2 picture. We'll need to do an YUV
238 * conversion, but at least it has got scaling. */
239 p_vout->p_sys->i_xvport =
240 XVideoGetPort( p_vout, X11_FOURCC('Y','U','Y','2'),
242 if( p_vout->p_sys->i_xvport < 0 )
244 /* It failed, but it's not completely lost ! We try to open an
245 * XVideo port for a simple 16bpp RGB picture. We'll need to do
246 * an YUV conversion, but at least it has got scaling. */
247 p_vout->p_sys->i_xvport =
248 XVideoGetPort( p_vout, X11_FOURCC('R','V','1','6'),
250 if( p_vout->p_sys->i_xvport < 0 )
252 XCloseDisplay( p_vout->p_sys->p_display );
253 free( p_vout->p_sys );
258 p_vout->output.i_chroma = vlc_fourcc_GetCodec( VIDEO_ES, X112VLC_FOURCC(p_vout->output.i_chroma) );
259 #elif defined(MODULE_NAME_IS_glx)
261 int i_opcode, i_evt, i_err = 0;
262 int i_maj, i_min = 0;
264 /* Check for GLX extension */
265 if( !XQueryExtension( p_vout->p_sys->p_display, "GLX",
266 &i_opcode, &i_evt, &i_err ) )
268 msg_Err( p_this, "GLX extension not supported" );
269 XCloseDisplay( p_vout->p_sys->p_display );
270 free( p_vout->p_sys );
273 if( !glXQueryExtension( p_vout->p_sys->p_display, &i_err, &i_evt ) )
275 msg_Err( p_this, "glXQueryExtension failed" );
276 XCloseDisplay( p_vout->p_sys->p_display );
277 free( p_vout->p_sys );
281 /* Check GLX version */
282 if (!glXQueryVersion( p_vout->p_sys->p_display, &i_maj, &i_min ) )
284 msg_Err( p_this, "glXQueryVersion failed" );
285 XCloseDisplay( p_vout->p_sys->p_display );
286 free( p_vout->p_sys );
289 if( i_maj <= 0 || ((i_maj == 1) && (i_min < 3)) )
291 p_vout->p_sys->b_glx13 = false;
292 msg_Dbg( p_this, "using GLX 1.2 API" );
296 p_vout->p_sys->b_glx13 = true;
297 msg_Dbg( p_this, "using GLX 1.3 API" );
302 /* Create blank cursor (for mouse cursor autohiding) */
303 p_vout->p_sys->i_time_mouse_last_moved = mdate();
304 p_vout->p_sys->i_mouse_hide_timeout =
305 var_GetInteger(p_vout, "mouse-hide-timeout") * 1000;
306 p_vout->p_sys->b_mouse_pointer_visible = 1;
307 CreateCursor( p_vout );
309 /* Set main window's size */
310 p_vout->p_sys->window.i_x = 0;
311 p_vout->p_sys->window.i_y = 0;
312 p_vout->p_sys->window.i_width = p_vout->i_window_width;
313 p_vout->p_sys->window.i_height = p_vout->i_window_height;
314 var_Create( p_vout, "video-title", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
315 /* Spawn base window - this window will include the video output window,
316 * but also command buttons, subtitles and other indicators */
317 if( CreateWindow( p_vout, &p_vout->p_sys->window ) )
319 msg_Err( p_vout, "cannot create X11 window" );
320 DestroyCursor( p_vout );
321 XCloseDisplay( p_vout->p_sys->p_display );
322 free( p_vout->p_sys );
326 /* Open and initialize device. */
327 if( InitDisplay( p_vout ) )
329 msg_Err( p_vout, "cannot initialize X11 display" );
330 DestroyCursor( p_vout );
331 DestroyWindow( p_vout, &p_vout->p_sys->window );
332 XCloseDisplay( p_vout->p_sys->p_display );
333 free( p_vout->p_sys );
337 /* Disable screen saver */
338 DisableXScreenSaver( p_vout );
341 p_vout->p_sys->i_time_button_last_pressed = 0;
343 #ifdef MODULE_NAME_IS_xvmc
344 p_vout->p_sys->p_last_subtitle_save = NULL;
345 psz_value = config_GetPsz( p_vout, "xvmc-deinterlace-mode" );
347 /* Look what method was requested */
348 //var_Create( p_vout, "xvmc-deinterlace-mode", VLC_VAR_STRING );
349 //var_Change( p_vout, "xvmc-deinterlace-mode", VLC_VAR_INHERITVALUE, &val, NULL );
352 if( (strcmp(psz_value, "bob") == 0) ||
353 (strcmp(psz_value, "blend") == 0) )
354 p_vout->p_sys->xvmc_deinterlace_method = 2;
355 else if (strcmp(psz_value, "discard") == 0)
356 p_vout->p_sys->xvmc_deinterlace_method = 1;
358 p_vout->p_sys->xvmc_deinterlace_method = 0;
362 p_vout->p_sys->xvmc_deinterlace_method = 0;
364 /* Look what method was requested */
365 //var_Create( p_vout, "xvmc-crop-style", VLC_VAR_STRING );
366 //var_Change( p_vout, "xvmc-crop-style", VLC_VAR_INHERITVALUE, &val, NULL );
367 psz_value = config_GetPsz( p_vout, "xvmc-crop-style" );
371 if( strncmp( psz_value, "eq", 2 ) == 0 )
372 p_vout->p_sys->xvmc_crop_style = 1;
373 else if( strncmp( psz_value, "4-16", 4 ) == 0)
374 p_vout->p_sys->xvmc_crop_style = 2;
375 else if( strncmp( psz_value, "16-4", 4 ) == 0)
376 p_vout->p_sys->xvmc_crop_style = 3;
378 p_vout->p_sys->xvmc_crop_style = 0;
382 p_vout->p_sys->xvmc_crop_style = 0;
384 msg_Dbg(p_vout, "Deinterlace = %d", p_vout->p_sys->xvmc_deinterlace_method);
385 msg_Dbg(p_vout, "Crop = %d", p_vout->p_sys->xvmc_crop_style);
387 if( checkXvMCCap( p_vout ) == VLC_EGENERIC )
389 msg_Err( p_vout, "no XVMC capability found" );
390 Deactivate( p_this );
393 subpicture_t sub_pic;
394 sub_pic.p_sys = NULL;
395 p_vout->p_sys->last_date = 0;
399 p_vout->p_sys->i_hw_scale = 1;
403 p_vout->p_sys->i_backlight_on_counter = i_backlight_on_interval;
404 p_vout->p_sys->p_octx = osso_initialize( "vlc", VERSION, 0, NULL );
405 if ( p_vout->p_sys->p_octx == NULL ) {
406 msg_Err( p_vout, "Could not get osso context" );
408 msg_Dbg( p_vout, "Initialized osso context" );
412 /* Variable to indicate if the window should be on top of others */
413 /* Trigger a callback right now */
414 var_TriggerCallback( p_vout, "video-on-top" );
419 /*****************************************************************************
420 * Deactivate: destroy X11 video thread output method
421 *****************************************************************************
422 * Terminate an output method created by Open
423 *****************************************************************************/
424 void Deactivate ( vlc_object_t *p_this )
426 vout_thread_t *p_vout = (vout_thread_t *)p_this;
428 /* Restore cursor if it was blanked */
429 if( !p_vout->p_sys->b_mouse_pointer_visible )
431 ToggleCursor( p_vout );
434 #if defined(MODULE_NAME_IS_xvmc)
435 if( p_vout->p_sys->xvmc_cap )
437 xvmc_context_writer_lock( &p_vout->p_sys->xvmc_lock );
438 xxmc_dispose_context( p_vout );
439 if( p_vout->p_sys->old_subpic )
441 xxmc_xvmc_free_subpicture( p_vout, p_vout->p_sys->old_subpic );
442 p_vout->p_sys->old_subpic = NULL;
444 if( p_vout->p_sys->new_subpic )
446 xxmc_xvmc_free_subpicture( p_vout, p_vout->p_sys->new_subpic );
447 p_vout->p_sys->new_subpic = NULL;
449 free( p_vout->p_sys->xvmc_cap );
450 xvmc_context_writer_unlock( &p_vout->p_sys->xvmc_lock );
455 DisablePixelDoubling(p_vout);
458 DestroyCursor( p_vout );
459 EnableXScreenSaver( p_vout );
460 DestroyWindow( p_vout, &p_vout->p_sys->window );
461 XCloseDisplay( p_vout->p_sys->p_display );
463 /* Destroy structure */
464 #ifdef MODULE_NAME_IS_xvmc
465 free_context_lock( &p_vout->p_sys->xvmc_lock );
469 if ( p_vout->p_sys->p_octx != NULL ) {
470 msg_Dbg( p_vout, "Deinitializing osso context" );
471 osso_deinitialize( p_vout->p_sys->p_octx );
475 free( p_vout->p_sys );
478 #ifdef MODULE_NAME_IS_xvmc
480 #define XINE_IMGFMT_YV12 (('2'<<24)|('1'<<16)|('V'<<8)|'Y')
483 static int xvmc_check_yv12( Display *display, XvPortID port )
485 XvImageFormatValues *formatValues;
489 formatValues = XvListImageFormats( display, port, &formats );
491 for( i = 0; i < formats; i++ )
493 if( ( formatValues[i].id == XINE_IMGFMT_YV12 ) &&
494 ( !( strncmp( formatValues[i].guid, "YV12", 4 ) ) ) )
496 XFree (formatValues);
501 XFree (formatValues);
505 static void xvmc_sync_surface( vout_thread_t *p_vout, XvMCSurface * srf )
507 XvMCSyncSurface( p_vout->p_sys->p_display, srf );
510 static void xvmc_update_XV_DOUBLE_BUFFER( vout_thread_t *p_vout )
513 int xv_double_buffer;
515 xv_double_buffer = 1;
517 XLockDisplay( p_vout->p_sys->p_display );
518 atom = XInternAtom( p_vout->p_sys->p_display, "XV_DOUBLE_BUFFER", False );
520 XvSetPortAttribute (p_vout->p_sys->p_display, p_vout->p_sys->i_xvport, atom, xv_double_buffer);
522 XvMCSetAttribute( p_vout->p_sys->p_display, &p_vout->p_sys->context, atom, xv_double_buffer );
523 XUnlockDisplay( p_vout->p_sys->p_display );
525 //xprintf(this->xine, XINE_VERBOSITY_DEBUG,
526 // "video_out_xxmc: double buffering mode = %d\n", xv_double_buffer);
529 static void RenderVideo( vout_thread_t *p_vout, picture_t *p_pic )
531 vlc_xxmc_t *xxmc = NULL;
533 xvmc_context_reader_lock( &p_vout->p_sys->xvmc_lock );
535 xxmc = &p_pic->p_sys->xxmc_data;
536 if( (!xxmc->decoded ||
537 !xxmc_xvmc_surface_valid( p_vout, p_pic->p_sys->xvmc_surf )) )
539 xvmc_context_reader_unlock( &p_vout->p_sys->xvmc_lock );
544 vlc_mutex_lock( &p_vout->lastsubtitle_lock );
545 if (p_vout->p_sys->p_last_subtitle != NULL)
547 if( p_vout->p_sys->p_last_subtitle_save != p_vout->p_sys->p_last_subtitle )
549 p_vout->p_sys->new_subpic =
550 xxmc_xvmc_alloc_subpicture( p_vout, &p_vout->p_sys->context,
551 p_vout->p_sys->xvmc_width,
552 p_vout->p_sys->xvmc_height,
553 p_vout->p_sys->xvmc_cap[p_vout->p_sys->xvmc_cur_cap].subPicType.id );
555 if (p_vout->p_sys->new_subpic)
557 XVMCLOCKDISPLAY( p_vout->p_sys->p_display );
558 XvMCClearSubpicture( p_vout->p_sys->p_display,
559 p_vout->p_sys->new_subpic,
562 p_vout->p_sys->xvmc_width,
563 p_vout->p_sys->xvmc_height,
565 XVMCUNLOCKDISPLAY( p_vout->p_sys->p_display );
566 clear_xx44_palette( &p_vout->p_sys->palette );
568 if( sub_pic.p_sys == NULL )
570 sub_pic.p_sys = malloc( sizeof( picture_sys_t ) );
571 if( sub_pic.p_sys != NULL )
573 sub_pic.p_sys->p_vout = p_vout;
574 sub_pic.p_sys->xvmc_surf = NULL;
575 sub_pic.p_sys->p_image = p_vout->p_sys->subImage;
578 sub_pic.p_sys->p_image = p_vout->p_sys->subImage;
579 sub_pic.p->p_pixels = sub_pic.p_sys->p_image->data;
580 sub_pic.p->i_pitch = p_vout->output.i_width;
582 memset( p_vout->p_sys->subImage->data, 0,
583 (p_vout->p_sys->subImage->width * p_vout->p_sys->subImage->height) );
585 if (p_vout->p_last_subtitle != NULL)
587 blend_xx44( p_vout->p_sys->subImage->data,
588 p_vout->p_last_subtitle,
589 p_vout->p_sys->subImage->width,
590 p_vout->p_sys->subImage->height,
591 p_vout->p_sys->subImage->width,
592 &p_vout->p_sys->palette,
593 (p_vout->p_sys->subImage->id == FOURCC_IA44) );
596 XVMCLOCKDISPLAY( p_vout->p_sys->p_display );
597 XvMCCompositeSubpicture( p_vout->p_sys->p_display,
598 p_vout->p_sys->new_subpic,
599 p_vout->p_sys->subImage,
602 p_vout->output.i_width, /* overlay->width, */
603 p_vout->output.i_height, /* overlay->height */
606 XVMCUNLOCKDISPLAY( p_vout->p_sys->p_display );
607 if (p_vout->p_sys->old_subpic)
609 xxmc_xvmc_free_subpicture( p_vout,
610 p_vout->p_sys->old_subpic);
611 p_vout->p_sys->old_subpic = NULL;
613 if (p_vout->p_sys->new_subpic)
615 p_vout->p_sys->old_subpic = p_vout->p_sys->new_subpic;
616 p_vout->p_sys->new_subpic = NULL;
617 xx44_to_xvmc_palette( &p_vout->p_sys->palette,
618 p_vout->p_sys->xvmc_palette,
620 p_vout->p_sys->old_subpic->num_palette_entries,
621 p_vout->p_sys->old_subpic->entry_bytes,
622 p_vout->p_sys->old_subpic->component_order );
623 XVMCLOCKDISPLAY( p_vout->p_sys->p_display );
624 XvMCSetSubpicturePalette( p_vout->p_sys->p_display,
625 p_vout->p_sys->old_subpic,
626 p_vout->p_sys->xvmc_palette );
627 XvMCFlushSubpicture( p_vout->p_sys->p_display,
628 p_vout->p_sys->old_subpic);
629 XvMCSyncSubpicture( p_vout->p_sys->p_display,
630 p_vout->p_sys->old_subpic );
631 XVMCUNLOCKDISPLAY( p_vout->p_sys->p_display );
634 XVMCLOCKDISPLAY( p_vout->p_sys->p_display);
635 if (p_vout->p_sys->xvmc_backend_subpic )
637 XvMCBlendSubpicture( p_vout->p_sys->p_display,
638 p_pic->p_sys->xvmc_surf,
639 p_vout->p_sys->old_subpic,
642 p_vout->p_sys->xvmc_width,
643 p_vout->p_sys->xvmc_height,
646 p_vout->p_sys->xvmc_width,
647 p_vout->p_sys->xvmc_height );
651 XvMCBlendSubpicture2( p_vout->p_sys->p_display,
652 p_pic->p_sys->xvmc_surf,
653 p_pic->p_sys->xvmc_surf,
654 p_vout->p_sys->old_subpic,
657 p_vout->p_sys->xvmc_width,
658 p_vout->p_sys->xvmc_height,
661 p_vout->p_sys->xvmc_width,
662 p_vout->p_sys->xvmc_height );
664 XVMCUNLOCKDISPLAY(p_vout->p_sys->p_display);
669 XVMCLOCKDISPLAY( p_vout->p_sys->p_display );
670 if( p_vout->p_sys->xvmc_backend_subpic )
672 XvMCBlendSubpicture( p_vout->p_sys->p_display,
673 p_pic->p_sys->xvmc_surf,
674 p_vout->p_sys->old_subpic,
676 p_vout->p_sys->xvmc_width,
677 p_vout->p_sys->xvmc_height,
679 p_vout->p_sys->xvmc_width,
680 p_vout->p_sys->xvmc_height );
684 XvMCBlendSubpicture2( p_vout->p_sys->p_display,
685 p_pic->p_sys->xvmc_surf,
686 p_pic->p_sys->xvmc_surf,
687 p_vout->p_sys->old_subpic,
689 p_vout->p_sys->xvmc_width,
690 p_vout->p_sys->xvmc_height,
692 p_vout->p_sys->xvmc_width,
693 p_vout->p_sys->xvmc_height );
695 XVMCUNLOCKDISPLAY( p_vout->p_sys->p_display );
698 p_vout->p_sys->p_last_subtitle_save = p_vout->p_last_subtitle;
700 vlc_mutex_unlock( &p_vout->lastsubtitle_lock );
702 xvmc_context_reader_unlock( &p_vout->p_sys->xvmc_lock );
707 /*****************************************************************************
708 * EnablePixelDoubling: Enables pixel doubling
709 *****************************************************************************
710 * Checks if the double size image fits in current window, and enables pixel
711 * doubling accordingly. The i_hw_scale is the integer scaling factor.
712 *****************************************************************************/
713 static void EnablePixelDoubling( vout_thread_t *p_vout )
715 int i_hor_scale = ( p_vout->p_sys->window.i_width ) / p_vout->render.i_width;
716 int i_vert_scale = ( p_vout->p_sys->window.i_height ) / p_vout->render.i_height;
717 if ( ( i_hor_scale > 1 ) && ( i_vert_scale > 1 ) ) {
718 p_vout->p_sys->i_hw_scale = 2;
719 msg_Dbg( p_vout, "Enabling pixel doubling, scaling factor %d", p_vout->p_sys->i_hw_scale );
720 XSPSetPixelDoubling( p_vout->p_sys->p_display, 0, 1 );
724 /*****************************************************************************
725 * DisablePixelDoubling: Disables pixel doubling
726 *****************************************************************************
727 * The scaling factor i_hw_scale is reset to the no-scaling value 1.
728 *****************************************************************************/
729 static void DisablePixelDoubling( vout_thread_t *p_vout )
731 if ( p_vout->p_sys->i_hw_scale > 1 ) {
732 msg_Dbg( p_vout, "Disabling pixel doubling" );
733 XSPSetPixelDoubling( p_vout->p_sys->p_display, 0, 0 );
734 p_vout->p_sys->i_hw_scale = 1;
740 #if !defined(MODULE_NAME_IS_glx)
741 /*****************************************************************************
742 * InitVideo: initialize X11 video thread output method
743 *****************************************************************************
744 * This function create the XImages needed by the output thread. It is called
745 * at the beginning of the thread, but also each time the window is resized.
746 *****************************************************************************/
747 static int InitVideo( vout_thread_t *p_vout )
749 unsigned int i_index = 0;
752 I_OUTPUTPICTURES = 0;
754 #if defined(MODULE_NAME_IS_xvmc)
755 /* Initialize the output structure; we already found an XVideo port,
756 * and the corresponding chroma we will be using. Since we can
757 * arbitrary scale, stick to the coordinates and aspect. */
758 p_vout->output.i_width = p_vout->render.i_width;
759 p_vout->output.i_height = p_vout->render.i_height;
760 p_vout->output.i_aspect = p_vout->render.i_aspect;
762 p_vout->fmt_out = p_vout->fmt_in;
763 p_vout->fmt_out.i_chroma = p_vout->output.i_chroma;
765 #if XvVersion < 2 || ( XvVersion == 2 && XvRevision < 2 )
766 switch( p_vout->output.i_chroma )
768 case VLC_CODEC_RGB16:
769 #if defined( WORDS_BIGENDIAN )
770 p_vout->output.i_rmask = 0xf800;
771 p_vout->output.i_gmask = 0x07e0;
772 p_vout->output.i_bmask = 0x001f;
774 p_vout->output.i_rmask = 0x001f;
775 p_vout->output.i_gmask = 0x07e0;
776 p_vout->output.i_bmask = 0xf800;
779 case VLC_CODEC_RGB15:
780 #if defined( WORDS_BIGENDIAN )
781 p_vout->output.i_rmask = 0x7c00;
782 p_vout->output.i_gmask = 0x03e0;
783 p_vout->output.i_bmask = 0x001f;
785 p_vout->output.i_rmask = 0x001f;
786 p_vout->output.i_gmask = 0x03e0;
787 p_vout->output.i_bmask = 0x7c00;
794 /* Try to initialize up to MAX_DIRECTBUFFERS direct buffers */
795 while( I_OUTPUTPICTURES < MAX_DIRECTBUFFERS )
799 /* Find an empty picture slot */
800 for( i_index = 0 ; i_index < VOUT_MAX_PICTURES ; i_index++ )
802 if( p_vout->p_picture[ i_index ].i_status == FREE_PICTURE )
804 p_pic = p_vout->p_picture + i_index;
809 /* Allocate the picture */
810 if( p_pic == NULL || NewPicture( p_vout, p_pic ) )
815 p_pic->i_status = DESTROYED_PICTURE;
816 p_pic->i_type = DIRECT_PICTURE;
818 PP_OUTPUTPICTURE[ I_OUTPUTPICTURES ] = p_pic;
823 if( p_vout->output.i_chroma == VLC_CODEC_YV12 )
825 /* U and V inverted compared to I420
826 * Fixme: this should be handled by the vout core */
827 p_vout->output.i_chroma = VLC_CODEC_I420;
828 p_vout->fmt_out.i_chroma = VLC_CODEC_I420;
834 /*****************************************************************************
835 * DisplayVideo: displays previously rendered output
836 *****************************************************************************
837 * This function sends the currently rendered image to X11 server.
838 * (The Xv extension takes care of "double-buffering".)
839 *****************************************************************************/
840 static void DisplayVideo( vout_thread_t *p_vout, picture_t *p_pic )
842 unsigned int i_width, i_height, i_x, i_y;
844 vout_PlacePicture( p_vout, p_vout->p_sys->window.i_width,
845 p_vout->p_sys->window.i_height,
846 &i_x, &i_y, &i_width, &i_height );
848 #ifdef MODULE_NAME_IS_xvmc
849 xvmc_context_reader_lock( &p_vout->p_sys->xvmc_lock );
851 vlc_xxmc_t *xxmc = &p_pic->p_sys->xxmc_data;
852 if( !xxmc->decoded ||
853 !xxmc_xvmc_surface_valid( p_vout, p_pic->p_sys->xvmc_surf ) )
855 msg_Dbg( p_vout, "DisplayVideo decoded=%d\tsurfacevalid=%d",
857 xxmc_xvmc_surface_valid( p_vout, p_pic->p_sys->xvmc_surf ) );
858 xvmc_context_reader_unlock( &p_vout->p_sys->xvmc_lock );
862 int src_width = p_vout->output.i_width;
863 int src_height = p_vout->output.i_height;
866 if( p_vout->p_sys->xvmc_crop_style == 1 )
873 else if( p_vout->p_sys->xvmc_crop_style == 2 )
880 else if( p_vout->p_sys->xvmc_crop_style == 3 )
894 if( p_vout->p_sys->xvmc_deinterlace_method > 0 )
895 { /* BOB DEINTERLACE */
896 if( (p_pic->p_sys->nb_display == 0) ||
897 (p_vout->p_sys->xvmc_deinterlace_method == 1) )
899 first_field = (p_pic->b_top_field_first) ?
900 XVMC_BOTTOM_FIELD : XVMC_TOP_FIELD;
904 first_field = (p_pic->b_top_field_first) ?
905 XVMC_TOP_FIELD : XVMC_BOTTOM_FIELD;
910 first_field = XVMC_FRAME_PICTURE;
913 XVMCLOCKDISPLAY( p_vout->p_sys->p_display );
914 XvMCFlushSurface( p_vout->p_sys->p_display, p_pic->p_sys->xvmc_surf );
915 /* XvMCSyncSurface(p_vout->p_sys->p_display, p_picture->p_sys->xvmc_surf); */
916 XvMCPutSurface( p_vout->p_sys->p_display,
917 p_pic->p_sys->xvmc_surf,
918 p_vout->p_sys->window.video_window,
929 XVMCUNLOCKDISPLAY( p_vout->p_sys->p_display );
930 if( p_vout->p_sys->xvmc_deinterlace_method == 2 )
931 { /* BOB DEINTERLACE */
932 if( p_pic->p_sys->nb_display == 0 )/* && ((t2-t1) < 15000)) */
934 mtime_t last_date = p_pic->date;
936 vlc_mutex_lock( &p_vout->picture_lock );
937 if( !p_vout->p_sys->last_date )
939 p_pic->date += 20000;
943 p_pic->date = ((3 * p_pic->date -
944 p_vout->p_sys->last_date) / 2 );
946 p_vout->p_sys->last_date = last_date;
948 p_pic->p_sys->nb_display = 1;
949 vlc_mutex_unlock( &p_vout->picture_lock );
953 p_pic->p_sys->nb_display = 0;
957 xvmc_context_reader_unlock( &p_vout->p_sys->xvmc_lock );
960 #ifdef HAVE_SYS_SHM_H
961 if( p_vout->p_sys->i_shm_opcode )
963 /* Display rendered image using shared memory extension */
964 XvShmPutImage( p_vout->p_sys->p_display, p_vout->p_sys->i_xvport,
965 p_vout->p_sys->window.video_window,
966 p_vout->p_sys->window.gc, p_pic->p_sys->p_image,
967 p_vout->fmt_out.i_x_offset,
968 p_vout->fmt_out.i_y_offset,
969 p_vout->fmt_out.i_visible_width,
970 p_vout->fmt_out.i_visible_height,
971 0 /*dest_x*/, 0 /*dest_y*/, i_width, i_height,
972 False /* Don't put True here or you'll waste your CPU */ );
975 #endif /* HAVE_SYS_SHM_H */
977 /* Use standard XPutImage -- this is gonna be slow ! */
978 XvPutImage( p_vout->p_sys->p_display, p_vout->p_sys->i_xvport,
979 p_vout->p_sys->window.video_window,
980 p_vout->p_sys->window.gc, p_pic->p_sys->p_image,
981 p_vout->fmt_out.i_x_offset,
982 p_vout->fmt_out.i_y_offset,
983 p_vout->fmt_out.i_visible_width,
984 p_vout->fmt_out.i_visible_height,
985 0 /*dest_x*/, 0 /*dest_y*/, i_width, i_height );
988 /* Make sure the command is sent now - do NOT use XFlush !*/
989 XSync( p_vout->p_sys->p_display, False );
993 /*****************************************************************************
994 * ManageVideo: handle X11 events
995 *****************************************************************************
996 * This function should be called regularly by video output thread. It manages
997 * X11 events and allows window resizing. It returns a non null value on
999 *****************************************************************************/
1000 static int ManageVideo( vout_thread_t *p_vout )
1002 XEvent xevent; /* X11 event */
1005 #ifdef MODULE_NAME_IS_xvmc
1006 xvmc_context_reader_lock( &p_vout->p_sys->xvmc_lock );
1009 /* Handle events from the owner window */
1010 while( XCheckWindowEvent( p_vout->p_sys->p_display,
1011 p_vout->p_sys->window.owner_window->handle.xid,
1012 StructureNotifyMask, &xevent ) == True )
1014 /* ConfigureNotify event: prepare */
1015 if( xevent.type == ConfigureNotify )
1016 /* Update dimensions */
1017 XResizeWindow( p_vout->p_sys->p_display,
1018 p_vout->p_sys->window.base_window,
1019 xevent.xconfigure.width,
1020 xevent.xconfigure.height );
1023 /* Handle X11 events: ConfigureNotify events are parsed to know if the
1024 * output window's size changed, MapNotify and UnmapNotify to know if the
1025 * window is mapped (and if the display is useful), and ClientMessages
1026 * to intercept window destruction requests */
1028 while( XCheckWindowEvent( p_vout->p_sys->p_display,
1029 p_vout->p_sys->window.base_window,
1030 StructureNotifyMask |
1031 ButtonPressMask | ButtonReleaseMask |
1032 PointerMotionMask | Button1MotionMask , &xevent )
1035 /* ConfigureNotify event: prepare */
1036 if( xevent.type == ConfigureNotify )
1038 if( (unsigned int)xevent.xconfigure.width
1039 != p_vout->p_sys->window.i_width
1040 || (unsigned int)xevent.xconfigure.height
1041 != p_vout->p_sys->window.i_height )
1043 /* Update dimensions */
1044 p_vout->i_changes |= VOUT_SIZE_CHANGE;
1045 p_vout->p_sys->window.i_width = xevent.xconfigure.width;
1046 p_vout->p_sys->window.i_height = xevent.xconfigure.height;
1050 else if( xevent.type == ButtonPress )
1052 switch( ((XButtonEvent *)&xevent)->button )
1055 var_Get( p_vout, "mouse-button-down", &val );
1057 var_Set( p_vout, "mouse-button-down", val );
1059 var_SetBool( p_vout->p_libvlc, "intf-popupmenu", false );
1061 /* detect double-clicks */
1062 if( ( ((XButtonEvent *)&xevent)->time -
1063 p_vout->p_sys->i_time_button_last_pressed ) < 300 )
1065 p_vout->i_changes |= VOUT_FULLSCREEN_CHANGE;
1068 p_vout->p_sys->i_time_button_last_pressed =
1069 ((XButtonEvent *)&xevent)->time;
1072 var_Get( p_vout, "mouse-button-down", &val );
1074 var_Set( p_vout, "mouse-button-down", val );
1078 var_Get( p_vout, "mouse-button-down", &val );
1080 var_Set( p_vout, "mouse-button-down", val );
1081 var_SetBool( p_vout->p_libvlc, "intf-popupmenu", true );
1085 var_Get( p_vout, "mouse-button-down", &val );
1087 var_Set( p_vout, "mouse-button-down", val );
1091 var_Get( p_vout, "mouse-button-down", &val );
1093 var_Set( p_vout, "mouse-button-down", val );
1098 else if( xevent.type == ButtonRelease )
1100 switch( ((XButtonEvent *)&xevent)->button )
1104 var_Get( p_vout, "mouse-button-down", &val );
1106 var_Set( p_vout, "mouse-button-down", val );
1108 var_SetBool( p_vout, "mouse-clicked", true );
1114 var_Get( p_vout, "mouse-button-down", &val );
1116 var_Set( p_vout, "mouse-button-down", val );
1118 var_Get( p_vout->p_libvlc, "intf-show", &val );
1119 val.b_bool = !val.b_bool;
1120 var_Set( p_vout->p_libvlc, "intf-show", val );
1126 var_Get( p_vout, "mouse-button-down", &val );
1128 var_Set( p_vout, "mouse-button-down", val );
1134 var_Get( p_vout, "mouse-button-down", &val );
1136 var_Set( p_vout, "mouse-button-down", val );
1140 var_Get( p_vout, "mouse-button-down", &val );
1142 var_Set( p_vout, "mouse-button-down", val );
1148 else if( xevent.type == MotionNotify )
1150 unsigned int i_width, i_height, i_x, i_y;
1152 /* somewhat different use for vout_PlacePicture:
1153 * here the values are needed to give to mouse coordinates
1154 * in the original picture space */
1155 vout_PlacePicture( p_vout, p_vout->p_sys->window.i_width,
1156 p_vout->p_sys->window.i_height,
1157 &i_x, &i_y, &i_width, &i_height );
1159 /* Compute the x coordinate and check if the value is
1160 in [0,p_vout->fmt_in.i_visible_width] */
1161 val.i_int = ( xevent.xmotion.x - i_x ) *
1162 p_vout->fmt_in.i_visible_width / i_width +
1163 p_vout->fmt_in.i_x_offset;
1165 if( (int)(xevent.xmotion.x - i_x) < 0 )
1167 else if( (unsigned int)val.i_int > p_vout->fmt_in.i_visible_width )
1168 val.i_int = p_vout->fmt_in.i_visible_width;
1170 var_Set( p_vout, "mouse-x", val );
1172 /* compute the y coordinate and check if the value is
1173 in [0,p_vout->fmt_in.i_visible_height] */
1174 val.i_int = ( xevent.xmotion.y - i_y ) *
1175 p_vout->fmt_in.i_visible_height / i_height +
1176 p_vout->fmt_in.i_y_offset;
1178 if( (int)(xevent.xmotion.y - i_y) < 0 )
1180 else if( (unsigned int)val.i_int > p_vout->fmt_in.i_visible_height )
1181 val.i_int = p_vout->fmt_in.i_visible_height;
1183 var_Set( p_vout, "mouse-y", val );
1185 var_SetBool( p_vout, "mouse-moved", true );
1187 p_vout->p_sys->i_time_mouse_last_moved = mdate();
1188 if( ! p_vout->p_sys->b_mouse_pointer_visible )
1190 ToggleCursor( p_vout );
1193 else if( xevent.type == ReparentNotify /* XXX: why do we get this? */
1194 || xevent.type == MapNotify
1195 || xevent.type == UnmapNotify )
1197 /* Ignore these events */
1199 else /* Other events */
1201 msg_Warn( p_vout, "unhandled event %d received", xevent.type );
1205 /* Handle events for video output sub-window */
1206 while( XCheckWindowEvent( p_vout->p_sys->p_display,
1207 p_vout->p_sys->window.video_window,
1208 ExposureMask, &xevent ) == True );
1210 /* ClientMessage event - only WM_PROTOCOLS with WM_DELETE_WINDOW data
1211 * are handled - according to the man pages, the format is always 32
1213 while( XCheckTypedEvent( p_vout->p_sys->p_display,
1214 ClientMessage, &xevent ) )
1216 if( (xevent.xclient.message_type == p_vout->p_sys->window.wm_protocols)
1217 && ((Atom)xevent.xclient.data.l[0]
1218 == p_vout->p_sys->window.wm_delete_window ) )
1220 /* the user wants to close the window */
1221 playlist_t * p_playlist = pl_Hold( p_vout );
1222 if( p_playlist != NULL )
1224 playlist_Stop( p_playlist );
1225 pl_Release( p_vout );
1233 if ( p_vout->i_changes & VOUT_FULLSCREEN_CHANGE )
1235 /* Update the object variable and trigger callback */
1236 var_SetBool( p_vout, "fullscreen", !p_vout->b_fullscreen );
1238 ToggleFullScreen( p_vout );
1239 p_vout->i_changes &= ~VOUT_FULLSCREEN_CHANGE;
1242 /* autoscale toggle */
1243 if( p_vout->i_changes & VOUT_SCALE_CHANGE )
1245 p_vout->i_changes &= ~VOUT_SCALE_CHANGE;
1247 p_vout->b_autoscale = var_GetBool( p_vout, "autoscale" );
1248 p_vout->i_zoom = ZOOM_FP_FACTOR;
1250 p_vout->i_changes |= VOUT_SIZE_CHANGE;
1253 /* scaling factor */
1254 if( p_vout->i_changes & VOUT_ZOOM_CHANGE )
1256 p_vout->i_changes &= ~VOUT_ZOOM_CHANGE;
1258 p_vout->b_autoscale = false;
1260 (int)( ZOOM_FP_FACTOR * var_GetFloat( p_vout, "scale" ) );
1262 p_vout->i_changes |= VOUT_SIZE_CHANGE;
1265 if( p_vout->i_changes & VOUT_CROP_CHANGE ||
1266 p_vout->i_changes & VOUT_ASPECT_CHANGE )
1268 p_vout->i_changes &= ~VOUT_CROP_CHANGE;
1269 p_vout->i_changes &= ~VOUT_ASPECT_CHANGE;
1271 p_vout->fmt_out.i_x_offset = p_vout->fmt_in.i_x_offset;
1272 p_vout->fmt_out.i_y_offset = p_vout->fmt_in.i_y_offset;
1273 p_vout->fmt_out.i_visible_width = p_vout->fmt_in.i_visible_width;
1274 p_vout->fmt_out.i_visible_height = p_vout->fmt_in.i_visible_height;
1275 p_vout->fmt_out.i_aspect = p_vout->fmt_in.i_aspect;
1276 p_vout->fmt_out.i_sar_num = p_vout->fmt_in.i_sar_num;
1277 p_vout->fmt_out.i_sar_den = p_vout->fmt_in.i_sar_den;
1278 p_vout->output.i_aspect = p_vout->fmt_in.i_aspect;
1280 p_vout->i_changes |= VOUT_SIZE_CHANGE;
1286 * (Needs to be placed after VOUT_FULLSREEN_CHANGE because we can activate
1287 * the size flag inside the fullscreen routine)
1289 if( p_vout->i_changes & VOUT_SIZE_CHANGE )
1291 unsigned int i_width, i_height, i_x, i_y;
1293 p_vout->i_changes &= ~VOUT_SIZE_CHANGE;
1295 vout_PlacePicture( p_vout, p_vout->p_sys->window.i_width,
1296 p_vout->p_sys->window.i_height,
1297 &i_x, &i_y, &i_width, &i_height );
1299 XMoveResizeWindow( p_vout->p_sys->p_display,
1300 p_vout->p_sys->window.video_window,
1301 i_x, i_y, i_width, i_height );
1304 /* cursor hiding depending on --vout-event option
1306 * value = 1 (Fullsupport) (default value)
1307 * or value = 2 (Fullscreen-Only) and condition met
1309 bool b_vout_event = ( ( p_vout->p_sys->i_vout_event == 1 )
1310 || ( p_vout->p_sys->i_vout_event == 2 && p_vout->b_fullscreen )
1313 /* Autohide Cursour */
1314 if( mdate() - p_vout->p_sys->i_time_mouse_last_moved >
1315 p_vout->p_sys->i_mouse_hide_timeout )
1317 /* Hide the mouse automatically */
1318 if( b_vout_event && p_vout->p_sys->b_mouse_pointer_visible )
1320 ToggleCursor( p_vout );
1324 #ifdef MODULE_NAME_IS_xvmc
1325 xvmc_context_reader_unlock( &p_vout->p_sys->xvmc_lock );
1329 if ( p_vout->p_sys->p_octx != NULL ) {
1330 if ( p_vout->p_sys->i_backlight_on_counter == i_backlight_on_interval ) {
1331 if ( osso_display_blanking_pause( p_vout->p_sys->p_octx ) != OSSO_OK ) {
1332 msg_Err( p_vout, "Could not disable backlight blanking" );
1334 msg_Dbg( p_vout, "Backlight blanking disabled" );
1336 p_vout->p_sys->i_backlight_on_counter = 0;
1338 p_vout->p_sys->i_backlight_on_counter ++;
1345 #if !defined( MODULE_NAME_IS_glx )
1346 /*****************************************************************************
1347 * EndVideo: terminate X11 video thread output method
1348 *****************************************************************************
1349 * Destroy the X11 XImages created by Init. It is called at the end of
1350 * the thread, but also each time the window is resized.
1351 *****************************************************************************/
1352 static void EndVideo( vout_thread_t *p_vout )
1356 /* Free the direct buffers we allocated */
1357 for( i_index = I_OUTPUTPICTURES ; i_index ; )
1360 FreePicture( p_vout, PP_OUTPUTPICTURE[ i_index ] );
1365 /* following functions are local */
1367 /*****************************************************************************
1368 * CreateWindow: open and set-up X11 main window
1369 *****************************************************************************/
1370 static int CreateWindow( vout_thread_t *p_vout, x11_window_t *p_win )
1372 XSizeHints xsize_hints;
1373 XSetWindowAttributes xwindow_attributes;
1374 XGCValues xgcvalues;
1377 bool b_map_notify = false;
1379 /* Prepare window manager hints and properties */
1380 p_win->wm_protocols =
1381 XInternAtom( p_vout->p_sys->p_display, "WM_PROTOCOLS", True );
1382 p_win->wm_delete_window =
1383 XInternAtom( p_vout->p_sys->p_display, "WM_DELETE_WINDOW", True );
1385 /* Never have a 0-pixel-wide window */
1386 xsize_hints.min_width = 2;
1387 xsize_hints.min_height = 1;
1389 /* Prepare window attributes */
1390 xwindow_attributes.backing_store = Always; /* save the hidden part */
1391 xwindow_attributes.background_pixel = BlackPixel(p_vout->p_sys->p_display,
1392 p_vout->p_sys->i_screen);
1393 xwindow_attributes.event_mask = ExposureMask | StructureNotifyMask;
1396 vout_window_cfg_t wnd_cfg;
1397 memset( &wnd_cfg, 0, sizeof(wnd_cfg) );
1398 wnd_cfg.type = VOUT_WINDOW_TYPE_XID;
1399 wnd_cfg.x = p_win->i_x;
1400 wnd_cfg.y = p_win->i_y;
1401 wnd_cfg.width = p_win->i_width;
1402 wnd_cfg.height = p_win->i_height;
1404 p_win->owner_window = vout_window_New( VLC_OBJECT(p_vout), NULL, &wnd_cfg );
1405 if( !p_win->owner_window )
1406 return VLC_EGENERIC;
1407 xsize_hints.base_width = xsize_hints.width = p_win->i_width;
1408 xsize_hints.base_height = xsize_hints.height = p_win->i_height;
1409 xsize_hints.flags = PSize | PMinSize;
1411 if( p_win->i_x >=0 || p_win->i_y >= 0 )
1413 xsize_hints.x = p_win->i_x;
1414 xsize_hints.y = p_win->i_y;
1415 xsize_hints.flags |= PPosition;
1418 /* Select events we are interested in. */
1419 XSelectInput( p_vout->p_sys->p_display,
1420 p_win->owner_window->handle.xid, StructureNotifyMask );
1422 /* Get the parent window's geometry information */
1423 XGetGeometry( p_vout->p_sys->p_display,
1424 p_win->owner_window->handle.xid,
1425 &(Window){ 0 }, &(int){ 0 }, &(int){ 0 },
1428 &(unsigned){ 0 }, &(unsigned){ 0 } );
1430 /* From man XSelectInput: only one client at a time can select a
1431 * ButtonPress event, so we need to open a new window anyway. */
1432 p_win->base_window =
1433 XCreateWindow( p_vout->p_sys->p_display,
1434 p_win->owner_window->handle.xid,
1436 p_win->i_width, p_win->i_height,
1438 0, CopyFromParent, 0,
1439 CWBackingStore | CWBackPixel | CWEventMask,
1440 &xwindow_attributes );
1443 if( (p_win->wm_protocols == None) /* use WM_DELETE_WINDOW */
1444 || (p_win->wm_delete_window == None)
1445 || !XSetWMProtocols( p_vout->p_sys->p_display, p_win->base_window,
1446 &p_win->wm_delete_window, 1 ) )
1448 /* WM_DELETE_WINDOW is not supported by window manager */
1449 msg_Warn( p_vout, "missing or bad window manager" );
1452 /* Creation of a graphic context that doesn't generate a GraphicsExpose
1453 * event when using functions like XCopyArea */
1454 xgcvalues.graphics_exposures = False;
1455 p_win->gc = XCreateGC( p_vout->p_sys->p_display,
1457 GCGraphicsExposures, &xgcvalues );
1459 /* Wait till the window is mapped */
1460 XMapWindow( p_vout->p_sys->p_display, p_win->base_window );
1463 XWindowEvent( p_vout->p_sys->p_display, p_win->base_window,
1464 SubstructureNotifyMask | StructureNotifyMask, &xevent);
1465 if( (xevent.type == MapNotify)
1466 && (xevent.xmap.window == p_win->base_window) )
1468 b_map_notify = true;
1470 else if( (xevent.type == ConfigureNotify)
1471 && (xevent.xconfigure.window == p_win->base_window) )
1473 p_win->i_width = xevent.xconfigure.width;
1474 p_win->i_height = xevent.xconfigure.height;
1476 } while( !b_map_notify );
1478 /* key and mouse events handling depending on --vout-event option
1480 * value = 1 (Fullsupport) (default value)
1481 * or value = 2 (Fullscreen-Only) and condition met
1483 bool b_vout_event = ( ( p_vout->p_sys->i_vout_event == 1 )
1484 || ( p_vout->p_sys->i_vout_event == 2 && p_vout->b_fullscreen )
1487 XSelectInput( p_vout->p_sys->p_display, p_win->base_window,
1488 StructureNotifyMask |
1489 ButtonPressMask | ButtonReleaseMask |
1490 PointerMotionMask );
1492 /* Create video output sub-window. */
1493 p_win->video_window = XCreateSimpleWindow(
1494 p_vout->p_sys->p_display,
1495 p_win->base_window, 0, 0,
1496 p_win->i_width, p_win->i_height,
1498 BlackPixel( p_vout->p_sys->p_display,
1499 p_vout->p_sys->i_screen ),
1500 WhitePixel( p_vout->p_sys->p_display,
1501 p_vout->p_sys->i_screen ) );
1503 XSetWindowBackground( p_vout->p_sys->p_display, p_win->video_window,
1504 BlackPixel( p_vout->p_sys->p_display,
1505 p_vout->p_sys->i_screen ) );
1507 XMapWindow( p_vout->p_sys->p_display, p_win->video_window );
1508 XSelectInput( p_vout->p_sys->p_display, p_win->video_window,
1511 /* make sure the video window will be centered in the next ManageVideo() */
1512 p_vout->i_changes |= VOUT_SIZE_CHANGE;
1514 /* If the cursor was formerly blank than blank it again */
1515 if( !p_vout->p_sys->b_mouse_pointer_visible )
1517 ToggleCursor( p_vout );
1518 ToggleCursor( p_vout );
1521 /* Do NOT use XFlush here ! */
1522 XSync( p_vout->p_sys->p_display, False );
1527 /*****************************************************************************
1528 * DestroyWindow: destroy the window
1529 *****************************************************************************
1531 *****************************************************************************/
1532 static void DestroyWindow( vout_thread_t *p_vout, x11_window_t *p_win )
1534 /* Do NOT use XFlush here ! */
1535 XSync( p_vout->p_sys->p_display, False );
1537 if( p_win->video_window != None )
1538 XDestroyWindow( p_vout->p_sys->p_display, p_win->video_window );
1540 XFreeGC( p_vout->p_sys->p_display, p_win->gc );
1542 XUnmapWindow( p_vout->p_sys->p_display, p_win->base_window );
1543 XDestroyWindow( p_vout->p_sys->p_display, p_win->base_window );
1545 /* make sure base window is destroyed before proceeding further */
1546 bool b_destroy_notify = false;
1550 XWindowEvent( p_vout->p_sys->p_display, p_win->base_window,
1551 SubstructureNotifyMask | StructureNotifyMask, &xevent);
1552 if( (xevent.type == DestroyNotify)
1553 && (xevent.xmap.window == p_win->base_window) )
1555 b_destroy_notify = true;
1557 } while( !b_destroy_notify );
1559 vout_window_Delete( p_win->owner_window );
1562 /*****************************************************************************
1563 * NewPicture: allocate a picture
1564 *****************************************************************************
1565 * Returns 0 on success, -1 otherwise
1566 *****************************************************************************/
1567 #if !defined(MODULE_NAME_IS_glx)
1568 static int NewPicture( vout_thread_t *p_vout, picture_t *p_pic )
1572 /* We know the chroma, allocate a buffer which will be used
1573 * directly by the decoder */
1574 p_pic->p_sys = malloc( sizeof( picture_sys_t ) );
1576 if( p_pic->p_sys == NULL )
1581 #ifdef MODULE_NAME_IS_xvmc
1582 p_pic->p_sys->p_vout = p_vout;
1583 p_pic->p_sys->xvmc_surf = NULL;
1584 p_pic->p_sys->xxmc_data.decoded = 0;
1585 p_pic->p_sys->xxmc_data.proc_xxmc_update_frame = xxmc_do_update_frame;
1586 // p_pic->p_accel_data = &p_pic->p_sys->xxmc_data;
1587 p_pic->p_sys->nb_display = 0;
1590 /* Fill in picture_t fields */
1591 if( picture_Setup( p_pic, p_vout->output.i_chroma,
1592 p_vout->output.i_width, p_vout->output.i_height,
1593 p_vout->output.i_aspect ) )
1596 #ifdef HAVE_SYS_SHM_H
1597 if( p_vout->p_sys->i_shm_opcode )
1599 /* Create image using XShm extension */
1600 p_pic->p_sys->p_image =
1601 CreateShmImage( p_vout, p_vout->p_sys->p_display,
1602 p_vout->p_sys->i_xvport,
1603 VLC2X11_FOURCC(p_vout->output.i_chroma),
1604 &p_pic->p_sys->shminfo,
1605 p_vout->output.i_width, p_vout->output.i_height );
1608 if( !p_vout->p_sys->i_shm_opcode || !p_pic->p_sys->p_image )
1609 #endif /* HAVE_SYS_SHM_H */
1611 /* Create image without XShm extension */
1612 p_pic->p_sys->p_image =
1613 CreateImage( p_vout, p_vout->p_sys->p_display,
1614 p_vout->p_sys->i_xvport,
1615 VLC2X11_FOURCC(p_vout->output.i_chroma),
1616 p_pic->format.i_bits_per_pixel,
1617 p_vout->output.i_width, p_vout->output.i_height );
1619 #ifdef HAVE_SYS_SHM_H
1620 if( p_pic->p_sys->p_image && p_vout->p_sys->i_shm_opcode )
1622 msg_Warn( p_vout, "couldn't create SHM image, disabling SHM" );
1623 p_vout->p_sys->i_shm_opcode = 0;
1625 #endif /* HAVE_SYS_SHM_H */
1628 if( p_pic->p_sys->p_image == NULL )
1630 free( p_pic->p_sys );
1634 switch( p_vout->output.i_chroma )
1636 case VLC_CODEC_I420:
1637 case VLC_CODEC_YV12:
1638 case VLC_CODEC_Y211:
1639 case VLC_CODEC_YUYV:
1640 case VLC_CODEC_UYVY:
1641 case VLC_CODEC_RGB15:
1642 case VLC_CODEC_RGB16:
1643 case VLC_CODEC_RGB24: /* Fixme: pixel pitch == 4 ? */
1644 case VLC_CODEC_RGB32:
1646 for( i_plane = 0; i_plane < p_pic->p_sys->p_image->num_planes;
1649 p_pic->p[i_plane].p_pixels = (uint8_t*)p_pic->p_sys->p_image->data
1650 + p_pic->p_sys->p_image->offsets[i_plane];
1651 p_pic->p[i_plane].i_pitch =
1652 p_pic->p_sys->p_image->pitches[i_plane];
1654 if( p_vout->output.i_chroma == VLC_CODEC_YV12 )
1656 /* U and V inverted compared to I420
1657 * Fixme: this should be handled by the vout core */
1658 p_pic->U_PIXELS = (uint8_t*)p_pic->p_sys->p_image->data
1659 + p_pic->p_sys->p_image->offsets[2];
1660 p_pic->V_PIXELS = (uint8_t*)p_pic->p_sys->p_image->data
1661 + p_pic->p_sys->p_image->offsets[1];
1667 /* Unknown chroma, tell the guy to get lost */
1668 IMAGE_FREE( p_pic->p_sys->p_image );
1669 free( p_pic->p_sys );
1670 msg_Err( p_vout, "never heard of chroma 0x%.8x (%4.4s)",
1671 p_vout->output.i_chroma, (char*)&p_vout->output.i_chroma );
1672 p_pic->i_planes = 0;
1678 /*****************************************************************************
1679 * FreePicture: destroy a picture allocated with NewPicture
1680 *****************************************************************************
1681 * Destroy XImage AND associated data. If using Shm, detach shared memory
1682 * segment from server and process, then free it. The XDestroyImage manpage
1683 * says that both the image structure _and_ the data pointed to by the
1684 * image structure are freed, so no need to free p_image->data.
1685 *****************************************************************************/
1686 static void FreePicture( vout_thread_t *p_vout, picture_t *p_pic )
1688 /* The order of operations is correct */
1689 #ifdef HAVE_SYS_SHM_H
1690 if( p_vout->p_sys->i_shm_opcode )
1692 XShmDetach( p_vout->p_sys->p_display, &p_pic->p_sys->shminfo );
1693 IMAGE_FREE( p_pic->p_sys->p_image );
1695 shmctl( p_pic->p_sys->shminfo.shmid, IPC_RMID, 0 );
1696 if( shmdt( p_pic->p_sys->shminfo.shmaddr ) )
1698 msg_Err( p_vout, "cannot detach shared memory (%m)" );
1704 IMAGE_FREE( p_pic->p_sys->p_image );
1707 #ifdef MODULE_NAME_IS_xvmc
1708 if( p_pic->p_sys->xvmc_surf != NULL )
1710 xxmc_xvmc_free_surface(p_vout , p_pic->p_sys->xvmc_surf);
1711 p_pic->p_sys->xvmc_surf = NULL;
1715 /* Do NOT use XFlush here ! */
1716 XSync( p_vout->p_sys->p_display, False );
1718 free( p_pic->p_sys );
1720 #endif /* !MODULE_NAME_IS_glx */
1722 /*****************************************************************************
1723 * ToggleFullScreen: Enable or disable full screen mode
1724 *****************************************************************************
1725 * This function will switch between fullscreen and window mode.
1726 *****************************************************************************/
1727 static void ToggleFullScreen ( vout_thread_t *p_vout )
1729 p_vout->b_fullscreen = !p_vout->b_fullscreen;
1730 vout_window_SetFullScreen( p_vout->p_sys->window.owner_window,
1731 p_vout->b_fullscreen );
1734 if( p_vout->b_fullscreen )
1735 EnablePixelDoubling( p_vout );
1737 DisablePixelDoubling( p_vout );
1741 /*****************************************************************************
1742 * EnableXScreenSaver: enable screen saver
1743 *****************************************************************************
1744 * This function enables the screen saver on a display after it has been
1745 * disabled by XDisableScreenSaver.
1746 * FIXME: what happens if multiple vlc sessions are running at the same
1748 *****************************************************************************/
1749 static void EnableXScreenSaver( vout_thread_t *p_vout )
1751 #ifdef DPMSINFO_IN_DPMS_H
1755 if( p_vout->p_sys->i_ss_timeout )
1757 XSetScreenSaver( p_vout->p_sys->p_display, p_vout->p_sys->i_ss_timeout,
1758 p_vout->p_sys->i_ss_interval,
1759 p_vout->p_sys->i_ss_blanking,
1760 p_vout->p_sys->i_ss_exposure );
1763 /* Restore DPMS settings */
1764 #ifdef DPMSINFO_IN_DPMS_H
1765 if( DPMSQueryExtension( p_vout->p_sys->p_display, &dummy, &dummy ) )
1767 if( p_vout->p_sys->b_ss_dpms )
1769 DPMSEnable( p_vout->p_sys->p_display );
1775 /*****************************************************************************
1776 * DisableXScreenSaver: disable screen saver
1777 *****************************************************************************
1778 * See XEnableXScreenSaver
1779 *****************************************************************************/
1780 static void DisableXScreenSaver( vout_thread_t *p_vout )
1782 #ifdef DPMSINFO_IN_DPMS_H
1786 /* Save screen saver information */
1787 XGetScreenSaver( p_vout->p_sys->p_display, &p_vout->p_sys->i_ss_timeout,
1788 &p_vout->p_sys->i_ss_interval,
1789 &p_vout->p_sys->i_ss_blanking,
1790 &p_vout->p_sys->i_ss_exposure );
1792 /* Disable screen saver */
1793 if( p_vout->p_sys->i_ss_timeout )
1795 XSetScreenSaver( p_vout->p_sys->p_display, 0,
1796 p_vout->p_sys->i_ss_interval,
1797 p_vout->p_sys->i_ss_blanking,
1798 p_vout->p_sys->i_ss_exposure );
1802 #ifdef DPMSINFO_IN_DPMS_H
1803 if( DPMSQueryExtension( p_vout->p_sys->p_display, &dummy, &dummy ) )
1806 /* Save DPMS current state */
1807 DPMSInfo( p_vout->p_sys->p_display, &unused,
1808 &p_vout->p_sys->b_ss_dpms );
1809 DPMSDisable( p_vout->p_sys->p_display );
1814 /*****************************************************************************
1815 * CreateCursor: create a blank mouse pointer
1816 *****************************************************************************/
1817 static void CreateCursor( vout_thread_t *p_vout )
1819 XColor cursor_color;
1821 p_vout->p_sys->cursor_pixmap =
1822 XCreatePixmap( p_vout->p_sys->p_display,
1823 DefaultRootWindow( p_vout->p_sys->p_display ),
1826 XParseColor( p_vout->p_sys->p_display,
1827 XCreateColormap( p_vout->p_sys->p_display,
1829 p_vout->p_sys->p_display ),
1831 p_vout->p_sys->p_display,
1832 p_vout->p_sys->i_screen ),
1834 "black", &cursor_color );
1836 p_vout->p_sys->blank_cursor =
1837 XCreatePixmapCursor( p_vout->p_sys->p_display,
1838 p_vout->p_sys->cursor_pixmap,
1839 p_vout->p_sys->cursor_pixmap,
1840 &cursor_color, &cursor_color, 1, 1 );
1843 /*****************************************************************************
1844 * DestroyCursor: destroy the blank mouse pointer
1845 *****************************************************************************/
1846 static void DestroyCursor( vout_thread_t *p_vout )
1848 XFreePixmap( p_vout->p_sys->p_display, p_vout->p_sys->cursor_pixmap );
1851 /*****************************************************************************
1852 * ToggleCursor: hide or show the mouse pointer
1853 *****************************************************************************
1854 * This function hides the X pointer if it is visible by setting the pointer
1855 * sprite to a blank one. To show it again, we disable the sprite.
1856 *****************************************************************************/
1857 static void ToggleCursor( vout_thread_t *p_vout )
1859 if( p_vout->p_sys->b_mouse_pointer_visible )
1861 XDefineCursor( p_vout->p_sys->p_display,
1862 p_vout->p_sys->window.base_window,
1863 p_vout->p_sys->blank_cursor );
1864 p_vout->p_sys->b_mouse_pointer_visible = 0;
1868 XUndefineCursor( p_vout->p_sys->p_display,
1869 p_vout->p_sys->window.base_window );
1870 p_vout->p_sys->b_mouse_pointer_visible = 1;
1874 #if defined(MODULE_NAME_IS_xvmc)
1875 /*****************************************************************************
1876 * XVideoGetPort: get YUV12 port
1877 *****************************************************************************/
1878 static int XVideoGetPort( vout_thread_t *p_vout,
1879 vlc_fourcc_t i_chroma, picture_heap_t *p_heap )
1881 XvAdaptorInfo *p_adaptor;
1883 unsigned int i_adaptor, i_num_adaptors;
1884 int i_requested_adaptor;
1885 int i_selected_port;
1887 switch( XvQueryExtension( p_vout->p_sys->p_display, &i, &i, &i, &i, &i ) )
1892 case XvBadExtension:
1893 msg_Warn( p_vout, "XvBadExtension" );
1897 msg_Warn( p_vout, "XvBadAlloc" );
1901 msg_Warn( p_vout, "XvQueryExtension failed" );
1905 switch( XvQueryAdaptors( p_vout->p_sys->p_display,
1906 DefaultRootWindow( p_vout->p_sys->p_display ),
1907 &i_num_adaptors, &p_adaptor ) )
1912 case XvBadExtension:
1913 msg_Warn( p_vout, "XvBadExtension for XvQueryAdaptors" );
1917 msg_Warn( p_vout, "XvBadAlloc for XvQueryAdaptors" );
1921 msg_Warn( p_vout, "XvQueryAdaptors failed" );
1925 i_selected_port = -1;
1926 i_requested_adaptor = config_GetInt( p_vout, "xvmc-adaptor" );
1927 for( i_adaptor = 0; i_adaptor < i_num_adaptors; ++i_adaptor )
1929 XvImageFormatValues *p_formats;
1930 int i_format, i_num_formats;
1933 /* If we requested an adaptor and it's not this one, we aren't
1935 if( i_requested_adaptor != -1 && ((int)i_adaptor != i_requested_adaptor) )
1940 /* If the adaptor doesn't have the required properties, skip it */
1941 if( !( p_adaptor[ i_adaptor ].type & XvInputMask ) ||
1942 !( p_adaptor[ i_adaptor ].type & XvImageMask ) )
1947 /* Check that adaptor supports our requested format... */
1948 p_formats = XvListImageFormats( p_vout->p_sys->p_display,
1949 p_adaptor[i_adaptor].base_id,
1953 i_format < i_num_formats && ( i_selected_port == -1 );
1956 XvAttribute *p_attr;
1957 int i_attr, i_num_attributes;
1958 Atom autopaint = None, colorkey = None;
1960 /* If this is not the format we want, or at least a
1961 * similar one, forget it */
1962 if( !vout_ChromaCmp( p_formats[ i_format ].id, i_chroma ) )
1967 /* Look for the first available port supporting this format */
1968 for( i_port = p_adaptor[i_adaptor].base_id;
1969 ( i_port < (int)(p_adaptor[i_adaptor].base_id
1970 + p_adaptor[i_adaptor].num_ports) )
1971 && ( i_selected_port == -1 );
1974 if( XvGrabPort( p_vout->p_sys->p_display, i_port, CurrentTime )
1977 i_selected_port = i_port;
1978 p_heap->i_chroma = p_formats[ i_format ].id;
1979 #if XvVersion > 2 || ( XvVersion == 2 && XvRevision >= 2 )
1980 p_heap->i_rmask = p_formats[ i_format ].red_mask;
1981 p_heap->i_gmask = p_formats[ i_format ].green_mask;
1982 p_heap->i_bmask = p_formats[ i_format ].blue_mask;
1987 /* If no free port was found, forget it */
1988 if( i_selected_port == -1 )
1993 /* If we found a port, print information about it */
1994 msg_Dbg( p_vout, "adaptor %i, port %i, format 0x%x (%4.4s) %s",
1995 i_adaptor, i_selected_port, p_formats[ i_format ].id,
1996 (char *)&p_formats[ i_format ].id,
1997 ( p_formats[ i_format ].format == XvPacked ) ?
1998 "packed" : "planar" );
2000 /* Use XV_AUTOPAINT_COLORKEY if supported, otherwise we will
2001 * manually paint the colour key */
2002 p_attr = XvQueryPortAttributes( p_vout->p_sys->p_display,
2004 &i_num_attributes );
2006 for( i_attr = 0; i_attr < i_num_attributes; i_attr++ )
2008 if( !strcmp( p_attr[i_attr].name, "XV_AUTOPAINT_COLORKEY" ) )
2010 autopaint = XInternAtom( p_vout->p_sys->p_display,
2011 "XV_AUTOPAINT_COLORKEY", False );
2012 XvSetPortAttribute( p_vout->p_sys->p_display,
2013 i_selected_port, autopaint, 1 );
2015 if( !strcmp( p_attr[i_attr].name, "XV_COLORKEY" ) )
2017 /* Find out the default colour key */
2018 colorkey = XInternAtom( p_vout->p_sys->p_display,
2019 "XV_COLORKEY", False );
2020 XvGetPortAttribute( p_vout->p_sys->p_display,
2021 i_selected_port, colorkey,
2022 &p_vout->p_sys->i_colourkey );
2025 p_vout->p_sys->b_paint_colourkey =
2026 autopaint == None && colorkey != None;
2028 if( p_attr != NULL )
2034 if( p_formats != NULL )
2041 if( i_num_adaptors > 0 )
2043 XvFreeAdaptorInfo( p_adaptor );
2046 if( i_selected_port == -1 )
2048 int i_chroma_tmp = X112VLC_FOURCC( i_chroma );
2049 if( i_requested_adaptor == -1 )
2051 msg_Warn( p_vout, "no free XVideo port found for format "
2052 "0x%.8x (%4.4s)", i_chroma_tmp, (char*)&i_chroma_tmp );
2056 msg_Warn( p_vout, "XVideo adaptor %i does not have a free "
2057 "XVideo port for format 0x%.8x (%4.4s)",
2058 i_requested_adaptor, i_chroma_tmp, (char*)&i_chroma_tmp );
2062 return i_selected_port;
2066 /*****************************************************************************
2067 * InitDisplay: open and initialize X11 device
2068 *****************************************************************************
2069 * Create a window according to video output given size, and set other
2070 * properties according to the display properties.
2071 *****************************************************************************/
2072 static int InitDisplay( vout_thread_t *p_vout )
2074 #ifdef HAVE_SYS_SHM_H
2075 p_vout->p_sys->i_shm_opcode = 0;
2077 if( config_GetInt( p_vout, MODULE_STRING "-shm" ) > 0 )
2079 int major, evt, err;
2081 if( XQueryExtension( p_vout->p_sys->p_display, "MIT-SHM", &major,
2083 && XShmQueryExtension( p_vout->p_sys->p_display ) )
2084 p_vout->p_sys->i_shm_opcode = major;
2086 if( p_vout->p_sys->i_shm_opcode )
2091 XShmQueryVersion( p_vout->p_sys->p_display, &major, &minor,
2093 msg_Dbg( p_vout, "XShm video extension v%d.%d "
2094 "(with%s pixmaps, opcode: %d)",
2095 major, minor, pixmaps ? "" : "out",
2096 p_vout->p_sys->i_shm_opcode );
2099 msg_Warn( p_vout, "XShm video extension not available" );
2102 msg_Dbg( p_vout, "XShm video extension disabled" );
2108 #ifndef MODULE_NAME_IS_glx
2110 #ifdef HAVE_SYS_SHM_H
2111 /*****************************************************************************
2112 * CreateShmImage: create an XImage or XvImage using shared memory extension
2113 *****************************************************************************
2114 * Prepare an XImage or XvImage for display function.
2115 * The order of the operations respects the recommandations of the mit-shm
2116 * document by J.Corbet and K.Packard. Most of the parameters were copied from
2117 * there. See http://ftp.xfree86.org/pub/XFree86/4.0/doc/mit-shm.TXT
2118 *****************************************************************************/
2119 IMAGE_TYPE * CreateShmImage( vout_thread_t *p_vout,
2120 Display* p_display, EXTRA_ARGS_SHM,
2121 int i_width, int i_height )
2123 IMAGE_TYPE *p_image;
2126 /* Create XImage / XvImage */
2127 #if defined(MODULE_NAME_IS_xvmc)
2128 p_image = XvShmCreateImage( p_display, i_xvport, i_chroma, 0,
2129 i_width, i_height, p_shm );
2131 if( p_image == NULL )
2133 msg_Err( p_vout, "image creation failed" );
2137 /* For too big image, the buffer returned is sometimes too small, prevent
2138 * VLC to segfault because of it
2139 * FIXME is it normal ? Is there a way to detect it
2140 * before (XvQueryBestSize did not) ? */
2141 if( p_image->width < i_width || p_image->height < i_height )
2143 msg_Err( p_vout, "cannot allocate shared image data with the right size "
2144 "(%dx%d instead of %dx%d)",
2145 p_image->width, p_image->height,
2146 i_width, i_height );
2147 IMAGE_FREE( p_image );
2151 /* Allocate shared memory segment. */
2152 p_shm->shmid = shmget( IPC_PRIVATE, DATA_SIZE(p_image), IPC_CREAT | 0600 );
2153 if( p_shm->shmid < 0 )
2155 msg_Err( p_vout, "cannot allocate shared image data (%m)" );
2156 IMAGE_FREE( p_image );
2160 /* Attach shared memory segment to process (read/write) */
2161 p_shm->shmaddr = p_image->data = shmat( p_shm->shmid, 0, 0 );
2162 if(! p_shm->shmaddr )
2164 msg_Err( p_vout, "cannot attach shared memory (%m)" );
2165 IMAGE_FREE( p_image );
2166 shmctl( p_shm->shmid, IPC_RMID, 0 );
2170 /* Read-only data. We won't be using XShmGetImage */
2171 p_shm->readOnly = True;
2173 /* Attach shared memory segment to X server */
2174 XSynchronize( p_display, True );
2175 i_shm_major = p_vout->p_sys->i_shm_opcode;
2176 result = XShmAttach( p_display, p_shm );
2177 if( result == False || !i_shm_major )
2179 msg_Err( p_vout, "cannot attach shared memory to X server" );
2180 IMAGE_FREE( p_image );
2181 shmctl( p_shm->shmid, IPC_RMID, 0 );
2182 shmdt( p_shm->shmaddr );
2185 XSynchronize( p_display, False );
2187 /* Send image to X server. This instruction is required, since having
2188 * built a Shm XImage and not using it causes an error on XCloseDisplay,
2189 * and remember NOT to use XFlush ! */
2190 XSync( p_display, False );
2193 /* Mark the shm segment to be removed when there are no more
2194 * attachements, so it is automatic on process exit or after shmdt */
2195 shmctl( p_shm->shmid, IPC_RMID, 0 );
2203 /*****************************************************************************
2204 * X11ErrorHandler: replace error handler so we can intercept some of them
2205 *****************************************************************************/
2206 static int X11ErrorHandler( Display * display, XErrorEvent * event )
2210 XGetErrorText( display, event->error_code, txt, sizeof( txt ) );
2212 "[????????] x11 video output error: X11 request %u.%u failed "
2213 "with error code %u:\n %s\n",
2214 event->request_code, event->minor_code, event->error_code, txt );
2216 switch( event->request_code )
2218 case X_SetInputFocus:
2219 /* Ignore errors on XSetInputFocus()
2220 * (they happen when a window is not yet mapped) */
2224 #ifdef HAVE_SYS_SHM_H
2225 if( event->request_code == i_shm_major ) /* MIT-SHM */
2228 "[????????] x11 video output notice:"
2229 " buggy X11 server claims shared memory\n"
2230 "[????????] x11 video output notice:"
2231 " support though it does not work (OpenSSH?)\n" );
2232 return i_shm_major = 0;
2237 XSetErrorHandler(NULL);
2238 return (XSetErrorHandler(X11ErrorHandler))( display, event );
2240 /* Work-around Maemo Xvideo bug */
2245 /*****************************************************************************
2246 * Control: control facility for the vout
2247 *****************************************************************************/
2248 static int Control( vout_thread_t *p_vout, int i_query, va_list args )
2250 unsigned int i_width, i_height;
2255 i_width = va_arg( args, unsigned int );
2256 i_height = va_arg( args, unsigned int );
2257 if( !i_width ) i_width = p_vout->i_window_width;
2258 if( !i_height ) i_height = p_vout->i_window_height;
2260 return vout_window_SetSize( p_vout->p_sys->window.owner_window,
2263 case VOUT_SET_STAY_ON_TOP:
2264 return vout_window_SetOnTop( p_vout->p_sys->window.owner_window,
2265 va_arg( args, int ) );
2268 return VLC_EGENERIC;