diff mbox series

[FFmpeg-devel,12/13] avcodec: add an LCEVC merger bsf

Message ID 20240831163114.4197-12-jamrial@gmail.com
State New
Headers show
Series [FFmpeg-devel,01/13,v3] avutil/frame: add an LCEVC enhancement data payload side data type | expand

Checks

Context Check Description
yinshiyou/make_loongarch64 success Make finished
yinshiyou/make_fate_loongarch64 success Make fate finished
andriy/make_x86 success Make finished
andriy/make_fate_x86 success Make fate finished

Commit Message

James Almer Aug. 31, 2024, 4:31 p.m. UTC
Signed-off-by: James Almer <jamrial@gmail.com>
---
 libavcodec/bitstream_filters.c   |   1 +
 libavcodec/bsf/Makefile          |   3 +-
 libavcodec/bsf/lcevc_merge_bsf.c | 265 +++++++++++++++++++++++++++++++
 3 files changed, 268 insertions(+), 1 deletion(-)
 create mode 100644 libavcodec/bsf/lcevc_merge_bsf.c
diff mbox series

Patch

diff --git a/libavcodec/bitstream_filters.c b/libavcodec/bitstream_filters.c
index f923411bee..fdd4fcf01b 100644
--- a/libavcodec/bitstream_filters.c
+++ b/libavcodec/bitstream_filters.c
@@ -45,6 +45,7 @@  extern const FFBitStreamFilter ff_hapqa_extract_bsf;
 extern const FFBitStreamFilter ff_hevc_metadata_bsf;
 extern const FFBitStreamFilter ff_hevc_mp4toannexb_bsf;
 extern const FFBitStreamFilter ff_imx_dump_header_bsf;
+extern const FFBitStreamFilter ff_lcevc_merge_bsf;
 extern const FFBitStreamFilter ff_media100_to_mjpegb_bsf;
 extern const FFBitStreamFilter ff_mjpeg2jpeg_bsf;
 extern const FFBitStreamFilter ff_mjpega_dump_header_bsf;
diff --git a/libavcodec/bsf/Makefile b/libavcodec/bsf/Makefile
index 40b7fc6e9b..850bcaf377 100644
--- a/libavcodec/bsf/Makefile
+++ b/libavcodec/bsf/Makefile
@@ -1,4 +1,4 @@ 
-clean::
+\clean::
 	$(RM) $(CLEANSUFFIXES:%=libavcodec/bsf/%)
 
 OBJS-$(CONFIG_AAC_ADTSTOASC_BSF)          += bsf/aac_adtstoasc.o
@@ -22,6 +22,7 @@  OBJS-$(CONFIG_HEVC_METADATA_BSF)          += bsf/h265_metadata.o
 OBJS-$(CONFIG_DOVI_RPU_BSF)               += bsf/dovi_rpu.o
 OBJS-$(CONFIG_HEVC_MP4TOANNEXB_BSF)       += bsf/hevc_mp4toannexb.o
 OBJS-$(CONFIG_IMX_DUMP_HEADER_BSF)        += bsf/imx_dump_header.o
+OBJS-$(CONFIG_LCEVC_MERGE_BSF)            += bsf/lcevc_merge_bsf.o
 OBJS-$(CONFIG_MEDIA100_TO_MJPEGB_BSF)     += bsf/media100_to_mjpegb.o
 OBJS-$(CONFIG_MJPEG2JPEG_BSF)             += bsf/mjpeg2jpeg.o
 OBJS-$(CONFIG_MJPEGA_DUMP_HEADER_BSF)     += bsf/mjpega_dump_header.o
