diff --git a/modules/MODULES_LIST b/modules/MODULES_LIST index 2f1656d930745cc9622757c055b1a928dd3860da..698118fe5eb14d72abb24d8428265114a7743940 100644 --- a/modules/MODULES_LIST +++ b/modules/MODULES_LIST @@ -167,6 +167,7 @@ $Id$ * glconv_vaapi_drm: VA-API OpenGL hardware converter for DRM * glconv_vaapi_wl: VA-API OpenGL hardware converter for Wayland * glconv_vaapi_x11: VA-API OpenGL hardware converter for X11 + * glconv_vdpau: VDPAU OpenGL surface converter * gles2: OpenGL ES2 video output using the generic OpenGL provider * glspectrum: 3D OpenGL spectrum visualization * glwin32: a opengl provider using DirectX OpenGL diff --git a/modules/hw/vdpau/vlc_vdpau.h b/modules/hw/vdpau/vlc_vdpau.h index af62bddcc1b87d630a1b68cb1622ce216db670d0..01289ffbae49e406817e246a8de775b6dd866434 100644 --- a/modules/hw/vdpau/vlc_vdpau.h +++ b/modules/hw/vdpau/vlc_vdpau.h @@ -261,6 +261,7 @@ struct picture_sys_t VdpOutputSurface surface; VdpDevice device; vdp_t *vdp; + void *gl_nv_surface; }; typedef struct vlc_vdp_video_frame diff --git a/modules/video_output/Makefile.am b/modules/video_output/Makefile.am index 004bc51952502db212b026b70615ce3fc2f04bbb..50fa5344996c669c587b22f482776955bd38d039 100644 --- a/modules/video_output/Makefile.am +++ b/modules/video_output/Makefile.am @@ -76,6 +76,11 @@ libglconv_vaapi_drm_plugin_la_CFLAGS = $(AM_CFLAGS) -DHAVE_VA_DRM libglconv_vaapi_drm_plugin_la_LIBADD = $(LIBVA_LIBS) $(LIBVA_EGL_LIBS) \ $(LIBVA_DRM_LIBS) +libglconv_vdpau_plugin_la_SOURCES = video_output/opengl/converter_vdpau.c \ + video_output/opengl/converter.h hw/vdpau/vlc_vdpau.h +libglconv_vdpau_plugin_la_CFLAGS = $(AM_CFLAGS) $(VDPAU_CFLAGS) +libglconv_vdpau_plugin_la_LIBADD = $(LIBDL) libvlc_vdpau.la $(X_LIBS) $(X_PRE_LIBS) -lX11 + if HAVE_GL vout_LTLIBRARIES += libgl_plugin.la if HAVE_EGL @@ -95,6 +100,10 @@ vout_LTLIBRARIES += libglconv_vaapi_drm_plugin.la endif endif endif # HAVE_EGL + +if HAVE_VDPAU +vout_LTLIBRARIES += libglconv_vdpau_plugin.la +endif endif # HAVE_GL ### XCB ### diff --git a/modules/video_output/opengl/converter_vdpau.c b/modules/video_output/opengl/converter_vdpau.c new file mode 100644 index 0000000000000000000000000000000000000000..6b3467c37ef9efcdc09c766971108490d3ad1abb --- /dev/null +++ b/modules/video_output/opengl/converter_vdpau.c @@ -0,0 +1,249 @@ +/***************************************************************************** + * converter_vdpau.c: OpenGL VDPAU opaque converter + ***************************************************************************** + * Copyright (C) 2017 VLC authors, VideoLAN and VideoLabs + * + * Author: Victorien Le Couviour--Tuffet <victorien.lecouviour.tuffet@gmail.com> + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. + *****************************************************************************/ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include <assert.h> + +#include <GL/gl.h> +#include <GL/glext.h> + +#include <vlc_common.h> +#include <vlc_vout_window.h> +#include <vlc_xlib.h> + +#include "../../hw/vdpau/vlc_vdpau.h" +#include "internal.h" + +#define INTEROP_CALL(fct, ...) \ + _##fct(__VA_ARGS__); \ + { \ + GLenum ret = tc->vt->GetError(); \ + if (ret != GL_NO_ERROR) \ + { \ + msg_Err(tc->gl, #fct " failed: 0x%x\n", ret); \ + return VLC_EGENERIC; \ + } \ + } + +struct priv +{ + vdp_t *vdp; + VdpDevice vdp_device; +}; + +static PFNGLVDPAUINITNVPROC _glVDPAUInitNV; +static PFNGLVDPAUFININVPROC _glVDPAUFiniNV; +static PFNGLVDPAUREGISTEROUTPUTSURFACENVPROC _glVDPAURegisterOutputSurfaceNV; +static PFNGLVDPAUISSURFACENVPROC _glVDPAUIsSurfaceNV; +static PFNGLVDPAUUNREGISTERSURFACENVPROC _glVDPAUUnregisterSurfaceNV; +static PFNGLVDPAUGETSURFACEIVNVPROC _glVDPAUGetSurfaceivNV; +static PFNGLVDPAUSURFACEACCESSNVPROC _glVDPAUSurfaceAccessNV; +static PFNGLVDPAUMAPSURFACESNVPROC _glVDPAUMapSurfacesNV; +static PFNGLVDPAUUNMAPSURFACESNVPROC _glVDPAUUnmapSurfacesNV; + +static void +pool_pic_destroy_cb(picture_t *pic) +{ + vdp_output_surface_destroy(pic->p_sys->vdp, pic->p_sys->surface); + vdp_release_x11(pic->p_sys->vdp); + free(pic->p_sys); + free(pic); +} + +static picture_pool_t * +tc_vdpau_gl_get_pool(opengl_tex_converter_t const *tc, + unsigned int requested_count) +{ + struct priv *priv = tc->priv; + picture_t *pics[requested_count]; + + unsigned int i; + for (i = 0; i < requested_count; ++i) + { + VdpOutputSurface surface; + + VdpStatus st; + if ((st = vdp_output_surface_create(priv->vdp, priv->vdp_device, + VDP_RGBA_FORMAT_B8G8R8A8, + tc->fmt.i_visible_width, + tc->fmt.i_visible_height, + &surface)) != VDP_STATUS_OK) + goto error; + + picture_sys_t *picsys = calloc(1, sizeof(*picsys)); + if (!picsys) + goto error; + + picsys->vdp = vdp_hold_x11(priv->vdp, NULL); + picsys->device = priv->vdp_device; + picsys->surface = surface; + + picture_resource_t rsc = { .p_sys = picsys, + .pf_destroy = pool_pic_destroy_cb }; + + pics[i] = picture_NewFromResource(&tc->fmt, &rsc); + if (!pics[i]) + goto error; + } + + picture_pool_t *pool = picture_pool_New(requested_count, pics); + if (!pool) + goto error; + + return pool; + +error: + while (i--) + picture_Release(pics[i]); + return NULL; +} + +static int +tc_vdpau_gl_update(opengl_tex_converter_t const *tc, GLuint textures[], + GLsizei const tex_widths[], GLsizei const tex_heights[], + picture_t *pic, size_t const plane_offsets[]) +{ + VLC_UNUSED(tex_widths); + VLC_UNUSED(tex_heights); + VLC_UNUSED(plane_offsets); + + GLvdpauSurfaceNV *p_gl_nv_surface = + (GLvdpauSurfaceNV *)&pic->p_sys->gl_nv_surface; + + if (*p_gl_nv_surface) + { + assert(_glVDPAUIsSurfaceNV(*p_gl_nv_surface) == GL_TRUE); + + GLint state; + GLsizei num_val; + INTEROP_CALL(glVDPAUGetSurfaceivNV, *p_gl_nv_surface, + GL_SURFACE_STATE_NV, 1, &num_val, &state); + assert(num_val == 1); assert(state == GL_SURFACE_MAPPED_NV); + + INTEROP_CALL(glVDPAUUnmapSurfacesNV, 1, p_gl_nv_surface); + INTEROP_CALL(glVDPAUUnregisterSurfaceNV, *p_gl_nv_surface); + } + + *p_gl_nv_surface = + INTEROP_CALL(glVDPAURegisterOutputSurfaceNV, + (void *)(size_t)pic->p_sys->surface, + GL_TEXTURE_2D, tc->tex_count, textures); + INTEROP_CALL(glVDPAUSurfaceAccessNV, *p_gl_nv_surface, GL_READ_ONLY); + INTEROP_CALL(glVDPAUMapSurfacesNV, 1, p_gl_nv_surface); + + return VLC_SUCCESS; +} + +static void +Close(vlc_object_t *obj) +{ + opengl_tex_converter_t *tc = (void *)obj; + _glVDPAUFiniNV(); assert(tc->vt->GetError() == GL_NO_ERROR); + vdp_release_x11(((struct priv *)tc->priv)->vdp); + free(tc->priv); +} + +static int +Open(vlc_object_t *obj) +{ + opengl_tex_converter_t *tc = (void *) obj; + if ((tc->fmt.i_chroma != VLC_CODEC_VDPAU_VIDEO_420 && + tc->fmt.i_chroma != VLC_CODEC_VDPAU_VIDEO_422 && + tc->fmt.i_chroma != VLC_CODEC_VDPAU_VIDEO_444) || + !HasExtension(tc->glexts, "GL_NV_vdpau_interop") || + tc->gl->surface->type != VOUT_WINDOW_TYPE_XID) + return VLC_EGENERIC; + + tc->fmt.i_chroma = VLC_CODEC_VDPAU_OUTPUT; + + if (!vlc_xlib_init(VLC_OBJECT(tc->gl))) + return VLC_EGENERIC; + + struct priv *priv = calloc(1, sizeof(*priv)); + if (!priv) + return VLC_EGENERIC; + tc->priv = priv; + + if (vdp_get_x11(tc->gl->surface->display.x11, -1, + &priv->vdp, &priv->vdp_device) != VDP_STATUS_OK) + { + free(priv); + return VLC_EGENERIC; + } + + void *vdp_gpa; + if (vdp_get_proc_address(priv->vdp, priv->vdp_device, + VDP_FUNC_ID_GET_PROC_ADDRESS, &vdp_gpa) + != VDP_STATUS_OK) + { + vdp_release_x11(priv->vdp); + free(priv); + return VLC_EGENERIC; + } + +#define SAFE_GPA(fct) \ + _##fct = vlc_gl_GetProcAddress(tc->gl, #fct); \ + if (!_##fct) \ + { \ + vdp_release_x11(priv->vdp); \ + free(priv); \ + return VLC_EGENERIC; \ + } + SAFE_GPA(glVDPAUInitNV); + SAFE_GPA(glVDPAUFiniNV); + SAFE_GPA(glVDPAURegisterOutputSurfaceNV); + SAFE_GPA(glVDPAUIsSurfaceNV); + SAFE_GPA(glVDPAUUnregisterSurfaceNV); + SAFE_GPA(glVDPAUGetSurfaceivNV); + SAFE_GPA(glVDPAUSurfaceAccessNV); + SAFE_GPA(glVDPAUMapSurfacesNV); + SAFE_GPA(glVDPAUUnmapSurfacesNV); +#undef SAFE_GPA + + INTEROP_CALL(glVDPAUInitNV, (void *)(size_t)priv->vdp_device, vdp_gpa); + + tc->fshader = opengl_fragment_shader_init(tc, GL_TEXTURE_2D, + VLC_CODEC_RGB32, + COLOR_SPACE_UNDEF); + if (!tc->fshader) + { + Close(obj); + return VLC_EGENERIC; + } + + tc->pf_get_pool = tc_vdpau_gl_get_pool; + tc->pf_update = tc_vdpau_gl_update; + + return VLC_SUCCESS; +} + +vlc_module_begin () + set_description("VDPAU OpenGL surface converter") + set_capability("glconv", 2) + set_callbacks(Open, Close) + set_category(CAT_VIDEO) + set_subcategory(SUBCAT_VIDEO_VOUT) + add_shortcut("vdpau") +vlc_module_end () diff --git a/po/POTFILES.in b/po/POTFILES.in index 57ef6ba7b56a75d2669854c53d989122c044bce4..767a9d25fc3e13d78c0a1a7836e3113de8fdc21b 100644 --- a/po/POTFILES.in +++ b/po/POTFILES.in @@ -1135,6 +1135,7 @@ modules/video_output/macosx.m modules/video_output/opengl/converter_android.c modules/video_output/opengl/converter_cvpx.c modules/video_output/opengl/converter_vaapi.c +modules/video_output/opengl/converter_vdpau.c modules/video_output/opengl/display.c modules/video_output/opengl/egl.c modules/video_output/win32/direct3d9.c