X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=modules%2Fvideo_output%2Fxcb%2Fglx.c;h=ce8f3ae547b74e774b00c5f867933ab5bc6bc027;hb=a8d2d402fe877607701fdd829bafd0982bfe1da9;hp=49b45b3db6d43ee55f91592bd944dadc1a419341;hpb=36558eb52ea617c0c6293dc0974bc41df3fca1ba;p=vlc diff --git a/modules/video_output/xcb/glx.c b/modules/video_output/xcb/glx.c index 49b45b3db6..ce8f3ae547 100644 --- a/modules/video_output/xcb/glx.c +++ b/modules/video_output/xcb/glx.c @@ -29,13 +29,14 @@ #include #include -#include #include +#include #include #include +#include #include -#include +#include #include "../opengl.h" #include "xcb_vlc.h" @@ -48,20 +49,19 @@ static void Close (vlc_object_t *); */ vlc_module_begin () set_shortname (N_("GLX")) - set_description (N_("GLX video output (XCB)")) + set_description (N_("OpenGL GLX video output (XCB)")) set_category (CAT_VIDEO) set_subcategory (SUBCAT_VIDEO_VOUT) - set_capability ("vout display", 20) + set_capability ("vout display", 150) set_callbacks (Open, Close) - add_shortcut ("xcb-glx") - add_shortcut ("glx") - add_shortcut ("opengl") + add_shortcut ("xcb-glx", "glx", "opengl", "xid") vlc_module_end () struct vout_display_sys_t { Display *display; /* Xlib instance */ + xcb_connection_t *conn; /**< XCB connection */ vout_window_t *embed; /* VLC window (when windowed) */ xcb_cursor_t cursor; /* blank cursor */ @@ -71,18 +71,19 @@ struct vout_display_sys_t bool v1_3; /* whether GLX >= 1.3 is available */ GLXContext ctx; - vout_opengl_t gl; - vout_display_opengl_t vgl; + vlc_gl_t gl; + vout_display_opengl_t *vgl; picture_pool_t *pool; /* picture pool */ }; static picture_pool_t *Pool (vout_display_t *, unsigned); -static void PictureRender (vout_display_t *, picture_t *); -static void PictureDisplay (vout_display_t *, picture_t *); +static void PictureRender (vout_display_t *, picture_t *, subpicture_t *); +static void PictureDisplay (vout_display_t *, picture_t *, subpicture_t *); static int Control (vout_display_t *, int, va_list); static void Manage (vout_display_t *); -static void SwapBuffers (vout_opengl_t *gl); +static void SwapBuffers (vlc_gl_t *gl); +static void *GetProcAddress (vlc_gl_t *gl, const char *); static vout_window_t *MakeWindow (vout_display_t *vd) { @@ -90,6 +91,8 @@ static vout_window_t *MakeWindow (vout_display_t *vd) memset (&wnd_cfg, 0, sizeof (wnd_cfg)); wnd_cfg.type = VOUT_WINDOW_TYPE_XID; + wnd_cfg.x = var_InheritInteger (vd, "video-x"); + wnd_cfg.y = var_InheritInteger (vd, "video-y"); wnd_cfg.width = vd->cfg->display.width; wnd_cfg.height = vd->cfg->display.height; @@ -108,7 +111,7 @@ FindWindow (vout_display_t *vd, xcb_connection_t *conn, xcb_get_geometry_reply_t *geo = xcb_get_geometry_reply (conn, - xcb_get_geometry (conn, sys->embed->xid), NULL); + xcb_get_geometry (conn, sys->embed->handle.xid), NULL); if (geo == NULL) { msg_Err (vd, "parent window not valid"); @@ -183,7 +186,7 @@ static int CreateWindow (vout_display_t *vd, xcb_connection_t *conn, xcb_void_cookie_t cc, cm; cc = xcb_create_window_checked (conn, depth, sys->window, - sys->embed->xid, 0, 0, + sys->embed->handle.xid, 0, 0, width, height, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT, vid, mask, values); @@ -201,6 +204,9 @@ static int CreateWindow (vout_display_t *vd, xcb_connection_t *conn, */ static int Open (vlc_object_t *obj) { + if (!vlc_xlib_init (obj)) + return VLC_EGENERIC; + vout_display_t *vd = (vout_display_t *)obj; vout_display_sys_t *sys = malloc (sizeof (*sys)); @@ -220,23 +226,30 @@ static int Open (vlc_object_t *obj) } /* Connect to X server */ - Display *dpy = XOpenDisplay (sys->embed->x11_display); + xcb_connection_t *conn = xcb_connect (sys->embed->display.x11, NULL); + if (unlikely(xcb_connection_has_error (conn))) + { + vout_display_DeleteWindow (vd, sys->embed); + free (sys); + return VLC_EGENERIC; + } + + Display *dpy = XOpenDisplay (sys->embed->display.x11); if (dpy == NULL) { + xcb_disconnect (conn); vout_display_DeleteWindow (vd, sys->embed); free (sys); return VLC_EGENERIC; } sys->display = dpy; + sys->conn = conn; sys->ctx = NULL; - XSetEventQueueOwner (dpy, XCBOwnsEventQueue); if (!CheckGLX (vd, dpy, &sys->v1_3)) goto error; - xcb_connection_t *conn = XGetXCBConnection (dpy); - assert (conn); - RegisterMouseEvents (obj, conn, sys->embed->xid); + RegisterMouseEvents (obj, conn, sys->embed->handle.xid); /* Find window parameters */ unsigned snum; @@ -263,7 +276,8 @@ static int Open (vlc_object_t *obj) xcb_get_window_attributes_reply_t *wa = xcb_get_window_attributes_reply (conn, - xcb_get_window_attributes (conn, sys->embed->xid), NULL); + xcb_get_window_attributes (conn, sys->embed->handle.xid), + NULL); if (wa == NULL) goto error; xcb_visualid_t visual = wa->visual; @@ -273,7 +287,7 @@ static int Open (vlc_object_t *obj) GLXFBConfig *confs = glXChooseFBConfig (dpy, snum, attr, &nelem); if (confs == NULL) { - msg_Err (vd, "no GLX frame bufer configurations"); + msg_Err (vd, "no GLX frame buffer configurations"); goto error; } @@ -353,13 +367,37 @@ static int Open (vlc_object_t *obj) sys->glwin = sys->window; } + const char *glx_extensions = glXQueryExtensionsString (dpy, snum); + + bool is_swap_interval_set = false; +#ifdef GLX_SGI_swap_control + if (HasExtension (glx_extensions, "GLX_SGI_swap_control")) { + PFNGLXSWAPINTERVALSGIPROC SwapIntervalSGI = (PFNGLXSWAPINTERVALSGIPROC)GetProcAddress (NULL, "glXSwapIntervalSGI"); + if (!is_swap_interval_set && SwapIntervalSGI) + is_swap_interval_set = !SwapIntervalSGI (1); + } +#endif +#ifdef GLX_EXT_swap_control + if (HasExtension (glx_extensions, "GLX_EXT_swap_control")) { + PFNGLXSWAPINTERVALEXTPROC SwapIntervalEXT = (PFNGLXSWAPINTERVALEXTPROC)GetProcAddress (NULL, "glXSwapIntervalEXT"); + if (!is_swap_interval_set && SwapIntervalEXT) + { + SwapIntervalEXT (dpy, sys->glwin, 1); + is_swap_interval_set = true; + } + } +#endif + /* Initialize common OpenGL video display */ sys->gl.lock = NULL; sys->gl.unlock = NULL; sys->gl.swap = SwapBuffers; + sys->gl.getProcAddress = GetProcAddress; sys->gl.sys = sys; - if (vout_display_opengl_Init (&sys->vgl, &vd->fmt, &sys->gl)) + const vlc_fourcc_t *subpicture_chromas; + sys->vgl = vout_display_opengl_New (&vd->fmt, &subpicture_chromas, &sys->gl); + if (!sys->vgl) { sys->gl.sys = NULL; goto error; @@ -371,6 +409,8 @@ static int Open (vlc_object_t *obj) /* */ vout_display_info_t info = vd->info; info.has_pictures_invalid = false; + info.has_event_thread = true; + info.subpicture_chromas = subpicture_chromas; /* Setup vout_display_t once everything is fine */ vd->info = info; @@ -382,8 +422,11 @@ static int Open (vlc_object_t *obj) vd->manage = Manage; /* */ - vout_display_SendEventFullscreen (vd, false); - vout_display_SendEventDisplaySize (vd, width, height, false); + bool is_fullscreen = vd->cfg->is_fullscreen; + if (is_fullscreen && vout_window_SetFullScreen (sys->embed, true)) + is_fullscreen = false; + vout_display_SendEventFullscreen (vd, is_fullscreen); + vout_display_SendEventDisplaySize (vd, width, height, is_fullscreen); return VLC_SUCCESS; @@ -403,57 +446,74 @@ static void Close (vlc_object_t *obj) Display *dpy = sys->display; if (sys->gl.sys != NULL) - vout_display_opengl_Clean (&sys->vgl); + vout_display_opengl_Delete (sys->vgl); if (sys->ctx != NULL) { if (sys->v1_3) - { glXMakeContextCurrent (dpy, None, None, NULL); - glXDestroyWindow (dpy, sys->glwin); - } else glXMakeCurrent (dpy, None, NULL); glXDestroyContext (dpy, sys->ctx); + if (sys->v1_3) + glXDestroyWindow (dpy, sys->glwin); } XCloseDisplay (dpy); + + /* show the default cursor */ + xcb_change_window_attributes (sys->conn, sys->embed->handle.xid, + XCB_CW_CURSOR, &(uint32_t) { XCB_CURSOR_NONE }); + xcb_flush (sys->conn); + xcb_disconnect (sys->conn); + vout_display_DeleteWindow (vd, sys->embed); free (sys); } -static void SwapBuffers (vout_opengl_t *gl) +static void SwapBuffers (vlc_gl_t *gl) { vout_display_sys_t *sys = gl->sys; glXSwapBuffers (sys->display, sys->glwin); } +static void *GetProcAddress (vlc_gl_t *gl, const char *name) +{ + (void)gl; +#ifdef GLX_ARB_get_proc_address + return glXGetProcAddressARB ((const GLubyte *)name); +#else + return NULL; +#endif +} + /** * Return a direct buffer */ static picture_pool_t *Pool (vout_display_t *vd, unsigned requested_count) { vout_display_sys_t *sys = vd->sys; - (void)requested_count; if (!sys->pool) - sys->pool = vout_display_opengl_GetPool (&sys->vgl); + sys->pool = vout_display_opengl_GetPool (sys->vgl, requested_count); return sys->pool; } -static void PictureRender (vout_display_t *vd, picture_t *pic) +static void PictureRender (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture) { - vout_display_sys_t *sys = vd->sys; + vout_display_sys_t *sys = vd->sys; - vout_display_opengl_Prepare (&sys->vgl, pic); + vout_display_opengl_Prepare (sys->vgl, pic, subpicture); } -static void PictureDisplay (vout_display_t *vd, picture_t *pic) +static void PictureDisplay (vout_display_t *vd, picture_t *pic, subpicture_t *subpicture) { vout_display_sys_t *sys = vd->sys; - vout_display_opengl_Display (&sys->vgl, &vd->source); + vout_display_opengl_Display (sys->vgl, &vd->source); picture_Release (pic); + if (subpicture) + subpicture_Delete(subpicture); } static int Control (vout_display_t *vd, int query, va_list ap) @@ -480,7 +540,6 @@ static int Control (vout_display_t *vd, int query, va_list ap) case VOUT_DISPLAY_CHANGE_SOURCE_ASPECT: case VOUT_DISPLAY_CHANGE_SOURCE_CROP: { - xcb_connection_t *conn = XGetXCBConnection (sys->display); const vout_display_cfg_t *cfg; const video_format_t *source; bool is_forced = false; @@ -515,11 +574,11 @@ static int Control (vout_display_t *vd, int query, va_list ap) const uint32_t values[] = { place.x, place.y, place.width, place.height, }; xcb_void_cookie_t ck = - xcb_configure_window_checked (conn, sys->window, + xcb_configure_window_checked (sys->conn, sys->window, XCB_CONFIG_WINDOW_X | XCB_CONFIG_WINDOW_Y | XCB_CONFIG_WINDOW_WIDTH | XCB_CONFIG_WINDOW_HEIGHT, values); - if (CheckError (vd, conn, "cannot resize X11 window", ck)) + if (CheckError (vd, sys->conn, "cannot resize X11 window", ck)) return VLC_EGENERIC; glViewport (0, 0, place.width, place.height); @@ -529,14 +588,14 @@ static int Control (vout_display_t *vd, int query, va_list ap) /* Hide the mouse. It will be send when * vout_display_t::info.b_hide_mouse is false */ case VOUT_DISPLAY_HIDE_MOUSE: - xcb_change_window_attributes (XGetXCBConnection (sys->display), - sys->embed->xid, + xcb_change_window_attributes (sys->conn, sys->embed->handle.xid, XCB_CW_CURSOR, &(uint32_t){ sys->cursor }); + xcb_flush (sys->conn); return VLC_SUCCESS; case VOUT_DISPLAY_GET_OPENGL: { - vout_opengl_t **gl = va_arg (ap, vout_opengl_t **); + vlc_gl_t **gl = va_arg (ap, vlc_gl_t **); *gl = &sys->gl; return VLC_SUCCESS; } @@ -552,7 +611,6 @@ static int Control (vout_display_t *vd, int query, va_list ap) static void Manage (vout_display_t *vd) { vout_display_sys_t *sys = vd->sys; - xcb_connection_t *conn = XGetXCBConnection (sys->display); - ManageEvent (vd, conn, &sys->visible); + ManageEvent (vd, sys->conn, &sys->visible); }