From 636690f2aaba7f8de742061a6ccec5122431bc90 Mon Sep 17 00:00:00 2001 From: Julian Bouzas Date: Tue, 1 Aug 2023 11:11:18 -0400 Subject: [PATCH] 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: --- .../ext/lcevcdecoder/gstlcevcdec.c | 720 ++++++++++++++++++ .../ext/lcevcdecoder/gstlcevcdec.h | 75 ++ .../ext/lcevcdecoder/gstlcevcdecutils.c | 92 +++ .../ext/lcevcdecoder/gstlcevcdecutils.h | 42 + .../ext/lcevcdecoder/meson.build | 26 + .../gst-plugins-bad/ext/lcevcdecoder/plugin.c | 42 + subprojects/gst-plugins-bad/ext/meson.build | 1 + subprojects/gst-plugins-bad/meson_options.txt | 1 + 8 files changed, 999 insertions(+) create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.c create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.h create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build create mode 100644 subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c new file mode 100644 index 0000000000..680adb2a35 --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.c @@ -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 +#include +#include + +#include + +#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 "); + + 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)); +} diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h new file mode 100644 index 0000000000..f24bf134fe --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdec.h @@ -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 +#include + +#include + +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__ */ diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.c b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.c new file mode 100644 index 0000000000..a8a0f680bf --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.c @@ -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; +} diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.h b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.h new file mode 100644 index 0000000000..59f0594f5d --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/gstlcevcdecutils.h @@ -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 +#include + +#include + +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__ */ diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build b/subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build new file mode 100644 index 0000000000..f95c83c2bb --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/meson.build @@ -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 diff --git a/subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c b/subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c new file mode 100644 index 0000000000..a15e89e088 --- /dev/null +++ b/subprojects/gst-plugins-bad/ext/lcevcdecoder/plugin.c @@ -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 + +#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) diff --git a/subprojects/gst-plugins-bad/ext/meson.build b/subprojects/gst-plugins-bad/ext/meson.build index fd35309d6c..2c5b212090 100644 --- a/subprojects/gst-plugins-bad/ext/meson.build +++ b/subprojects/gst-plugins-bad/ext/meson.build @@ -29,6 +29,7 @@ subdir('iqa') subdir('isac') subdir('ladspa') subdir('lc3') +subdir('lcevcdecoder') subdir('ldac') subdir('libde265') subdir('lv2') diff --git a/subprojects/gst-plugins-bad/meson_options.txt b/subprojects/gst-plugins-bad/meson_options.txt index 4f882176b8..20999d29e2 100644 --- a/subprojects/gst-plugins-bad/meson_options.txt +++ b/subprojects/gst-plugins-bad/meson_options.txt @@ -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')