/* GStreamer Intel MSDK plugin * Copyright (c) 2016, Oblong Industries, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* TODO: * - discover dri_path instead of having it hardcoded */ #include #include #include #include "msdk.h" #include "msdk_libva.h" GST_DEBUG_CATEGORY_EXTERN (gst_msdkenc_debug); #define GST_CAT_DEFAULT gst_msdkenc_debug struct fourcc_map { mfxU32 mfx_fourcc; guint32 va_fourcc; }; struct rt_map { mfxU32 mfx_rt_format; guint32 va_rt_format; }; #define FOURCC_MFX_TO_VA(MFX, VA) \ { MFX_FOURCC_##MFX, VA_FOURCC_##VA } #define RT_MFX_TO_VA(MFX, VA) \ { MFX_CHROMAFORMAT_##MFX, VA_RT_FORMAT_##VA } static const struct fourcc_map gst_msdk_fourcc_mfx_to_va[] = { FOURCC_MFX_TO_VA (NV12, NV12), FOURCC_MFX_TO_VA (YUY2, YUY2), FOURCC_MFX_TO_VA (UYVY, UYVY), FOURCC_MFX_TO_VA (YV12, YV12), FOURCC_MFX_TO_VA (RGB4, ARGB), FOURCC_MFX_TO_VA (P8, P208), {0, 0} }; static const struct rt_map gst_msdk_rt_mfx_to_va[] = { RT_MFX_TO_VA (YUV420, YUV420), RT_MFX_TO_VA (YUV422, YUV422), RT_MFX_TO_VA (YUV444, YUV444), {0, 0} }; struct _MsdkContext { mfxSession session; gint fd; VADisplay dpy; }; static gboolean msdk_use_vaapi_on_context (MsdkContext * context) { gint fd; gint maj_ver, min_ver; VADisplay va_dpy = NULL; VAStatus va_status; mfxStatus status; /* maybe /dev/dri/renderD128 */ static const gchar *dri_path = "/dev/dri/card0"; fd = open (dri_path, O_RDWR); if (fd < 0) { GST_ERROR ("Couldn't open %s", dri_path); return FALSE; } va_dpy = vaGetDisplayDRM (fd); if (!va_dpy) { GST_ERROR ("Couldn't get a VA DRM display"); goto failed; } va_status = vaInitialize (va_dpy, &maj_ver, &min_ver); if (va_status != VA_STATUS_SUCCESS) { GST_ERROR ("Couldn't initialize VA DRM display"); goto failed; } status = MFXVideoCORE_SetHandle (context->session, MFX_HANDLE_VA_DISPLAY, (mfxHDL) va_dpy); if (status != MFX_ERR_NONE) { GST_ERROR ("Setting VAAPI handle failed (%s)", msdk_status_to_string (status)); goto failed; } context->fd = fd; context->dpy = va_dpy; return TRUE; failed: if (va_dpy) vaTerminate (va_dpy); close (fd); return FALSE; } MsdkContext * msdk_open_context (gboolean hardware) { MsdkContext *context = g_slice_new0 (MsdkContext); context->fd = -1; context->session = msdk_open_session (hardware); if (!context->session) goto failed; if (hardware) { if (!msdk_use_vaapi_on_context (context)) goto failed; } return context; failed: msdk_close_session (context->session); g_slice_free (MsdkContext, context); return NULL; } void msdk_close_context (MsdkContext * context) { if (!context) return; msdk_close_session (context->session); if (context->dpy) vaTerminate (context->dpy); if (context->fd >= 0) close (context->fd); g_slice_free (MsdkContext, context); } mfxSession msdk_context_get_session (MsdkContext * context) { return context->session; } mfxStatus gst_msdk_get_mfx_status_from_va_status (VAStatus va_res) { mfxStatus mfxRes = MFX_ERR_NONE; switch (va_res) { case VA_STATUS_SUCCESS: mfxRes = MFX_ERR_NONE; break; case VA_STATUS_ERROR_ALLOCATION_FAILED: mfxRes = MFX_ERR_MEMORY_ALLOC; break; case VA_STATUS_ERROR_ATTR_NOT_SUPPORTED: case VA_STATUS_ERROR_UNSUPPORTED_PROFILE: case VA_STATUS_ERROR_UNSUPPORTED_ENTRYPOINT: case VA_STATUS_ERROR_UNSUPPORTED_RT_FORMAT: case VA_STATUS_ERROR_UNSUPPORTED_BUFFERTYPE: case VA_STATUS_ERROR_FLAG_NOT_SUPPORTED: case VA_STATUS_ERROR_RESOLUTION_NOT_SUPPORTED: mfxRes = MFX_ERR_UNSUPPORTED; break; case VA_STATUS_ERROR_INVALID_DISPLAY: case VA_STATUS_ERROR_INVALID_CONFIG: case VA_STATUS_ERROR_INVALID_CONTEXT: case VA_STATUS_ERROR_INVALID_SURFACE: case VA_STATUS_ERROR_INVALID_BUFFER: case VA_STATUS_ERROR_INVALID_IMAGE: case VA_STATUS_ERROR_INVALID_SUBPICTURE: mfxRes = MFX_ERR_NOT_INITIALIZED; break; case VA_STATUS_ERROR_INVALID_PARAMETER: mfxRes = MFX_ERR_INVALID_VIDEO_PARAM; default: mfxRes = MFX_ERR_UNKNOWN; break; } return mfxRes; } guint gst_msdk_get_va_fourcc_from_mfx_fourcc (mfxU32 fourcc) { const struct fourcc_map *m = gst_msdk_fourcc_mfx_to_va; for (; m->mfx_fourcc != 0; m++) { if (m->mfx_fourcc == fourcc) return m->va_fourcc; } return 0; } guint gst_msdk_get_mfx_fourcc_from_va_fourcc (guint32 fourcc) { const struct fourcc_map *m = gst_msdk_fourcc_mfx_to_va; for (; m->va_fourcc != 0; m++) { if (m->va_fourcc == fourcc) return m->mfx_fourcc; } return 0; } guint gst_msdk_get_va_rt_format_from_mfx_rt_format (mfxU32 format) { const struct rt_map *m = gst_msdk_rt_mfx_to_va; for (; m->mfx_rt_format != 0; m++) { if (m->mfx_rt_format == format) return m->va_rt_format; } return 0; } guint gst_msdk_get_mfx_rt_format_from_va_rt_format (guint32 format) { const struct rt_map *m = gst_msdk_rt_mfx_to_va; for (; m->va_rt_format != 0; m++) { if (m->va_rt_format == format) return m->mfx_rt_format; } return 0; }