271 lines
6.5 KiB
C
271 lines
6.5 KiB
C
/* 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 <fcntl.h>
|
|
#include <unistd.h>
|
|
|
|
#include <va/va_drm.h>
|
|
#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;
|
|
}
|