mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-09 00:45:56 +00:00
lcevcdecoder: Add new LCEVC Decoder plugin
This new LCEVC decoder plugin is meant to implement all LCEVC decoder elements. For now, it only implements the LCEVC enhancement decoder (lcevcdec) element. This element essentially enhances raw video frames using the LCEVC metadata attached to input buffers into a higher resolution frame. The element is only meant to be used after any base decoder (eg avdec_h264). Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/7330>
This commit is contained in:
parent
9accf21e53
commit
636690f2aa
8 changed files with 999 additions and 0 deletions
720
subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c
Normal file
720
subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c
Normal file
|
@ -0,0 +1,720 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2024> V-Nova International Limited
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Library General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Library General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Library General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
#include "config.h"
|
||||
#endif
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <gst/codecparsers/gstlcevcmeta.h>
|
||||
|
||||
#include "gstlcevcdecutils.h"
|
||||
#include "gstlcevcdec.h"
|
||||
|
||||
enum
|
||||
{
|
||||
PROP_0,
|
||||
PROP_VERBOSE,
|
||||
PROP_MAX_WIDTH,
|
||||
PROP_MAX_HEIGHT,
|
||||
PROP_MAX_LATENCY
|
||||
};
|
||||
|
||||
#define DEFAULT_MAX_WIDTH 3840
|
||||
#define DEFAULT_MAX_HEIGHT 2160
|
||||
#define DEFAULT_MAX_LATENCY 0
|
||||
|
||||
#define GST_CAT_DEFAULT gst_lcevc_dec_debug
|
||||
GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT);
|
||||
|
||||
static GstStaticPadTemplate sink_template_factory =
|
||||
GST_STATIC_PAD_TEMPLATE ("sink",
|
||||
GST_PAD_SINK,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE
|
||||
(GST_LCEVC_DEC_UTILS_SUPPORTED_FORMATS))
|
||||
);
|
||||
|
||||
static GstStaticPadTemplate src_template_factory =
|
||||
GST_STATIC_PAD_TEMPLATE ("src",
|
||||
GST_PAD_SRC,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE
|
||||
(GST_LCEVC_DEC_UTILS_SUPPORTED_FORMATS))
|
||||
);
|
||||
|
||||
#define DEBUG_INIT \
|
||||
GST_DEBUG_CATEGORY_INIT (gst_lcevc_dec_debug, \
|
||||
"lcevcdec", 0, "LCEVC Decoder element");
|
||||
|
||||
G_DEFINE_TYPE_WITH_CODE (GstLcevcDec, gst_lcevc_dec, GST_TYPE_VIDEO_DECODER,
|
||||
DEBUG_INIT);
|
||||
GST_ELEMENT_REGISTER_DEFINE (lcevcdec, "lcevcdec",
|
||||
GST_RANK_MARGINAL, GST_TYPE_LCEVC_DEC);
|
||||
|
||||
#define GST_LCEVC_DEC_PICTURE_DATA gst_lcevc_dec_picture_data ()
|
||||
|
||||
static GQuark
|
||||
gst_lcevc_dec_picture_data (void)
|
||||
{
|
||||
static GQuark quark = 0;
|
||||
|
||||
if (quark == 0)
|
||||
quark = g_quark_from_string ("GstLcevcDecPictureData");
|
||||
|
||||
return quark;
|
||||
}
|
||||
|
||||
typedef struct
|
||||
{
|
||||
LCEVC_DecoderHandle decoder_handle;
|
||||
LCEVC_PictureHandle picture_handle;
|
||||
guint32 width;
|
||||
guint height;
|
||||
} PictureData;
|
||||
|
||||
static PictureData *
|
||||
picture_data_new (LCEVC_DecoderHandle decoder_handle, GstVideoFrame * frame)
|
||||
{
|
||||
PictureData *ret = g_new0 (PictureData, 1);
|
||||
|
||||
ret->decoder_handle = decoder_handle;
|
||||
ret->width = GST_VIDEO_FRAME_WIDTH (frame);
|
||||
ret->height = GST_VIDEO_FRAME_HEIGHT (frame);
|
||||
|
||||
/* Alloc LCEVC picture handle */
|
||||
if (!gst_lcevc_dec_utils_alloc_picture_handle (decoder_handle, frame,
|
||||
&ret->picture_handle)) {
|
||||
g_free (ret);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
picture_data_free (gpointer p)
|
||||
{
|
||||
PictureData *data = p;
|
||||
LCEVC_FreePicture (data->decoder_handle, data->picture_handle);
|
||||
g_free (data);
|
||||
}
|
||||
|
||||
static void
|
||||
gst_lcevc_dec_init (GstLcevcDec * lcevc)
|
||||
{
|
||||
lcevc->max_width = DEFAULT_MAX_WIDTH;
|
||||
lcevc->max_height = DEFAULT_MAX_HEIGHT;
|
||||
lcevc->max_latency = DEFAULT_MAX_LATENCY;
|
||||
}
|
||||
|
||||
static void
|
||||
gst_lcevc_dec_set_property (GObject * object, guint prop_id,
|
||||
const GValue * value, GParamSpec * pspec)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (object);
|
||||
|
||||
switch (prop_id) {
|
||||
case PROP_VERBOSE:
|
||||
lcevc->verbose = g_value_get_boolean (value);
|
||||
break;
|
||||
case PROP_MAX_WIDTH:
|
||||
lcevc->max_width = g_value_get_int (value);
|
||||
break;
|
||||
case PROP_MAX_HEIGHT:
|
||||
lcevc->max_height = g_value_get_int (value);
|
||||
break;
|
||||
case PROP_MAX_LATENCY:
|
||||
lcevc->max_latency = g_value_get_int (value);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
gst_lcevc_dec_get_property (GObject * object, guint prop_id,
|
||||
GValue * value, GParamSpec * pspec)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (object);
|
||||
|
||||
switch (prop_id) {
|
||||
case PROP_VERBOSE:
|
||||
g_value_set_boolean (value, lcevc->verbose);
|
||||
break;
|
||||
case PROP_MAX_WIDTH:
|
||||
g_value_set_int (value, lcevc->max_width);
|
||||
break;
|
||||
case PROP_MAX_HEIGHT:
|
||||
g_value_set_int (value, lcevc->max_height);
|
||||
break;
|
||||
case PROP_MAX_LATENCY:
|
||||
g_value_set_int (value, lcevc->max_latency);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
event_callback (LCEVC_DecoderHandle dec, LCEVC_Event event,
|
||||
LCEVC_PictureHandle pic, const LCEVC_DecodeInformation * info,
|
||||
const uint8_t * data, uint32_t size, void *user_data)
|
||||
{
|
||||
GstLcevcDec *lcevc = user_data;
|
||||
|
||||
switch (event) {
|
||||
case LCEVC_Log:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC Log");
|
||||
break;
|
||||
case LCEVC_Exit:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC Exit");
|
||||
break;
|
||||
case LCEVC_CanSendBase:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC CanSendBase");
|
||||
break;
|
||||
case LCEVC_CanSendEnhancement:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC CanSendEnhancement");
|
||||
break;
|
||||
case LCEVC_CanSendPicture:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC CanSendPicure");
|
||||
break;
|
||||
case LCEVC_CanReceive:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC CanReceive");
|
||||
break;
|
||||
case LCEVC_BasePictureDone:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC Base Picure Done");
|
||||
break;
|
||||
case LCEVC_OutputPictureDone:
|
||||
GST_DEBUG_OBJECT (lcevc, "LCEVC Output Picure Done");
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
initialize_lcevc_decoder (GstLcevcDec * lcevc)
|
||||
{
|
||||
LCEVC_AccelContextHandle accel_context = { 0, };
|
||||
int32_t events[] = { LCEVC_Log, LCEVC_Exit, LCEVC_CanSendBase,
|
||||
LCEVC_CanSendEnhancement, LCEVC_CanSendPicture, LCEVC_CanReceive,
|
||||
LCEVC_BasePictureDone, LCEVC_OutputPictureDone
|
||||
};
|
||||
|
||||
if (LCEVC_CreateDecoder (&lcevc->decoder_handle, accel_context) !=
|
||||
LCEVC_Success)
|
||||
return FALSE;
|
||||
|
||||
if (lcevc->max_width > 0)
|
||||
LCEVC_ConfigureDecoderInt (lcevc->decoder_handle, "max_width",
|
||||
lcevc->max_width);
|
||||
if (lcevc->max_height > 0)
|
||||
LCEVC_ConfigureDecoderInt (lcevc->decoder_handle, "max_height",
|
||||
lcevc->max_height);
|
||||
if (lcevc->max_latency > 0)
|
||||
LCEVC_ConfigureDecoderInt (lcevc->decoder_handle, "max_latency",
|
||||
lcevc->max_latency);
|
||||
|
||||
if (lcevc->verbose) {
|
||||
LCEVC_ConfigureDecoderBool (lcevc->decoder_handle, "log_stdout", TRUE);
|
||||
LCEVC_ConfigureDecoderInt (lcevc->decoder_handle, "log_level", 2);
|
||||
}
|
||||
|
||||
LCEVC_ConfigureDecoderIntArray (lcevc->decoder_handle, "events", 8, events);
|
||||
|
||||
LCEVC_SetDecoderEventCallback (lcevc->decoder_handle, event_callback, lcevc);
|
||||
|
||||
if (LCEVC_InitializeDecoder (lcevc->decoder_handle) != LCEVC_Success)
|
||||
return FALSE;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_lcevc_dec_start (GstVideoDecoder * decoder)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (decoder);
|
||||
|
||||
/* Initialize LCEVC decoder */
|
||||
if (!initialize_lcevc_decoder (lcevc)) {
|
||||
GST_ELEMENT_ERROR (decoder, LIBRARY, INIT, (NULL),
|
||||
("Couldn't initialize LCEVC decoder"));
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_lcevc_dec_stop (GstVideoDecoder * decoder)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (decoder);
|
||||
|
||||
/* Destry LCEVC decoder */
|
||||
LCEVC_DestroyDecoder (lcevc->decoder_handle);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_lcevc_dec_decide_allocation (GstVideoDecoder * decoder, GstQuery * query)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (decoder);
|
||||
|
||||
if (!GST_VIDEO_DECODER_CLASS (gst_lcevc_dec_parent_class)->decide_allocation
|
||||
(decoder, query))
|
||||
return FALSE;
|
||||
|
||||
lcevc->can_crop = gst_query_find_allocation_meta (query,
|
||||
GST_VIDEO_CROP_META_API_TYPE, NULL);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
ensure_output_resolution (GstLcevcDec * lcevc, guint32 width, guint32 height,
|
||||
GstVideoCodecState * state)
|
||||
{
|
||||
/* Set output state with input resolution to do passthrough */
|
||||
if (width != lcevc->out_width || height != lcevc->out_height) {
|
||||
GstVideoCodecState *s;
|
||||
|
||||
s = gst_video_decoder_set_output_state (GST_VIDEO_DECODER (lcevc),
|
||||
GST_VIDEO_INFO_FORMAT (&lcevc->in_info), width, height, state);
|
||||
if (!s)
|
||||
return FALSE;
|
||||
|
||||
lcevc->out_width = width;
|
||||
lcevc->out_height = height;
|
||||
|
||||
gst_video_codec_state_unref (s);
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_lcevc_dec_set_format (GstVideoDecoder * decoder, GstVideoCodecState * state)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (decoder);
|
||||
LCEVC_ColorFormat format;
|
||||
gint par_n, par_d;
|
||||
guint32 w, h;
|
||||
|
||||
/* Make sure format is supported */
|
||||
format =
|
||||
gst_lcevc_dec_utils_get_color_format (GST_VIDEO_INFO_FORMAT
|
||||
(&state->info));
|
||||
if (format == LCEVC_ColorFormat_Unknown)
|
||||
return FALSE;
|
||||
|
||||
/* Keep input info */
|
||||
lcevc->in_info = state->info;
|
||||
|
||||
/* Output resultion is always twice as big as input resultion divided by
|
||||
* pixel aspect ratio */
|
||||
par_n = GST_VIDEO_INFO_PAR_N (&state->info);
|
||||
if (par_n == 0)
|
||||
par_n = 1;
|
||||
par_d = GST_VIDEO_INFO_PAR_D (&state->info);
|
||||
if (par_d == 0)
|
||||
par_d = 1;
|
||||
w = (GST_VIDEO_INFO_WIDTH (&state->info) * 2) / par_d;
|
||||
h = (GST_VIDEO_INFO_HEIGHT (&state->info) * 2) / par_n;
|
||||
|
||||
/* Set pixel aspect ratio back to 1/1 */
|
||||
GST_VIDEO_INFO_PAR_N (&state->info) = 1;
|
||||
GST_VIDEO_INFO_PAR_D (&state->info) = 1;
|
||||
|
||||
/* Set output resolution */
|
||||
if (!ensure_output_resolution (lcevc, w, h, state))
|
||||
return FALSE;
|
||||
|
||||
/* We always work with full RAW video frames */
|
||||
gst_video_decoder_set_subframe_mode (decoder, FALSE);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static GstVideoCodecFrame *
|
||||
find_pending_frame_from_picture_handle (GstLcevcDec * lcevc,
|
||||
LCEVC_PictureHandle picture_handle)
|
||||
{
|
||||
GList *iter;
|
||||
GList *pending_frames;
|
||||
GstVideoCodecFrame *found = NULL;
|
||||
|
||||
pending_frames = gst_video_decoder_get_frames (GST_VIDEO_DECODER (lcevc));
|
||||
|
||||
for (iter = pending_frames; iter; iter = g_list_next (iter)) {
|
||||
GstVideoCodecFrame *frame = (GstVideoCodecFrame *) iter->data;
|
||||
PictureData *pd;
|
||||
|
||||
pd = gst_mini_object_get_qdata (GST_MINI_OBJECT (frame->output_buffer),
|
||||
GST_LCEVC_DEC_PICTURE_DATA);
|
||||
if (pd->picture_handle.hdl == picture_handle.hdl) {
|
||||
found = gst_video_codec_frame_ref (frame);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
g_list_free_full (pending_frames,
|
||||
(GDestroyNotify) gst_video_codec_frame_unref);
|
||||
|
||||
return found;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
receive_enhanced_picture (GstLcevcDec * lcevc)
|
||||
{
|
||||
LCEVC_PictureHandle picture_handle = { 0, };
|
||||
LCEVC_DecodeInformation decode_info = { 0, };
|
||||
|
||||
while (LCEVC_ReceiveDecoderPicture (lcevc->decoder_handle, &picture_handle,
|
||||
&decode_info) == LCEVC_Success) {
|
||||
LCEVC_PictureDesc pic_desc = { 0, };
|
||||
GstVideoCodecFrame *received_frame;
|
||||
GstVideoCropMeta *cmeta;
|
||||
|
||||
if (LCEVC_GetPictureDesc (lcevc->decoder_handle, picture_handle,
|
||||
&pic_desc) != LCEVC_Success) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not get desciption of received enhanced picutre"));
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
GST_INFO_OBJECT (lcevc,
|
||||
"Received enhanced picture: ts=%ld e=%d w=%d h=%d t=%d b=%d l=%d r=%d",
|
||||
decode_info.timestamp, decode_info.enhanced, pic_desc.width,
|
||||
pic_desc.height, pic_desc.cropTop, pic_desc.cropBottom,
|
||||
pic_desc.cropLeft, pic_desc.cropRight);
|
||||
|
||||
received_frame = find_pending_frame_from_picture_handle (lcevc,
|
||||
picture_handle);
|
||||
if (received_frame) {
|
||||
/* Change output allocation if enhanced picutre resolution changed */
|
||||
if (!ensure_output_resolution (lcevc, pic_desc.width, pic_desc.height,
|
||||
NULL)) {
|
||||
gst_video_codec_frame_unref (received_frame);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/* Add crop meta if downstream can crop */
|
||||
if (lcevc->can_crop) {
|
||||
cmeta = gst_buffer_add_video_crop_meta (received_frame->output_buffer);
|
||||
cmeta->x = pic_desc.cropLeft;
|
||||
cmeta->y = pic_desc.cropTop;
|
||||
cmeta->width =
|
||||
pic_desc.width - (pic_desc.cropLeft + pic_desc.cropRight);
|
||||
cmeta->height =
|
||||
pic_desc.height - (pic_desc.cropTop + pic_desc.cropBottom);
|
||||
|
||||
/* Change output caps if crop values changed */
|
||||
if (lcevc->out_crop_top != pic_desc.cropTop ||
|
||||
lcevc->out_crop_bottom != pic_desc.cropBottom ||
|
||||
lcevc->out_crop_left != pic_desc.cropLeft ||
|
||||
lcevc->out_crop_right != pic_desc.cropRight) {
|
||||
GstVideoCodecState *s;
|
||||
|
||||
lcevc->out_crop_top = pic_desc.cropTop;
|
||||
lcevc->out_crop_bottom = pic_desc.cropBottom;
|
||||
lcevc->out_crop_left = pic_desc.cropLeft;
|
||||
lcevc->out_crop_right = pic_desc.cropRight;
|
||||
|
||||
s = gst_video_decoder_get_output_state (GST_VIDEO_DECODER (lcevc));
|
||||
if (!s) {
|
||||
gst_video_codec_frame_unref (received_frame);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
s->caps = gst_video_info_to_caps (&s->info);
|
||||
gst_caps_set_simple (s->caps,
|
||||
"width", G_TYPE_INT,
|
||||
pic_desc.width - (pic_desc.cropLeft + pic_desc.cropRight),
|
||||
"height", G_TYPE_INT,
|
||||
pic_desc.height - (pic_desc.cropTop + pic_desc.cropBottom), NULL);
|
||||
gst_video_decoder_negotiate (GST_VIDEO_DECODER (lcevc));
|
||||
|
||||
gst_video_codec_state_unref (s);
|
||||
}
|
||||
}
|
||||
|
||||
/* Finish frame */
|
||||
received_frame->output_buffer->pts = decode_info.timestamp;
|
||||
gst_video_decoder_finish_frame (GST_VIDEO_DECODER (lcevc),
|
||||
received_frame);
|
||||
gst_video_codec_frame_unref (received_frame);
|
||||
}
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
receive_base_picture (GstLcevcDec * lcevc)
|
||||
{
|
||||
LCEVC_PictureHandle picture_handle = { 0, };
|
||||
|
||||
while (LCEVC_ReceiveDecoderBase (lcevc->decoder_handle, &picture_handle)
|
||||
== LCEVC_Success) {
|
||||
GST_DEBUG_OBJECT (lcevc, "Received base picture %ld", picture_handle.hdl);
|
||||
|
||||
if (LCEVC_FreePicture (lcevc->decoder_handle, picture_handle)
|
||||
!= LCEVC_Success) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not free base picture %ld", picture_handle.hdl));
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
send_enhancement_data (GstLcevcDec * lcevc, GstBuffer * input_buffer)
|
||||
{
|
||||
gboolean ret = FALSE;
|
||||
GstLcevcMeta *lcevc_meta;
|
||||
GstMapInfo enhancement_info;
|
||||
|
||||
lcevc_meta = gst_buffer_get_lcevc_meta (input_buffer);
|
||||
if (!lcevc_meta) {
|
||||
GST_INFO_OBJECT (lcevc,
|
||||
"Input buffer %ld enhancement data not found, doing passthrough",
|
||||
input_buffer->pts);
|
||||
|
||||
/* Set output state with input resolution to do passthrough */
|
||||
return ensure_output_resolution (lcevc,
|
||||
GST_VIDEO_INFO_WIDTH (&lcevc->in_info),
|
||||
GST_VIDEO_INFO_HEIGHT (&lcevc->in_info), NULL);
|
||||
}
|
||||
|
||||
if (!gst_buffer_map (lcevc_meta->enhancement_data, &enhancement_info,
|
||||
GST_MAP_READ)) {
|
||||
GST_INFO_OBJECT (lcevc, "Could not map input buffer %ld enhancement data",
|
||||
input_buffer->pts);
|
||||
goto done;
|
||||
}
|
||||
|
||||
if (LCEVC_SendDecoderEnhancementData (lcevc->decoder_handle,
|
||||
input_buffer->pts, TRUE, enhancement_info.data,
|
||||
enhancement_info.size) != LCEVC_Success) {
|
||||
GST_INFO_OBJECT (lcevc,
|
||||
"Could not send input buffer %ld enhancement data with size %ld",
|
||||
input_buffer->pts, enhancement_info.size);
|
||||
goto done;
|
||||
}
|
||||
|
||||
GST_INFO_OBJECT (lcevc,
|
||||
"Sent input buffer %ld enhancement data with size %lu",
|
||||
input_buffer->pts, enhancement_info.size);
|
||||
|
||||
ret = TRUE;
|
||||
|
||||
done:
|
||||
gst_buffer_unmap (lcevc_meta->enhancement_data, &enhancement_info);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
send_base_picture (GstLcevcDec * lcevc, GstBuffer * input_buffer)
|
||||
{
|
||||
gboolean ret = FALSE;
|
||||
LCEVC_PictureHandle picture_handle;
|
||||
GstVideoFrame frame = { 0, };
|
||||
|
||||
if (!gst_video_frame_map (&frame, &lcevc->in_info, input_buffer,
|
||||
GST_MAP_READ)) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not map input buffer %ld", input_buffer->pts));
|
||||
goto done;
|
||||
}
|
||||
|
||||
if (!gst_lcevc_dec_utils_alloc_picture_handle (lcevc->decoder_handle, &frame,
|
||||
&picture_handle)) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not allocate input picture handle %ld", input_buffer->pts));
|
||||
goto done;
|
||||
}
|
||||
|
||||
if (LCEVC_SendDecoderBase (lcevc->decoder_handle, input_buffer->pts, TRUE,
|
||||
picture_handle, 1000000, NULL) != LCEVC_Success) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not send input buffer %ld base picture", input_buffer->pts));
|
||||
goto done;
|
||||
}
|
||||
|
||||
GST_INFO_OBJECT (lcevc, "Sent input buffer %ld base picture",
|
||||
input_buffer->pts);
|
||||
|
||||
ret = TRUE;
|
||||
|
||||
done:
|
||||
gst_video_frame_unmap (&frame);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
send_enhanced_picture (GstLcevcDec * lcevc, GstVideoCodecFrame * frame)
|
||||
{
|
||||
GstVideoCodecState *s = NULL;
|
||||
GstVideoFrame map = { 0, };
|
||||
PictureData *pd;
|
||||
gboolean res = FALSE;
|
||||
|
||||
/* Get output state */
|
||||
s = gst_video_decoder_get_output_state (GST_VIDEO_DECODER (lcevc));
|
||||
if (!s) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not get output state"));
|
||||
goto done;
|
||||
}
|
||||
|
||||
/* Map the video frame */
|
||||
if (!gst_video_frame_map (&map, &s->info, frame->output_buffer,
|
||||
GST_MAP_WRITE)) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not map output buffer for writting"));
|
||||
goto done;
|
||||
}
|
||||
|
||||
/* Get pic data if any and size didn't change, otherwise create a new one */
|
||||
pd = gst_mini_object_get_qdata (GST_MINI_OBJECT (frame->output_buffer),
|
||||
GST_LCEVC_DEC_PICTURE_DATA);
|
||||
if (!pd || pd->width != lcevc->out_width || pd->height != lcevc->out_height) {
|
||||
/* Create picture data */
|
||||
pd = picture_data_new (lcevc->decoder_handle, &map);
|
||||
if (!pd) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not create output picture data"));
|
||||
goto done;
|
||||
}
|
||||
|
||||
/* Set picture data on buffer */
|
||||
gst_mini_object_set_qdata (GST_MINI_OBJECT (frame->output_buffer),
|
||||
GST_LCEVC_DEC_PICTURE_DATA, pd, picture_data_free);
|
||||
}
|
||||
|
||||
/* Send enhanced picture */
|
||||
if (LCEVC_SendDecoderPicture (lcevc->decoder_handle, pd->picture_handle)
|
||||
!= LCEVC_Success) {
|
||||
GST_ELEMENT_ERROR (lcevc, STREAM, DECODE, (NULL),
|
||||
("Could not send output buffer enhanced picture"));
|
||||
goto done;
|
||||
}
|
||||
|
||||
res = TRUE;
|
||||
|
||||
done:
|
||||
gst_video_frame_unmap (&map);
|
||||
g_clear_pointer (&s, gst_video_codec_state_unref);
|
||||
return res;
|
||||
}
|
||||
|
||||
static GstFlowReturn
|
||||
gst_lcevc_dec_handle_frame (GstVideoDecoder * decoder,
|
||||
GstVideoCodecFrame * frame)
|
||||
{
|
||||
GstLcevcDec *lcevc = GST_LCEVC_DEC (decoder);
|
||||
|
||||
GST_DEBUG_OBJECT (decoder, "Handling frame %d with timestamp %ld",
|
||||
frame->system_frame_number, frame->input_buffer->pts);
|
||||
|
||||
if (!send_enhancement_data (lcevc, frame->input_buffer))
|
||||
goto error;
|
||||
|
||||
if (!send_base_picture (lcevc, frame->input_buffer))
|
||||
goto error;
|
||||
|
||||
if (gst_video_decoder_allocate_output_frame (decoder, frame) != GST_FLOW_OK) {
|
||||
GST_ELEMENT_ERROR (decoder, STREAM, DECODE, (NULL),
|
||||
("Could not allocate output frame"));
|
||||
goto error;
|
||||
}
|
||||
|
||||
if (!send_enhanced_picture (lcevc, frame))
|
||||
goto error;
|
||||
|
||||
if (!receive_enhanced_picture (lcevc))
|
||||
goto error;
|
||||
|
||||
if (!receive_base_picture (lcevc))
|
||||
goto error;
|
||||
|
||||
return GST_FLOW_OK;
|
||||
|
||||
error:
|
||||
gst_video_decoder_drop_frame (decoder, frame);
|
||||
return GST_FLOW_ERROR;
|
||||
}
|
||||
|
||||
static void
|
||||
gst_lcevc_dec_class_init (GstLcevcDecClass * klass)
|
||||
{
|
||||
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||||
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
|
||||
GstVideoDecoderClass *bt_class = GST_VIDEO_DECODER_CLASS (klass);
|
||||
|
||||
gst_element_class_add_static_pad_template (element_class,
|
||||
&sink_template_factory);
|
||||
gst_element_class_add_static_pad_template (element_class,
|
||||
&src_template_factory);
|
||||
gst_element_class_set_static_metadata (element_class,
|
||||
"LCEVC Decoder", "Codec/Decoder/Video",
|
||||
"Enhances video frames using attached LCEVC metadata",
|
||||
"Julian Bouzas <julian.bouzas@collabora.com>");
|
||||
|
||||
gobject_class->set_property = gst_lcevc_dec_set_property;
|
||||
gobject_class->get_property = gst_lcevc_dec_get_property;
|
||||
|
||||
bt_class->start = gst_lcevc_dec_start;
|
||||
bt_class->stop = gst_lcevc_dec_stop;
|
||||
bt_class->decide_allocation = gst_lcevc_dec_decide_allocation;
|
||||
bt_class->set_format = gst_lcevc_dec_set_format;
|
||||
bt_class->handle_frame = gst_lcevc_dec_handle_frame;
|
||||
|
||||
g_object_class_install_property (gobject_class, PROP_VERBOSE,
|
||||
g_param_spec_boolean ("verbose", "Verbose",
|
||||
"Output status information of the LCEVC Decoder SDK",
|
||||
FALSE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
g_object_class_install_property (gobject_class, PROP_MAX_WIDTH,
|
||||
g_param_spec_int ("max-width", "Maximum Width",
|
||||
"The maximum width for the LCEVC decoder (0 = default)",
|
||||
0, G_MAXINT, DEFAULT_MAX_WIDTH,
|
||||
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
g_object_class_install_property (gobject_class, PROP_MAX_HEIGHT,
|
||||
g_param_spec_int ("max-height", "Maximum Height",
|
||||
"The maximum height for the LCEVC decoder (0 = default)",
|
||||
0, G_MAXINT, DEFAULT_MAX_HEIGHT,
|
||||
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
g_object_class_install_property (gobject_class, PROP_MAX_LATENCY,
|
||||
g_param_spec_int ("max-latency", "Maximum Latency",
|
||||
"The maximum latency in frames for the LCEVC decoder (0 = default)",
|
||||
0, G_MAXINT, DEFAULT_MAX_LATENCY,
|
||||
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||
}
|
75
subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h
Normal file
75
subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h
Normal file
|
@ -0,0 +1,75 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2024> V-Nova International Limited
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Library General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Library General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Library General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
#ifndef __GST_LCEVC_DEC_H__
|
||||
#define __GST_LCEVC_DEC_H__
|
||||
|
||||
#include <gst/video/gstvideodecoder.h>
|
||||
#include <gst/video/video-info.h>
|
||||
|
||||
#include <LCEVC/lcevc_dec.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
#define GST_TYPE_LCEVC_DEC \
|
||||
(gst_lcevc_dec_get_type())
|
||||
#define GST_LCEVC_DEC(obj) \
|
||||
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_LCEVC_DEC,GstLcevcDec))
|
||||
#define GST_LCEVC_DEC_CLASS(klass) \
|
||||
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_LCEVC_DEC,GstLcevcDecClass))
|
||||
#define GST_IS_LCEVC_DEC(obj) \
|
||||
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_LCEVC_DEC))
|
||||
#define GST_IS_LCEVC_DEC_CLASS(klass) \
|
||||
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_LCEVC_DEC))
|
||||
|
||||
typedef struct _GstLcevcDec GstLcevcDec;
|
||||
typedef struct _GstLcevcDecClass GstLcevcDecClass;
|
||||
|
||||
struct _GstLcevcDec {
|
||||
GstVideoDecoder video_decoder;
|
||||
|
||||
/* Props */
|
||||
gboolean verbose;
|
||||
gint max_width;
|
||||
gint max_height;
|
||||
gint max_latency;
|
||||
|
||||
LCEVC_DecoderHandle decoder_handle;
|
||||
GstVideoInfo in_info;
|
||||
gboolean can_crop;
|
||||
|
||||
guint32 out_width;
|
||||
guint32 out_height;
|
||||
guint32 out_crop_top;
|
||||
guint32 out_crop_bottom;
|
||||
guint32 out_crop_left;
|
||||
guint32 out_crop_right;
|
||||
};
|
||||
|
||||
struct _GstLcevcDecClass {
|
||||
GstVideoDecoderClass video_decoder_class;
|
||||
};
|
||||
|
||||
GType gst_lcevc_dec_get_type(void);
|
||||
|
||||
GST_ELEMENT_REGISTER_DECLARE (lcevcdec);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* __GST_LCEVC_DEC_H__ */
|
|
@ -0,0 +1,92 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2024> V-Nova International Limited
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Library General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Library General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Library General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
#include "gstlcevcdecutils.h"
|
||||
|
||||
LCEVC_ColorFormat
|
||||
gst_lcevc_dec_utils_get_color_format (GstVideoFormat format)
|
||||
{
|
||||
switch (format) {
|
||||
case GST_VIDEO_FORMAT_I420:
|
||||
return LCEVC_I420_8;
|
||||
case GST_VIDEO_FORMAT_NV12:
|
||||
return LCEVC_NV12_8;
|
||||
case GST_VIDEO_FORMAT_NV21:
|
||||
return LCEVC_NV21_8;
|
||||
case GST_VIDEO_FORMAT_RGB:
|
||||
return LCEVC_RGB_8;
|
||||
case GST_VIDEO_FORMAT_BGR:
|
||||
return LCEVC_BGR_8;
|
||||
case GST_VIDEO_FORMAT_RGBA:
|
||||
return LCEVC_RGBA_8;
|
||||
case GST_VIDEO_FORMAT_BGRA:
|
||||
return LCEVC_BGRA_8;
|
||||
case GST_VIDEO_FORMAT_ARGB:
|
||||
return LCEVC_ARGB_8;
|
||||
case GST_VIDEO_FORMAT_ABGR:
|
||||
return LCEVC_ABGR_8;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return LCEVC_ColorFormat_Unknown;
|
||||
}
|
||||
|
||||
gboolean
|
||||
gst_lcevc_dec_utils_alloc_picture_handle (LCEVC_DecoderHandle decoder_handle,
|
||||
GstVideoFrame * frame, LCEVC_PictureHandle * picture_handle)
|
||||
{
|
||||
LCEVC_PictureDesc picture_desc = { 0, };
|
||||
LCEVC_PictureBufferDesc buffer_desc = { 0, };
|
||||
LCEVC_PicturePlaneDesc plane_desc[GST_VIDEO_MAX_PLANES] = { 0, };
|
||||
LCEVC_ColorFormat fmt;
|
||||
guint i;
|
||||
|
||||
fmt = gst_lcevc_dec_utils_get_color_format (GST_VIDEO_FRAME_FORMAT (frame));
|
||||
if (fmt == LCEVC_ColorFormat_Unknown)
|
||||
return FALSE;
|
||||
|
||||
/* Set LCEVC Picture Description */
|
||||
if (LCEVC_DefaultPictureDesc (&picture_desc, fmt,
|
||||
GST_VIDEO_FRAME_WIDTH (frame), GST_VIDEO_FRAME_HEIGHT (frame))
|
||||
!= LCEVC_Success)
|
||||
return FALSE;
|
||||
|
||||
/* Set buffer description */
|
||||
buffer_desc.data = GST_VIDEO_FRAME_PLANE_DATA (frame, 0);
|
||||
buffer_desc.byteSize = GST_VIDEO_FRAME_SIZE (frame);
|
||||
buffer_desc.access = LCEVC_Access_Write;
|
||||
|
||||
/* Set plane description */
|
||||
for (i = 0; i < GST_VIDEO_FRAME_N_PLANES (frame); i++) {
|
||||
plane_desc[i].firstSample = GST_VIDEO_FRAME_PLANE_DATA (frame, i);
|
||||
plane_desc[i].rowByteStride = GST_VIDEO_FRAME_PLANE_STRIDE (frame, i);
|
||||
}
|
||||
|
||||
/* FIXME: We set the stride on all the array (needed for LCEVCdec 2.0.0) */
|
||||
for (; i < GST_VIDEO_MAX_PLANES; i++)
|
||||
plane_desc[i].rowByteStride = GST_VIDEO_FRAME_WIDTH (frame);
|
||||
|
||||
/* Allocate LCEVC Picture */
|
||||
if (LCEVC_AllocPictureExternal (decoder_handle, &picture_desc, &buffer_desc,
|
||||
plane_desc, picture_handle) != LCEVC_Success)
|
||||
return FALSE;
|
||||
|
||||
return TRUE;
|
||||
}
|
|
@ -0,0 +1,42 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2024> V-Nova International Limited
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Library General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Library General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Library General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
#ifndef __GST_LCEVC_DEC_UTILS_H__
|
||||
#define __GST_LCEVC_DEC_UTILS_H__
|
||||
|
||||
#include <gst/gst.h>
|
||||
#include <gst/video/video-info.h>
|
||||
|
||||
#include <LCEVC/lcevc_dec.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
/* TODO: Only I420 and NV12 are currently working with the SDK */
|
||||
#define GST_LCEVC_DEC_UTILS_SUPPORTED_FORMATS \
|
||||
"{ I420, NV12 }"
|
||||
|
||||
LCEVC_ColorFormat gst_lcevc_dec_utils_get_color_format (GstVideoFormat format);
|
||||
|
||||
gboolean gst_lcevc_dec_utils_alloc_picture_handle (
|
||||
LCEVC_DecoderHandle decoder_handle, GstVideoFrame *frame,
|
||||
LCEVC_PictureHandle *picture_handle);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* __GST_LCEVC_DEC_UTILS_H__ */
|
26
subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build
Normal file
26
subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build
Normal file
|
@ -0,0 +1,26 @@
|
|||
lcevcdecoder_sources = [
|
||||
'plugin.c',
|
||||
'gstlcevcdecutils.c',
|
||||
'gstlcevcdec.c',
|
||||
'gstlcevcdecodebin.c',
|
||||
'gstlcevch264decodebin.c',
|
||||
]
|
||||
|
||||
lcevc_dec_dep = dependency ('lcevc_dec', required: get_option('lcevcdecoder'))
|
||||
|
||||
if lcevc_dec_dep.found()
|
||||
gstlcevcdecoder = library('gstlcevcdecoder',
|
||||
lcevcdecoder_sources,
|
||||
c_args : gst_plugins_bad_args,
|
||||
include_directories : [configinc],
|
||||
dependencies : [
|
||||
gstpbutils_dep,
|
||||
gstvideo_dep,
|
||||
gstcodecparsers_dep,
|
||||
lcevc_dec_dep,
|
||||
],
|
||||
install : true,
|
||||
install_dir : plugins_install_dir,
|
||||
)
|
||||
plugins += [gstlcevcdecoder]
|
||||
endif
|
42
subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c
Normal file
42
subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c
Normal file
|
@ -0,0 +1,42 @@
|
|||
/* GStreamer
|
||||
* Copyright (C) <2024> V-Nova International Limited
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Library General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library 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
|
||||
* Library General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Library General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
#include "config.h"
|
||||
#endif
|
||||
|
||||
#include <gst/gst.h>
|
||||
|
||||
#include "gstlcevcdec.h"
|
||||
|
||||
static gboolean
|
||||
plugin_init (GstPlugin * plugin)
|
||||
{
|
||||
gboolean ret = FALSE;
|
||||
|
||||
ret |= GST_ELEMENT_REGISTER (lcevcdec, plugin);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
||||
GST_VERSION_MINOR,
|
||||
lcevcdecoder,
|
||||
"LCEVC decoder",
|
||||
plugin_init, VERSION, GST_LICENSE, GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN)
|
|
@ -29,6 +29,7 @@ subdir('iqa')
|
|||
subdir('isac')
|
||||
subdir('ladspa')
|
||||
subdir('lc3')
|
||||
subdir('lcevcdecoder')
|
||||
subdir('ldac')
|
||||
subdir('libde265')
|
||||
subdir('lv2')
|
||||
|
|
|
@ -40,6 +40,7 @@ option('ivfparse', type : 'feature', value : 'auto')
|
|||
option('ivtc', type : 'feature', value : 'auto')
|
||||
option('jp2kdecimator', type : 'feature', value : 'auto')
|
||||
option('jpegformat', type : 'feature', value : 'auto')
|
||||
option('lcevcdecoder', type : 'feature', value : 'auto')
|
||||
option('librfb', type : 'feature', value : 'auto')
|
||||
option('midi', type : 'feature', value : 'auto')
|
||||
option('mpegdemux', type : 'feature', value : 'auto')
|
||||
|
|
Loading…
Reference in a new issue