diff --git a/libavcodec/bsf/lcevc_merge_bsf.c b/libavcodec/bsf/lcevc_merge_bsf.c
new file mode 100644
index 0000000000..fcaeb7ef7f
--- /dev/null
+++ b/libavcodec/bsf/lcevc_merge_bsf.c
@@ -0,0 +1,265 @@ 
+/*
+ * Copyright (c) 2022 James Almer
+ *
+ * This file is part of FFmpeg.
+ *
+ * FFmpeg 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.
+ *
+ * FFmpeg 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 FFmpeg; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+/**
+ * @file
+ * Derive PTS by reordering DTS from supported streams
+ */
+
+#include "libavutil/avassert.h"
+#include "libavutil/attributes.h"
+#include "libavutil/fifo.h"
+#include "libavutil/mem.h"
+#include "libavutil/opt.h"
+#include "libavutil/tree.h"
+
+#include "bsf.h"
+#include "bsf_internal.h"
+
+typedef struct LCEVCMergeContext {
+    const AVClass *class;
+
+    struct AVTreeNode *base;
+    struct AVTreeNode *enhancement;
+    AVFifo *nodes;
+
+    int base_idx, enhancement_idx;
+} LCEVCMergeContext;
+
+// AVTreeNode callbacks
+static int cmp_insert(const void *_key, const void *_node)
+{
+    const AVPacket *key = _key, *node = _node;
+
+    return FFDIFFSIGN(key->pts, node->pts);
+}
+
+static int cmp_find(const void *_key, const void *_node)
+{
+    int64_t key = *(const int64_t *)_key;
+    const AVPacket *node = _node;
+
+    return FFDIFFSIGN(key, node->pts);
+}
+
+#define WARN_BUFFERED(type)                                              \
+static int warn_##type##_buffered(void *logctx, void *elem)              \
+{                                                                        \
+    const AVPacket *pkt = (const AVPacket *)elem;                        \
+    av_log(logctx, AV_LOG_WARNING, #type" packet with PTS %"PRId64       \
+                                   " left buffered at EOF\n", pkt->pts); \
+    return 0;                                                            \
+}
+
+WARN_BUFFERED(base)
+WARN_BUFFERED(enhanced)
+
+static int lcevc_merge_init(AVBSFContext *ctx)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+
+    if (s->base_idx < 0 || s->enhancement_idx < 0) {
+        av_log(ctx, AV_LOG_ERROR, "Both base and enhancement stream index must be set\n");
+        return AVERROR(EINVAL);
+    }
+
+    s->nodes = av_fifo_alloc2(1, sizeof(struct AVTreeNode*), AV_FIFO_FLAG_AUTO_GROW);
+    if (!s->nodes)
+        return AVERROR(ENOMEM);
+
+    return 0;
+}
+
+static int merge_packet(AVBSFContext *ctx, struct AVTreeNode **root,
+                        AVPacket *out, AVPacket *in)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+    struct AVTreeNode *node = NULL;
+    uint8_t *side_data;
+    int ret;
+
+    // It doesn't matter if the packet is from the base or enhancement stream
+    // as both share the pts cmp_insert() will look for to remove the element.
+    av_tree_insert(root, in, cmp_insert, &node);
+    memset(node, 0, av_tree_node_size);
+    ret = av_fifo_write(s->nodes, &node, 1);
+    if (ret < 0) {
+        av_free(node);
+        return ret;
+    }
+
+    side_data = av_packet_new_side_data(out, AV_PKT_DATA_LCEVC, in->size);
+    if (!side_data)
+        return AVERROR(ENOMEM);
+
+    memcpy(side_data, in->data, in->size);
+
+    return 0;
+}
+
+static int buffer_packet(AVBSFContext *ctx, struct AVTreeNode **root,
+                         AVPacket **p_in)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+    AVPacket *in = *p_in, *pkt;
+    struct AVTreeNode *node = NULL;
+
+    if (av_fifo_can_read(s->nodes)) {
+        int av_unused ret = av_fifo_read(s->nodes, &node, 1);
+        av_assert2(ret >= 0);
+    } else
+        node = av_tree_node_alloc();
+    if (!node)
+        return AVERROR(ENOMEM);
+
+    pkt = av_tree_insert(root, in, cmp_insert, &node);
+    if (pkt && pkt != in) {
+        av_log(ctx, AV_LOG_ERROR, "Duplicate packet with PTS %"PRId64
+                                  " for stream_index %d \n", in->pts, in->stream_index);
+        av_free(node);
+        return AVERROR_INVALIDDATA;
+    }
+    *p_in = NULL;
+
+    return 0;
+}
+
+#define HANDLE_PACKET(type1, type2, pkt1, pkt2)                                       \
+static int handle_##type1##_packet(AVBSFContext *ctx, AVPacket *out, AVPacket **p_in) \
+{                                                                                     \
+    LCEVCMergeContext *s = ctx->priv_data;                                            \
+    AVPacket *in = *p_in, *pkt;                                                       \
+    int ret;                                                                          \
+                                                                                      \
+    pkt = av_tree_find(s->type2, &in->pts, cmp_find, NULL);                           \
+    if (pkt) {                                                                        \
+        ret = merge_packet(ctx, &s->type2, pkt1, pkt2);                               \
+        if (!ret)                                                                     \
+            av_packet_move_ref(out, pkt1);                                            \
+        av_packet_free(&pkt);                                                         \
+        av_packet_free(p_in);                                                         \
+        return ret;                                                                   \
+    }                                                                                 \
+                                                                                      \
+    return buffer_packet(ctx, &s->type1, p_in);                                       \
+}
+
+HANDLE_PACKET(base, enhancement, in, pkt)
+HANDLE_PACKET(enhancement, base, pkt, in)
+
+static int lcevc_merge_filter(AVBSFContext *ctx, AVPacket *out)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+    AVPacket *in;
+    int ret;
+
+    do {
+        ret = ff_bsf_get_packet(ctx, &in);
+        if (ret < 0) {
+            if (ret == AVERROR_EOF) {
+                av_tree_enumerate(s->base, ctx, NULL, warn_base_buffered);
+                av_tree_enumerate(s->enhancement, ctx, NULL, warn_enhanced_buffered);
+            }
+            return ret;
+        }
+
+        if (!in->size || in->pts < 0) {
+            ret = AVERROR_INVALIDDATA;
+            goto fail;
+        }
+
+        if (in->stream_index == s->base_idx)
+            ret = handle_base_packet(ctx, out, &in);
+        else if (in->stream_index == s->enhancement_idx)
+            ret = handle_enhancement_packet(ctx, out, &in);
+        else {
+            av_log(ctx, AV_LOG_ERROR, "Input packet neither base or enhacement\n");
+            ret = AVERROR(EINVAL);
+        }
+        if (ret < 0)
+            goto fail;
+    } while (!out->data);
+
+    ret = 0;
+fail:
+    if (ret < 0)
+        av_packet_free(&in);
+
+    return ret;
+}
+
+static int free_node(void *opaque, void *elem)
+{
+    AVPacket *pkt = elem;
+    av_packet_free(&pkt);
+    return 0;
+}
+
+static void lcevc_merge_flush(AVBSFContext *ctx)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+
+    av_tree_enumerate(s->base, NULL, NULL, free_node);
+    av_tree_enumerate(s->enhancement, NULL, NULL, free_node);
+    av_tree_destroy(s->base);
+    av_tree_destroy(s->enhancement);
+    s->base = NULL;
+    s->enhancement = NULL;
+}
+
+static void lcevc_merge_close(AVBSFContext *ctx)
+{
+    LCEVCMergeContext *s = ctx->priv_data;
+
+    lcevc_merge_flush(ctx);
+
+    if (s->nodes) {
+        struct AVTreeNode *node;
+        while (av_fifo_read(s->nodes, &node, 1) >= 0)
+            av_free(node);
+    }
+
+    av_fifo_freep2(&s->nodes);
+}
+
+#define OFFSET(x) offsetof(LCEVCMergeContext, x)
+#define FLAGS (AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_BSF_PARAM)
+static const AVOption lcevc_merge_options[] = {
+    { "base_idx", NULL, OFFSET(base_idx), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, INT_MAX, FLAGS },
+    { "enhancement_idx", NULL, OFFSET(enhancement_idx), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, INT_MAX, FLAGS },
+    { NULL }
+};
+
+static const AVClass lcevc_merge_class = {
+    .class_name = "lcevc_merge_bsf",
+    .item_name  = av_default_item_name,
+    .option     = lcevc_merge_options,
+    .version    = LIBAVUTIL_VERSION_INT,
+};
+
+const FFBitStreamFilter ff_lcevc_merge_bsf = {
+    .p.name         = "lcevc_merge",
+    .p.priv_class   = &lcevc_merge_class,
+    .priv_data_size = sizeof(LCEVCMergeContext),
+    .init           = lcevc_merge_init,
+    .flush          = lcevc_merge_flush,
+    .close          = lcevc_merge_close,
+    .filter         = lcevc_merge_filter,
+};