mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-23 15:48:23 +00:00
9a2aa2ca77
This mirrors the behaviour in vp8enc / vp9enc and is generally more useful than using any framerate from the caps as it provides some degree of accuracy if the stream doesn't have timestamps perfectly according to the framerate. Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/6922>
1359 lines
48 KiB
C
1359 lines
48 KiB
C
/* GStreamer
|
|
* Copyright (C) <2017> Sean DuBois <sean@siobud.com>
|
|
*
|
|
* 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.
|
|
*/
|
|
/**
|
|
* SECTION:element-av1enc
|
|
*
|
|
* AV1 Encoder.
|
|
*
|
|
* ## Example launch line
|
|
*
|
|
* |[
|
|
* gst-launch-1.0 videotestsrc num-buffers=50 ! av1enc ! webmmux ! filesink location=av1.webm
|
|
* ]|
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "gstav1enc.h"
|
|
#include "gstav1utils.h"
|
|
#include <gst/video/video.h>
|
|
#include <gst/video/gstvideometa.h>
|
|
#include <gst/base/base.h>
|
|
|
|
#define GST_AV1_ENC_APPLY_CODEC_CONTROL(av1enc, flag, value) \
|
|
if (av1enc->encoder_inited) { \
|
|
if (aom_codec_control (&av1enc->encoder, flag, \
|
|
value) != AOM_CODEC_OK) { \
|
|
gst_av1_codec_error (&av1enc->encoder, "Failed to set " #flag); \
|
|
} \
|
|
}
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (av1_enc_debug);
|
|
#define GST_CAT_DEFAULT av1_enc_debug
|
|
|
|
#define GST_TYPE_RESIZE_MODE (gst_resize_mode_get_type())
|
|
static GType
|
|
gst_resize_mode_get_type (void)
|
|
{
|
|
static GType resize_mode_type = 0;
|
|
static const GEnumValue resize_mode[] = {
|
|
{GST_AV1_ENC_RESIZE_NONE, "No frame resizing allowed", "none"},
|
|
{GST_AV1_ENC_RESIZE_FIXED, "All frames are coded at the specified scale",
|
|
"fixed"},
|
|
{GST_AV1_ENC_RESIZE_RANDOM, "All frames are coded at a random scale",
|
|
"random"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!resize_mode_type) {
|
|
resize_mode_type =
|
|
g_enum_register_static ("GstAV1EncResizeMode", resize_mode);
|
|
}
|
|
return resize_mode_type;
|
|
}
|
|
|
|
#define GST_TYPE_SUPERRES_MODE (gst_superres_mode_get_type())
|
|
static GType
|
|
gst_superres_mode_get_type (void)
|
|
{
|
|
static GType superres_mode_type = 0;
|
|
static const GEnumValue superres_mode[] = {
|
|
{GST_AV1_ENC_SUPERRES_NONE, "No frame superres allowed", "none"},
|
|
{GST_AV1_ENC_SUPERRES_FIXED,
|
|
"All frames are coded at the specified scale and super-resolved",
|
|
"fixed"},
|
|
{GST_AV1_ENC_SUPERRES_RANDOM,
|
|
"All frames are coded at a random scale and super-resolved",
|
|
"random"},
|
|
{GST_AV1_ENC_SUPERRES_QTHRESH,
|
|
"Superres scale for a frame is determined based on q_index",
|
|
"qthresh"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!superres_mode_type) {
|
|
superres_mode_type =
|
|
g_enum_register_static ("GstAV1EncSuperresMode", superres_mode);
|
|
}
|
|
return superres_mode_type;
|
|
}
|
|
|
|
#define GST_TYPE_END_USAGE_MODE (gst_end_usage_mode_get_type())
|
|
static GType
|
|
gst_end_usage_mode_get_type (void)
|
|
{
|
|
static GType end_usage_mode_type = 0;
|
|
static const GEnumValue end_usage_mode[] = {
|
|
{GST_AV1_ENC_END_USAGE_VBR, "Variable Bit Rate Mode", "vbr"},
|
|
{GST_AV1_ENC_END_USAGE_CBR, "Constant Bit Rate Mode", "cbr"},
|
|
{GST_AV1_ENC_END_USAGE_CQ, "Constrained Quality Mode", "cq"},
|
|
{GST_AV1_ENC_END_USAGE_Q, "Constant Quality Mode", "q"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!end_usage_mode_type) {
|
|
end_usage_mode_type =
|
|
g_enum_register_static ("GstAV1EncEndUsageMode", end_usage_mode);
|
|
}
|
|
return end_usage_mode_type;
|
|
}
|
|
|
|
#define GST_TYPE_KF_MODE (gst_kf_mode_get_type())
|
|
static GType
|
|
gst_kf_mode_get_type (void)
|
|
{
|
|
static GType kf_mode_type = 0;
|
|
static const GEnumValue kf_mode[] = {
|
|
{GST_AV1_ENC_KF_AUTO,
|
|
"Encoder determines optimal keyframe placement automatically",
|
|
"auto"},
|
|
{GST_AV1_ENC_KF_DISABLED, "Encoder does not place keyframes", "disabled"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!kf_mode_type) {
|
|
kf_mode_type = g_enum_register_static ("GstAV1EncKFMode", kf_mode);
|
|
}
|
|
return kf_mode_type;
|
|
}
|
|
|
|
#define GST_TYPE_ENC_PASS (gst_enc_pass_get_type())
|
|
static GType
|
|
gst_enc_pass_get_type (void)
|
|
{
|
|
static GType enc_pass_type = 0;
|
|
static const GEnumValue enc_pass[] = {
|
|
{GST_AV1_ENC_ONE_PASS, "Single pass mode", "one-pass"},
|
|
{GST_AV1_ENC_FIRST_PASS, "First pass of multi-pass mode", "first-pass"},
|
|
{GST_AV1_ENC_SECOND_PASS, "Second pass of multi-pass mode", "second-pass"},
|
|
{GST_AV1_ENC_THIRD_PASS, "Third pass of multi-pass mode", "third-pass"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!enc_pass_type) {
|
|
enc_pass_type = g_enum_register_static ("GstAV1EncEncPass", enc_pass);
|
|
}
|
|
return enc_pass_type;
|
|
}
|
|
|
|
#define GST_TYPE_USAGE_PROFILE (gst_usage_profile_get_type())
|
|
static GType
|
|
gst_usage_profile_get_type (void)
|
|
{
|
|
static GType usage_profile_type = 0;
|
|
static const GEnumValue usage_profile[] = {
|
|
{GST_AV1_ENC_USAGE_GOOD_QUALITY, "Good Quality profile", "good-quality"},
|
|
{GST_AV1_ENC_USAGE_REALTIME, "Realtime profile", "realtime"},
|
|
{GST_AV1_ENC_USAGE_ALL_INTRA, "All Intra profile", "all-intra"},
|
|
{0, NULL, NULL},
|
|
};
|
|
|
|
if (!usage_profile_type) {
|
|
usage_profile_type =
|
|
g_enum_register_static ("GstAV1EncUsageProfile", usage_profile);
|
|
}
|
|
return usage_profile_type;
|
|
}
|
|
|
|
enum
|
|
{
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_CPU_USED,
|
|
PROP_DROP_FRAME,
|
|
PROP_RESIZE_MODE,
|
|
PROP_RESIZE_DENOMINATOR,
|
|
PROP_RESIZE_KF_DENOMINATOR,
|
|
PROP_SUPERRES_MODE,
|
|
PROP_SUPERRES_DENOMINATOR,
|
|
PROP_SUPERRES_KF_DENOMINATOR,
|
|
PROP_SUPERRES_QTHRESH,
|
|
PROP_SUPERRES_KF_QTHRESH,
|
|
PROP_END_USAGE,
|
|
PROP_TARGET_BITRATE,
|
|
PROP_MIN_QUANTIZER,
|
|
PROP_MAX_QUANTIZER,
|
|
PROP_UNDERSHOOT_PCT,
|
|
PROP_OVERSHOOT_PCT,
|
|
PROP_BUF_SZ,
|
|
PROP_BUF_INITIAL_SZ,
|
|
PROP_BUF_OPTIMAL_SZ,
|
|
PROP_THREADS,
|
|
PROP_ROW_MT,
|
|
PROP_TILE_COLUMNS,
|
|
PROP_TILE_ROWS,
|
|
PROP_KF_MODE,
|
|
PROP_ENC_PASS,
|
|
PROP_USAGE_PROFILE,
|
|
PROP_LAG_IN_FRAMES,
|
|
PROP_KEYFRAME_MAX_DIST
|
|
};
|
|
|
|
/* From av1/av1_cx_iface.c */
|
|
#define DEFAULT_PROFILE 0
|
|
#define DEFAULT_CPU_USED 0
|
|
#define DEFAULT_DROP_FRAME 0
|
|
#define DEFAULT_RESIZE_MODE GST_AV1_ENC_RESIZE_NONE
|
|
#define DEFAULT_RESIZE_DENOMINATOR 8
|
|
#define DEFAULT_RESIZE_KF_DENOMINATOR 8
|
|
#define DEFAULT_SUPERRES_MODE GST_AV1_ENC_SUPERRES_NONE
|
|
#define DEFAULT_SUPERRES_DENOMINATOR 8
|
|
#define DEFAULT_SUPERRES_KF_DENOMINATOR 8
|
|
#define DEFAULT_SUPERRES_QTHRESH 63
|
|
#define DEFAULT_SUPERRES_KF_QTHRESH 63
|
|
#define DEFAULT_END_USAGE GST_AV1_ENC_END_USAGE_VBR
|
|
#define DEFAULT_TARGET_BITRATE 256
|
|
#define DEFAULT_MIN_QUANTIZER 0
|
|
#define DEFAULT_MAX_QUANTIZER 0
|
|
#define DEFAULT_UNDERSHOOT_PCT 25
|
|
#define DEFAULT_OVERSHOOT_PCT 25
|
|
#define DEFAULT_BUF_SZ 6000
|
|
#define DEFAULT_BUF_INITIAL_SZ 4000
|
|
#define DEFAULT_BUF_OPTIMAL_SZ 5000
|
|
#define DEFAULT_TIMEBASE_N 1
|
|
#define DEFAULT_TIMEBASE_D 90000
|
|
#define DEFAULT_BIT_DEPTH AOM_BITS_8
|
|
#define DEFAULT_THREADS 0
|
|
#define DEFAULT_ROW_MT TRUE
|
|
#define DEFAULT_TILE_COLUMNS 0
|
|
#define DEFAULT_TILE_ROWS 0
|
|
#define DEFAULT_KF_MODE GST_AV1_ENC_KF_AUTO
|
|
#define DEFAULT_ENC_PASS GST_AV1_ENC_ONE_PASS
|
|
#define DEFAULT_USAGE_PROFILE GST_AV1_ENC_USAGE_GOOD_QUALITY
|
|
#define DEFAULT_LAG_IN_FRAMES 0
|
|
#define DEFAULT_KEYFRAME_MAX_DIST 30
|
|
|
|
static void gst_av1_enc_finalize (GObject * object);
|
|
static void gst_av1_enc_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec);
|
|
static void gst_av1_enc_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec);
|
|
|
|
static gboolean gst_av1_enc_start (GstVideoEncoder * encoder);
|
|
static gboolean gst_av1_enc_stop (GstVideoEncoder * encoder);
|
|
static gboolean gst_av1_enc_set_format (GstVideoEncoder * encoder,
|
|
GstVideoCodecState * state);
|
|
static GstFlowReturn gst_av1_enc_handle_frame (GstVideoEncoder * encoder,
|
|
GstVideoCodecFrame * frame);
|
|
static GstFlowReturn gst_av1_enc_finish (GstVideoEncoder * encoder);
|
|
static gboolean gst_av1_enc_propose_allocation (GstVideoEncoder * encoder,
|
|
GstQuery * query);
|
|
|
|
static void gst_av1_enc_destroy_encoder (GstAV1Enc * av1enc);
|
|
|
|
#define gst_av1_enc_parent_class parent_class
|
|
G_DEFINE_TYPE (GstAV1Enc, gst_av1_enc, GST_TYPE_VIDEO_ENCODER);
|
|
GST_ELEMENT_REGISTER_DEFINE (av1enc, "av1enc", GST_RANK_PRIMARY,
|
|
GST_TYPE_AV1_ENC);
|
|
|
|
/* *INDENT-OFF* */
|
|
static GstStaticPadTemplate gst_av1_enc_sink_pad_template =
|
|
GST_STATIC_PAD_TEMPLATE ("sink",
|
|
GST_PAD_SINK,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("video/x-raw, "
|
|
"format = (string) { I420, Y42B, Y444, YV12 }, "
|
|
"framerate = (fraction) [0, MAX], "
|
|
"width = (int) [ 4, MAX ], "
|
|
"height = (int) [ 4, MAX ]")
|
|
);
|
|
/* *INDENT-ON* */
|
|
|
|
static GstStaticPadTemplate gst_av1_enc_src_pad_template =
|
|
GST_STATIC_PAD_TEMPLATE ("src",
|
|
GST_PAD_SRC,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("video/x-av1, "
|
|
"stream-format = (string) obu-stream, " "alignment = (string) tu")
|
|
);
|
|
|
|
static void
|
|
gst_av1_enc_class_init (GstAV1EncClass * klass)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GstElementClass *element_class;
|
|
GstVideoEncoderClass *venc_class;
|
|
|
|
gobject_class = (GObjectClass *) klass;
|
|
element_class = (GstElementClass *) klass;
|
|
venc_class = (GstVideoEncoderClass *) klass;
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
gobject_class->finalize = gst_av1_enc_finalize;
|
|
gobject_class->set_property = gst_av1_enc_set_property;
|
|
gobject_class->get_property = gst_av1_enc_get_property;
|
|
|
|
gst_element_class_add_static_pad_template (element_class,
|
|
&gst_av1_enc_sink_pad_template);
|
|
gst_element_class_add_static_pad_template (element_class,
|
|
&gst_av1_enc_src_pad_template);
|
|
gst_element_class_set_static_metadata (element_class, "AV1 Encoder",
|
|
"Codec/Encoder/Video", "Encode AV1 video streams",
|
|
"Sean DuBois <sean@siobud.com>");
|
|
|
|
venc_class->start = gst_av1_enc_start;
|
|
venc_class->stop = gst_av1_enc_stop;
|
|
venc_class->set_format = gst_av1_enc_set_format;
|
|
venc_class->handle_frame = gst_av1_enc_handle_frame;
|
|
venc_class->finish = gst_av1_enc_finish;
|
|
venc_class->propose_allocation = gst_av1_enc_propose_allocation;
|
|
|
|
klass->codec_algo = &aom_codec_av1_cx_algo;
|
|
GST_DEBUG_CATEGORY_INIT (av1_enc_debug, "av1enc", 0, "AV1 encoding element");
|
|
|
|
g_object_class_install_property (gobject_class, PROP_CPU_USED,
|
|
g_param_spec_int ("cpu-used", "CPU Used",
|
|
"CPU Used. A Value greater than 0 will increase encoder speed at the expense of quality.",
|
|
0, 10, DEFAULT_CPU_USED, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/* Rate control configurations */
|
|
g_object_class_install_property (gobject_class, PROP_DROP_FRAME,
|
|
g_param_spec_uint ("drop-frame", "Drop frame",
|
|
"Temporal resampling configuration, drop frames as a strategy to meet "
|
|
"its target data rate Set to zero (0) to disable this feature.",
|
|
0, G_MAXUINT, DEFAULT_DROP_FRAME,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_RESIZE_MODE,
|
|
g_param_spec_enum ("resize-mode", "Resize mode",
|
|
"Frame resize mode", GST_TYPE_RESIZE_MODE,
|
|
DEFAULT_RESIZE_MODE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_RESIZE_DENOMINATOR,
|
|
g_param_spec_uint ("resize-denominator", "Resize denominator",
|
|
"Frame resize denominator, assuming 8 as the numerator",
|
|
8, 16, DEFAULT_RESIZE_DENOMINATOR,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_RESIZE_KF_DENOMINATOR,
|
|
g_param_spec_uint ("resize-kf-denominator", "Resize keyframe denominator",
|
|
"Frame resize keyframe denominator, assuming 8 as the numerator",
|
|
8, 16, DEFAULT_RESIZE_KF_DENOMINATOR,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SUPERRES_MODE,
|
|
g_param_spec_enum ("superres-mode", "Super-resolution scaling mode",
|
|
"It integrates upscaling after the encode/decode process",
|
|
GST_TYPE_SUPERRES_MODE,
|
|
DEFAULT_SUPERRES_MODE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SUPERRES_DENOMINATOR,
|
|
g_param_spec_uint ("superres-denominator", "Super-resolution denominator",
|
|
"Frame super-resolution denominator, used only by SUPERRES_FIXED mode",
|
|
8, 16, DEFAULT_SUPERRES_DENOMINATOR,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SUPERRES_KF_DENOMINATOR,
|
|
g_param_spec_uint ("superres-kf-denominator",
|
|
"Keyframe super-resolution denominator",
|
|
"Keyframe super-resolution denominator",
|
|
8, 16, DEFAULT_SUPERRES_KF_DENOMINATOR,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SUPERRES_QTHRESH,
|
|
g_param_spec_uint ("superres-qthresh",
|
|
"Frame super-resolution qindex threshold",
|
|
"Frame super-resolution qindex threshold, used only by SUPERRES_QTHRESH mode",
|
|
1, 63, DEFAULT_SUPERRES_QTHRESH,
|
|
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_SUPERRES_KF_QTHRESH,
|
|
g_param_spec_uint ("superres-kf-qthresh",
|
|
"Keyframe super-resolution qindex threshold",
|
|
"Keyframe super-resolution qindex threshold, used only by SUPERRES_QTHRESH mode",
|
|
1, 63, DEFAULT_SUPERRES_KF_QTHRESH,
|
|
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_END_USAGE,
|
|
g_param_spec_enum ("end-usage", "Rate control mode",
|
|
"Rate control algorithm to use, indicates the end usage of this stream",
|
|
GST_TYPE_END_USAGE_MODE, DEFAULT_END_USAGE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_TARGET_BITRATE,
|
|
g_param_spec_uint ("target-bitrate", "Target bitrate",
|
|
"Target bitrate, in kilobits per second",
|
|
1, G_MAXUINT, DEFAULT_TARGET_BITRATE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_MIN_QUANTIZER,
|
|
g_param_spec_uint ("min-quantizer", "Minimum (best quality) quantizer",
|
|
"Minimum (best quality) quantizer",
|
|
0, G_MAXUINT, DEFAULT_MIN_QUANTIZER,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_MAX_QUANTIZER,
|
|
g_param_spec_uint ("max-quantizer", "Maximum (worst quality) quantizer",
|
|
"Maximum (worst quality) quantizer",
|
|
0, G_MAXUINT, DEFAULT_MAX_QUANTIZER,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_UNDERSHOOT_PCT,
|
|
g_param_spec_uint ("undershoot-pct", "Datarate undershoot (min) target",
|
|
"Rate control adaptation undershoot control",
|
|
0, 1000, DEFAULT_UNDERSHOOT_PCT,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_OVERSHOOT_PCT,
|
|
g_param_spec_uint ("overshoot-pct", "Datarate overshoot (max) target",
|
|
"Rate control adaptation overshoot control",
|
|
0, 1000, DEFAULT_OVERSHOOT_PCT,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_BUF_SZ,
|
|
g_param_spec_uint ("buf-sz", "Decoder buffer size",
|
|
"Decoder buffer size, expressed in units of time (milliseconds)",
|
|
0, G_MAXUINT, DEFAULT_BUF_SZ,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_BUF_INITIAL_SZ,
|
|
g_param_spec_uint ("buf-initial-sz", "Decoder buffer initial size",
|
|
"Decoder buffer initial size, expressed in units of time (milliseconds)",
|
|
0, G_MAXUINT, DEFAULT_BUF_INITIAL_SZ,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_BUF_OPTIMAL_SZ,
|
|
g_param_spec_uint ("buf-optimal-sz", "Decoder buffer optimal size",
|
|
"Decoder buffer optimal size, expressed in units of time (milliseconds)",
|
|
0, G_MAXUINT, DEFAULT_BUF_OPTIMAL_SZ,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_THREADS,
|
|
g_param_spec_uint ("threads", "Max number of threads to use",
|
|
"Max number of threads to use encoding, set to 0 determine the "
|
|
"approximate number of threads that the system schedule",
|
|
0, G_MAXUINT, DEFAULT_THREADS,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
#ifdef AOM_CTRL_AV1E_SET_ROW_MT
|
|
g_object_class_install_property (gobject_class, PROP_ROW_MT,
|
|
g_param_spec_boolean ("row-mt", "Row based multi-threading",
|
|
"Enable row based multi-threading",
|
|
DEFAULT_ROW_MT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
#endif
|
|
|
|
g_object_class_install_property (gobject_class, PROP_TILE_COLUMNS,
|
|
g_param_spec_uint ("tile-columns", "Number of tile columns",
|
|
"Partition into separate vertical tile columns from image frame which "
|
|
"can enable parallel encoding",
|
|
0, 6, DEFAULT_TILE_COLUMNS,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (gobject_class, PROP_TILE_ROWS,
|
|
g_param_spec_uint ("tile-rows", "Number of tile rows",
|
|
"Partition into separate horizontal tile rows from image frame which "
|
|
"can enable parallel encoding",
|
|
0, 6, DEFAULT_TILE_ROWS, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* av1enc:keyframe-mode:
|
|
*
|
|
* Since: 1.22
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_KF_MODE,
|
|
g_param_spec_enum ("keyframe-mode", "Keyframe placement mode",
|
|
"Determines whether keyframes are placed automatically by the encoder",
|
|
GST_TYPE_KF_MODE, DEFAULT_KF_MODE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* av1enc:enc-pass:
|
|
*
|
|
* Since: 1.22
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_ENC_PASS,
|
|
g_param_spec_enum ("enc-pass", "Multi-pass Encoding Pass",
|
|
"Current phase for multi-pass encoding or @GST_AV1_ENC_ONE_PASS for single pass",
|
|
GST_TYPE_ENC_PASS, DEFAULT_ENC_PASS,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* av1enc:usage-profile:
|
|
*
|
|
* Since: 1.22
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_USAGE_PROFILE,
|
|
g_param_spec_enum ("usage-profile", "Usage value",
|
|
"Usage profile is used to guide the default config for the encoder",
|
|
GST_TYPE_USAGE_PROFILE, DEFAULT_USAGE_PROFILE,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* av1enc:lag-in-frames:
|
|
*
|
|
* Since: 1.22
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_LAG_IN_FRAMES,
|
|
g_param_spec_uint ("lag-in-frames", "Allow lagged encoding",
|
|
"Maximum number of future frames the encoder is allowed to consume "
|
|
"before producing the current output frame. "
|
|
"Set value to 0 for disabling lagged encoding.",
|
|
0, G_MAXUINT, DEFAULT_LAG_IN_FRAMES,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* av1enc:keyframe-max-dist:
|
|
*
|
|
* Since: 1.22
|
|
*/
|
|
g_object_class_install_property (gobject_class, PROP_KEYFRAME_MAX_DIST,
|
|
g_param_spec_int ("keyframe-max-dist", "Keyframe max distance",
|
|
"Maximum distance between keyframes (number of frames)",
|
|
0, G_MAXINT, DEFAULT_KEYFRAME_MAX_DIST,
|
|
(GParamFlags) (G_PARAM_READWRITE |
|
|
G_PARAM_STATIC_STRINGS | GST_PARAM_DOC_SHOW_DEFAULT)));
|
|
|
|
gst_type_mark_as_plugin_api (GST_TYPE_END_USAGE_MODE, 0);
|
|
gst_type_mark_as_plugin_api (GST_TYPE_RESIZE_MODE, 0);
|
|
gst_type_mark_as_plugin_api (GST_TYPE_SUPERRES_MODE, 0);
|
|
gst_type_mark_as_plugin_api (GST_TYPE_KF_MODE, 0);
|
|
gst_type_mark_as_plugin_api (GST_TYPE_ENC_PASS, 0);
|
|
gst_type_mark_as_plugin_api (GST_TYPE_USAGE_PROFILE, 0);
|
|
}
|
|
|
|
static void
|
|
gst_av1_codec_error (aom_codec_ctx_t * ctx, const char *s)
|
|
{
|
|
const char *detail = aom_codec_error_detail (ctx);
|
|
|
|
GST_ERROR ("%s: %s %s", s, aom_codec_error (ctx), detail ? detail : "");
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_init (GstAV1Enc * av1enc)
|
|
{
|
|
GST_PAD_SET_ACCEPT_TEMPLATE (GST_VIDEO_ENCODER_SINK_PAD (av1enc));
|
|
|
|
av1enc->encoder_inited = FALSE;
|
|
|
|
av1enc->cpu_used = DEFAULT_CPU_USED;
|
|
av1enc->format = AOM_IMG_FMT_I420;
|
|
av1enc->threads = DEFAULT_THREADS;
|
|
av1enc->row_mt = DEFAULT_ROW_MT;
|
|
av1enc->tile_columns = DEFAULT_TILE_COLUMNS;
|
|
av1enc->tile_rows = DEFAULT_TILE_ROWS;
|
|
|
|
#ifdef FIXED_QP_OFFSET_COUNT
|
|
av1enc->aom_cfg.fixed_qp_offsets[0] = -1;
|
|
av1enc->aom_cfg.fixed_qp_offsets[1] = -1;
|
|
av1enc->aom_cfg.fixed_qp_offsets[2] = -1;
|
|
av1enc->aom_cfg.fixed_qp_offsets[3] = -1;
|
|
av1enc->aom_cfg.fixed_qp_offsets[4] = -1;
|
|
#endif
|
|
av1enc->aom_cfg.kf_max_dist = DEFAULT_KEYFRAME_MAX_DIST;
|
|
av1enc->aom_cfg.rc_dropframe_thresh = DEFAULT_DROP_FRAME;
|
|
av1enc->aom_cfg.rc_resize_mode = DEFAULT_RESIZE_MODE;
|
|
av1enc->aom_cfg.rc_resize_denominator = DEFAULT_RESIZE_DENOMINATOR;
|
|
av1enc->aom_cfg.rc_resize_kf_denominator = DEFAULT_RESIZE_KF_DENOMINATOR;
|
|
av1enc->aom_cfg.rc_superres_mode = DEFAULT_SUPERRES_MODE;
|
|
av1enc->aom_cfg.rc_superres_denominator = DEFAULT_SUPERRES_DENOMINATOR;
|
|
av1enc->aom_cfg.rc_superres_kf_denominator = DEFAULT_SUPERRES_KF_DENOMINATOR;
|
|
av1enc->aom_cfg.rc_superres_qthresh = DEFAULT_SUPERRES_QTHRESH;
|
|
av1enc->aom_cfg.rc_superres_kf_qthresh = DEFAULT_SUPERRES_KF_QTHRESH;
|
|
av1enc->aom_cfg.rc_end_usage = (enum aom_rc_mode) DEFAULT_END_USAGE;
|
|
av1enc->aom_cfg.rc_target_bitrate = DEFAULT_TARGET_BITRATE;
|
|
av1enc->aom_cfg.rc_min_quantizer = DEFAULT_MIN_QUANTIZER;
|
|
av1enc->aom_cfg.rc_max_quantizer = DEFAULT_MAX_QUANTIZER;
|
|
av1enc->aom_cfg.rc_undershoot_pct = DEFAULT_UNDERSHOOT_PCT;
|
|
av1enc->aom_cfg.rc_overshoot_pct = DEFAULT_OVERSHOOT_PCT;
|
|
av1enc->aom_cfg.rc_buf_sz = DEFAULT_BUF_SZ;
|
|
av1enc->aom_cfg.rc_buf_initial_sz = DEFAULT_BUF_INITIAL_SZ;
|
|
av1enc->aom_cfg.rc_buf_optimal_sz = DEFAULT_BUF_OPTIMAL_SZ;
|
|
av1enc->aom_cfg.g_timebase.num = DEFAULT_TIMEBASE_N;
|
|
av1enc->aom_cfg.g_timebase.den = DEFAULT_TIMEBASE_D;
|
|
av1enc->aom_cfg.g_bit_depth = DEFAULT_BIT_DEPTH;
|
|
av1enc->aom_cfg.g_input_bit_depth = (unsigned int) DEFAULT_BIT_DEPTH;
|
|
av1enc->aom_cfg.kf_mode = (enum aom_kf_mode) DEFAULT_KF_MODE;
|
|
av1enc->aom_cfg.g_pass = (enum aom_enc_pass) DEFAULT_ENC_PASS;
|
|
av1enc->aom_cfg.g_usage = (unsigned int) DEFAULT_USAGE_PROFILE;
|
|
av1enc->aom_cfg.g_lag_in_frames = DEFAULT_LAG_IN_FRAMES;
|
|
|
|
g_mutex_init (&av1enc->encoder_lock);
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_finalize (GObject * object)
|
|
{
|
|
GstAV1Enc *av1enc = GST_AV1_ENC (object);
|
|
|
|
if (av1enc->input_state) {
|
|
gst_video_codec_state_unref (av1enc->input_state);
|
|
}
|
|
av1enc->input_state = NULL;
|
|
|
|
gst_av1_enc_destroy_encoder (av1enc);
|
|
g_mutex_clear (&av1enc->encoder_lock);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_set_latency (GstAV1Enc * av1enc)
|
|
{
|
|
GstClockTime latency;
|
|
gint fps_n, fps_d;
|
|
|
|
if (av1enc->input_state->info.fps_n && av1enc->input_state->info.fps_d) {
|
|
fps_n = av1enc->input_state->info.fps_n;
|
|
fps_d = av1enc->input_state->info.fps_d;
|
|
} else {
|
|
fps_n = 25;
|
|
fps_d = 1;
|
|
}
|
|
|
|
latency =
|
|
gst_util_uint64_scale (av1enc->aom_cfg.g_lag_in_frames * GST_SECOND,
|
|
fps_d, fps_n);
|
|
gst_video_encoder_set_latency (GST_VIDEO_ENCODER (av1enc), latency, latency);
|
|
|
|
GST_DEBUG_OBJECT (av1enc, "Latency set to %" GST_TIME_FORMAT
|
|
" = %d frames at %d/%d fps ", GST_TIME_ARGS (latency),
|
|
av1enc->aom_cfg.g_lag_in_frames, fps_n, fps_d);
|
|
}
|
|
|
|
static const gchar *
|
|
gst_av1_enc_get_aom_rc_mode_name (enum aom_rc_mode rc_mode)
|
|
{
|
|
switch (rc_mode) {
|
|
case AOM_VBR:
|
|
return "VBR (Variable Bit Rate)";
|
|
case AOM_CBR:
|
|
return "CBR (Constant Bit Rate)";
|
|
case AOM_CQ:
|
|
return "CQ (Constrained Quality)";
|
|
case AOM_Q:
|
|
return "Q (Constant Quality)";
|
|
default:
|
|
return "<UNKNOWN>";
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_debug_encoder_cfg (struct aom_codec_enc_cfg *cfg)
|
|
{
|
|
GST_DEBUG ("g_usage : %u", cfg->g_usage);
|
|
GST_DEBUG ("g_threads : %u", cfg->g_threads);
|
|
GST_DEBUG ("g_profile : %u", cfg->g_profile);
|
|
GST_DEBUG ("g_w x g_h : %u x %u", cfg->g_w, cfg->g_h);
|
|
GST_DEBUG ("g_bit_depth : %d", cfg->g_bit_depth);
|
|
GST_DEBUG ("g_input_bit_depth : %u", cfg->g_input_bit_depth);
|
|
GST_DEBUG ("g_timebase : %d / %d", cfg->g_timebase.num, cfg->g_timebase.den);
|
|
GST_DEBUG ("g_error_resilient : 0x%x", cfg->g_error_resilient);
|
|
GST_DEBUG ("g_pass : %d", cfg->g_pass);
|
|
GST_DEBUG ("g_lag_in_frames : %u", cfg->g_lag_in_frames);
|
|
GST_DEBUG ("rc_dropframe_thresh : %u", cfg->rc_dropframe_thresh);
|
|
GST_DEBUG ("rc_resize_mode : %u", cfg->rc_resize_mode);
|
|
GST_DEBUG ("rc_resize_denominator : %u", cfg->rc_resize_denominator);
|
|
GST_DEBUG ("rc_resize_kf_denominator : %u", cfg->rc_resize_kf_denominator);
|
|
GST_DEBUG ("rc_superres_mode : %u", cfg->rc_superres_mode);
|
|
GST_DEBUG ("rc_superres_denominator : %u", cfg->rc_superres_denominator);
|
|
GST_DEBUG ("rc_superres_kf_denominator : %u",
|
|
cfg->rc_superres_kf_denominator);
|
|
GST_DEBUG ("rc_superres_qthresh : %u", cfg->rc_superres_qthresh);
|
|
GST_DEBUG ("rc_superres_kf_qthresh : %u", cfg->rc_superres_kf_qthresh);
|
|
GST_DEBUG ("rc_end_usage : %s",
|
|
gst_av1_enc_get_aom_rc_mode_name (cfg->rc_end_usage));
|
|
/* rc_twopass_stats_in */
|
|
/* rc_firstpass_mb_stats_in */
|
|
GST_DEBUG ("rc_target_bitrate : %u (kbps)", cfg->rc_target_bitrate);
|
|
GST_DEBUG ("rc_min_quantizer : %u", cfg->rc_min_quantizer);
|
|
GST_DEBUG ("rc_max_quantizer : %u", cfg->rc_max_quantizer);
|
|
GST_DEBUG ("rc_undershoot_pct : %u", cfg->rc_undershoot_pct);
|
|
GST_DEBUG ("rc_overshoot_pct : %u", cfg->rc_overshoot_pct);
|
|
GST_DEBUG ("rc_buf_sz : %u (ms)", cfg->rc_buf_sz);
|
|
GST_DEBUG ("rc_buf_initial_sz : %u (ms)", cfg->rc_buf_initial_sz);
|
|
GST_DEBUG ("rc_buf_optimal_sz : %u (ms)", cfg->rc_buf_optimal_sz);
|
|
GST_DEBUG ("rc_2pass_vbr_bias_pct : %u (%%)", cfg->rc_2pass_vbr_bias_pct);
|
|
GST_DEBUG ("rc_2pass_vbr_minsection_pct : %u (%%)",
|
|
cfg->rc_2pass_vbr_minsection_pct);
|
|
GST_DEBUG ("rc_2pass_vbr_maxsection_pct : %u (%%)",
|
|
cfg->rc_2pass_vbr_maxsection_pct);
|
|
GST_DEBUG ("kf_mode : %u", cfg->kf_mode);
|
|
GST_DEBUG ("kf_min_dist : %u", cfg->kf_min_dist);
|
|
GST_DEBUG ("kf_max_dist : %u", cfg->kf_max_dist);
|
|
GST_DEBUG ("large_scale_tile : %u", cfg->large_scale_tile);
|
|
/* Tile-related values */
|
|
}
|
|
|
|
static gint
|
|
gst_av1_enc_get_downstream_profile (GstAV1Enc * av1enc)
|
|
{
|
|
GstCaps *allowed;
|
|
GstStructure *s;
|
|
gint profile = DEFAULT_PROFILE;
|
|
|
|
allowed = gst_pad_get_allowed_caps (GST_VIDEO_ENCODER_SRC_PAD (av1enc));
|
|
if (allowed) {
|
|
allowed = gst_caps_truncate (allowed);
|
|
s = gst_caps_get_structure (allowed, 0);
|
|
if (gst_structure_has_field (s, "profile")) {
|
|
const GValue *v = gst_structure_get_value (s, "profile");
|
|
const gchar *profile_str = NULL;
|
|
|
|
if (GST_VALUE_HOLDS_LIST (v) && gst_value_list_get_size (v) > 0) {
|
|
profile_str = g_value_get_string (gst_value_list_get_value (v, 0));
|
|
} else if (G_VALUE_HOLDS_STRING (v)) {
|
|
profile_str = g_value_get_string (v);
|
|
}
|
|
|
|
if (profile_str) {
|
|
gchar *endptr = NULL;
|
|
|
|
if (g_strcmp0 (profile_str, "main") == 0) {
|
|
GST_DEBUG_OBJECT (av1enc, "Downstream profile is \"main\"");
|
|
profile = 0;
|
|
} else if (g_strcmp0 (profile_str, "high") == 0) {
|
|
profile = 1;
|
|
GST_DEBUG_OBJECT (av1enc, "Downstream profile is \"high\"");
|
|
} else if (g_strcmp0 (profile_str, "professional") == 0) {
|
|
profile = 2;
|
|
GST_DEBUG_OBJECT (av1enc, "Downstream profile is \"professional\"");
|
|
} else {
|
|
profile = g_ascii_strtoull (profile_str, &endptr, 10);
|
|
if (*endptr != '\0' || profile < 0 || profile > 3) {
|
|
GST_ERROR_OBJECT (av1enc, "Invalid profile '%s'", profile_str);
|
|
profile = DEFAULT_PROFILE;
|
|
} else {
|
|
GST_DEBUG_OBJECT (av1enc,
|
|
"Downstream profile is \"%s\"", profile_str);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
gst_caps_unref (allowed);
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (av1enc, "Using profile %d", profile);
|
|
|
|
return profile;
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_adjust_profile (GstAV1Enc * av1enc, GstVideoFormat format)
|
|
{
|
|
guint depth = av1enc->aom_cfg.g_bit_depth;
|
|
guint profile = av1enc->aom_cfg.g_profile;
|
|
gboolean update = FALSE;
|
|
|
|
switch (profile) {
|
|
case 0:
|
|
if (depth < 12 && format == GST_VIDEO_FORMAT_Y444) {
|
|
profile = 1;
|
|
update = TRUE;
|
|
} else if (depth == 12 || format == GST_VIDEO_FORMAT_Y42B) {
|
|
profile = 2;
|
|
update = TRUE;
|
|
}
|
|
break;
|
|
case 1:
|
|
if (depth == 12 || format == GST_VIDEO_FORMAT_Y42B) {
|
|
profile = 2;
|
|
update = TRUE;
|
|
} else if (depth < 12 && format == GST_VIDEO_FORMAT_I420) {
|
|
profile = 0;
|
|
update = TRUE;
|
|
}
|
|
break;
|
|
case 2:
|
|
if (depth < 12) {
|
|
if (format == GST_VIDEO_FORMAT_Y444) {
|
|
profile = 1;
|
|
update = TRUE;
|
|
} else if (format == GST_VIDEO_FORMAT_I420) {
|
|
profile = 0;
|
|
update = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (update) {
|
|
GST_INFO_OBJECT (av1enc, "profile updated to %d from %d",
|
|
profile, av1enc->aom_cfg.g_profile);
|
|
av1enc->aom_cfg.g_profile = profile;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_av1_enc_set_format (GstVideoEncoder * encoder, GstVideoCodecState * state)
|
|
{
|
|
GstVideoCodecState *output_state;
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (encoder);
|
|
GstAV1EncClass *av1enc_class = GST_AV1_ENC_GET_CLASS (av1enc);
|
|
GstVideoInfo *info = &state->info;
|
|
|
|
output_state =
|
|
gst_video_encoder_set_output_state (encoder,
|
|
gst_pad_get_pad_template_caps (GST_VIDEO_ENCODER_SRC_PAD (encoder)),
|
|
state);
|
|
gst_video_codec_state_unref (output_state);
|
|
|
|
if (av1enc->input_state) {
|
|
gst_video_codec_state_unref (av1enc->input_state);
|
|
}
|
|
av1enc->input_state = gst_video_codec_state_ref (state);
|
|
|
|
g_mutex_lock (&av1enc->encoder_lock);
|
|
gst_av1_enc_set_latency (av1enc);
|
|
|
|
av1enc->aom_cfg.g_profile = gst_av1_enc_get_downstream_profile (av1enc);
|
|
|
|
/* Scale default bitrate to our size */
|
|
if (!av1enc->target_bitrate_set)
|
|
av1enc->aom_cfg.rc_target_bitrate =
|
|
gst_util_uint64_scale (DEFAULT_TARGET_BITRATE,
|
|
GST_VIDEO_INFO_WIDTH (info) * GST_VIDEO_INFO_HEIGHT (info), 320 * 240);
|
|
|
|
av1enc->aom_cfg.g_w = GST_VIDEO_INFO_WIDTH (info);
|
|
av1enc->aom_cfg.g_h = GST_VIDEO_INFO_HEIGHT (info);
|
|
/* Zero framerate and max-framerate but still need to setup the timebase to avoid
|
|
* a divide by zero error. Presuming the lowest common denominator will be RTP -
|
|
* VP8 payload draft states clock rate of 90000 which should work for anyone where
|
|
* FPS < 90000 (shouldn't be too many cases where it's higher) though wouldn't be optimal. RTP specification
|
|
* http://tools.ietf.org/html/draft-ietf-payload-vp8-01 section 6.3.1 */
|
|
av1enc->aom_cfg.g_timebase.num = 1;
|
|
av1enc->aom_cfg.g_timebase.den = 90000;
|
|
av1enc->aom_cfg.g_error_resilient = AOM_ERROR_RESILIENT_DEFAULT;
|
|
|
|
if (av1enc->threads == DEFAULT_THREADS)
|
|
av1enc->aom_cfg.g_threads = g_get_num_processors ();
|
|
else
|
|
av1enc->aom_cfg.g_threads = av1enc->threads;
|
|
/* TODO: do more configuration including bit_depth config */
|
|
|
|
av1enc->format =
|
|
gst_video_format_to_av1_img_format (GST_VIDEO_INFO_FORMAT (info));
|
|
|
|
if (av1enc->aom_cfg.g_bit_depth != DEFAULT_BIT_DEPTH) {
|
|
av1enc->aom_cfg.g_input_bit_depth = av1enc->aom_cfg.g_bit_depth;
|
|
if (av1enc->aom_cfg.g_bit_depth > 8)
|
|
av1enc->format |= AOM_IMG_FMT_HIGHBITDEPTH;
|
|
}
|
|
|
|
/* Adjust profile according to format and bit-depth */
|
|
gst_av1_enc_adjust_profile (av1enc, GST_VIDEO_INFO_FORMAT (info));
|
|
|
|
GST_DEBUG_OBJECT (av1enc, "Calling encoder init with config:");
|
|
gst_av1_enc_debug_encoder_cfg (&av1enc->aom_cfg);
|
|
|
|
if (aom_codec_enc_init (&av1enc->encoder, av1enc_class->codec_algo,
|
|
&av1enc->aom_cfg, 0)) {
|
|
gst_av1_codec_error (&av1enc->encoder, "Failed to initialize encoder");
|
|
g_mutex_unlock (&av1enc->encoder_lock);
|
|
return FALSE;
|
|
}
|
|
av1enc->encoder_inited = TRUE;
|
|
av1enc->last_pts = GST_CLOCK_TIME_NONE;
|
|
av1enc->last_input_duration = GST_CLOCK_TIME_NONE;
|
|
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AOME_SET_CPUUSED, av1enc->cpu_used);
|
|
#ifdef AOM_CTRL_AV1E_SET_ROW_MT
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_ROW_MT,
|
|
(av1enc->row_mt ? 1 : 0));
|
|
#endif
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_TILE_COLUMNS,
|
|
av1enc->tile_columns);
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_TILE_ROWS,
|
|
av1enc->tile_rows);
|
|
g_mutex_unlock (&av1enc->encoder_lock);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_av1_enc_process (GstAV1Enc * encoder)
|
|
{
|
|
aom_codec_iter_t iter = NULL;
|
|
const aom_codec_cx_pkt_t *pkt;
|
|
GstVideoCodecFrame *frame;
|
|
GstVideoEncoder *video_encoder;
|
|
GstFlowReturn ret = GST_FLOW_CUSTOM_SUCCESS;
|
|
|
|
video_encoder = GST_VIDEO_ENCODER (encoder);
|
|
|
|
while ((pkt = aom_codec_get_cx_data (&encoder->encoder, &iter)) != NULL) {
|
|
if (pkt->kind == AOM_CODEC_STATS_PKT) {
|
|
GST_WARNING_OBJECT (encoder, "Unhandled stats packet");
|
|
} else if (pkt->kind == AOM_CODEC_FPMB_STATS_PKT) {
|
|
GST_WARNING_OBJECT (encoder, "Unhandled FPMB pkt");
|
|
} else if (pkt->kind == AOM_CODEC_PSNR_PKT) {
|
|
GST_WARNING_OBJECT (encoder, "Unhandled PSNR packet");
|
|
} else if (pkt->kind == AOM_CODEC_CX_FRAME_PKT) {
|
|
frame = gst_video_encoder_get_oldest_frame (video_encoder);
|
|
g_assert (frame != NULL);
|
|
if ((pkt->data.frame.flags & AOM_FRAME_IS_KEY) != 0) {
|
|
GST_VIDEO_CODEC_FRAME_SET_SYNC_POINT (frame);
|
|
} else {
|
|
GST_VIDEO_CODEC_FRAME_UNSET_SYNC_POINT (frame);
|
|
}
|
|
|
|
frame->output_buffer =
|
|
gst_buffer_new_memdup (pkt->data.frame.buf, pkt->data.frame.sz);
|
|
|
|
if ((pkt->data.frame.flags & AOM_FRAME_IS_DROPPABLE) != 0)
|
|
GST_BUFFER_FLAG_SET (frame->output_buffer, GST_BUFFER_FLAG_DROPPABLE);
|
|
|
|
ret = gst_video_encoder_finish_frame (video_encoder, frame);
|
|
if (ret != GST_FLOW_OK)
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_fill_image (GstAV1Enc * enc, GstVideoFrame * frame,
|
|
aom_image_t * image)
|
|
{
|
|
image->planes[AOM_PLANE_Y] = GST_VIDEO_FRAME_COMP_DATA (frame, 0);
|
|
image->planes[AOM_PLANE_U] = GST_VIDEO_FRAME_COMP_DATA (frame, 1);
|
|
image->planes[AOM_PLANE_V] = GST_VIDEO_FRAME_COMP_DATA (frame, 2);
|
|
|
|
image->stride[AOM_PLANE_Y] = GST_VIDEO_FRAME_COMP_STRIDE (frame, 0);
|
|
image->stride[AOM_PLANE_U] = GST_VIDEO_FRAME_COMP_STRIDE (frame, 1);
|
|
image->stride[AOM_PLANE_V] = GST_VIDEO_FRAME_COMP_STRIDE (frame, 2);
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_av1_enc_handle_frame (GstVideoEncoder * encoder, GstVideoCodecFrame * frame)
|
|
{
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (encoder);
|
|
aom_image_t raw;
|
|
int flags = 0;
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
GstVideoFrame vframe;
|
|
aom_codec_pts_t scaled_pts;
|
|
GstClockTime pts_rt;
|
|
unsigned long duration;
|
|
|
|
if (!aom_img_alloc (&raw, av1enc->format, av1enc->aom_cfg.g_w,
|
|
av1enc->aom_cfg.g_h, 1)) {
|
|
GST_ERROR_OBJECT (encoder, "Failed to initialize encoder");
|
|
return FALSE;
|
|
}
|
|
|
|
gst_video_frame_map (&vframe, &av1enc->input_state->info,
|
|
frame->input_buffer, GST_MAP_READ);
|
|
gst_av1_enc_fill_image (av1enc, &vframe, &raw);
|
|
gst_video_frame_unmap (&vframe);
|
|
|
|
// aom_codec_encode requires pts to be strictly increasing
|
|
pts_rt =
|
|
gst_segment_to_running_time (&encoder->input_segment,
|
|
GST_FORMAT_TIME, frame->pts);
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (av1enc->last_pts)
|
|
&& pts_rt <= av1enc->last_pts) {
|
|
GST_WARNING_OBJECT (av1enc,
|
|
"decreasing pts %" GST_TIME_FORMAT " previous buffer was %"
|
|
GST_TIME_FORMAT " enforce increasing pts", GST_TIME_ARGS (pts_rt),
|
|
GST_TIME_ARGS (av1enc->last_pts));
|
|
pts_rt = av1enc->last_pts + 1;
|
|
}
|
|
|
|
av1enc->last_pts = pts_rt;
|
|
|
|
// Convert the pts from nanoseconds to timebase units
|
|
scaled_pts =
|
|
gst_util_uint64_scale (pts_rt,
|
|
av1enc->aom_cfg.g_timebase.den,
|
|
av1enc->aom_cfg.g_timebase.num * (GstClockTime) GST_SECOND);
|
|
|
|
if (frame->duration != GST_CLOCK_TIME_NONE) {
|
|
duration =
|
|
gst_util_uint64_scale_round (frame->duration,
|
|
av1enc->aom_cfg.g_timebase.den,
|
|
av1enc->aom_cfg.g_timebase.num * (GstClockTime) GST_SECOND);
|
|
|
|
if (duration > 0) {
|
|
av1enc->last_input_duration = frame->duration;
|
|
} else {
|
|
/* We force the path ignoring the duration if we end up with a zero
|
|
* value for duration after scaling (e.g. duration value too small) */
|
|
GST_WARNING_OBJECT (av1enc,
|
|
"Ignoring too small frame duration %" GST_TIME_FORMAT,
|
|
GST_TIME_ARGS (frame->duration));
|
|
duration = 1;
|
|
}
|
|
} else {
|
|
duration = 1;
|
|
}
|
|
|
|
if (aom_codec_encode (&av1enc->encoder, &raw, scaled_pts, duration, flags)
|
|
!= AOM_CODEC_OK) {
|
|
gst_av1_codec_error (&av1enc->encoder, "Failed to encode frame");
|
|
ret = GST_FLOW_ERROR;
|
|
}
|
|
|
|
aom_img_free (&raw);
|
|
gst_video_codec_frame_unref (frame);
|
|
|
|
if (ret == GST_FLOW_ERROR)
|
|
return ret;
|
|
|
|
ret = gst_av1_enc_process (av1enc);
|
|
|
|
if (ret == GST_FLOW_CUSTOM_SUCCESS)
|
|
ret = GST_FLOW_OK;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_av1_enc_finish (GstVideoEncoder * encoder)
|
|
{
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (encoder);
|
|
aom_codec_pts_t scaled_pts;
|
|
GstClockTime pts = 0;
|
|
|
|
while (ret == GST_FLOW_OK) {
|
|
GST_DEBUG_OBJECT (encoder, "Calling finish");
|
|
g_mutex_lock (&av1enc->encoder_lock);
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (av1enc->last_pts))
|
|
pts = av1enc->last_pts;
|
|
if (GST_CLOCK_TIME_IS_VALID (av1enc->last_input_duration))
|
|
pts += av1enc->last_input_duration;
|
|
|
|
scaled_pts =
|
|
gst_util_uint64_scale (pts,
|
|
av1enc->aom_cfg.g_timebase.den,
|
|
av1enc->aom_cfg.g_timebase.num * (GstClockTime) GST_SECOND);
|
|
|
|
if (aom_codec_encode (&av1enc->encoder, NULL, scaled_pts, 1, 0)
|
|
!= AOM_CODEC_OK) {
|
|
gst_av1_codec_error (&av1enc->encoder, "Failed to encode frame");
|
|
ret = GST_FLOW_ERROR;
|
|
}
|
|
g_mutex_unlock (&av1enc->encoder_lock);
|
|
|
|
ret = gst_av1_enc_process (av1enc);
|
|
}
|
|
|
|
|
|
if (ret == GST_FLOW_CUSTOM_SUCCESS)
|
|
ret = GST_FLOW_OK;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_destroy_encoder (GstAV1Enc * av1enc)
|
|
{
|
|
g_mutex_lock (&av1enc->encoder_lock);
|
|
if (av1enc->encoder_inited) {
|
|
aom_codec_destroy (&av1enc->encoder);
|
|
av1enc->encoder_inited = FALSE;
|
|
}
|
|
|
|
av1enc->last_pts = GST_CLOCK_TIME_NONE;
|
|
av1enc->last_input_duration = GST_CLOCK_TIME_NONE;
|
|
|
|
g_mutex_unlock (&av1enc->encoder_lock);
|
|
}
|
|
|
|
static gboolean
|
|
gst_av1_enc_propose_allocation (GstVideoEncoder * encoder, GstQuery * query)
|
|
{
|
|
gst_query_add_allocation_meta (query, GST_VIDEO_META_API_TYPE, NULL);
|
|
|
|
return GST_VIDEO_ENCODER_CLASS (parent_class)->propose_allocation (encoder,
|
|
query);
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (object);
|
|
gboolean global = FALSE;
|
|
aom_codec_err_t status;
|
|
|
|
GST_OBJECT_LOCK (av1enc);
|
|
|
|
g_mutex_lock (&av1enc->encoder_lock);
|
|
switch (prop_id) {
|
|
case PROP_CPU_USED:
|
|
av1enc->cpu_used = g_value_get_int (value);
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AOME_SET_CPUUSED,
|
|
av1enc->cpu_used);
|
|
break;
|
|
case PROP_DROP_FRAME:
|
|
av1enc->aom_cfg.rc_dropframe_thresh = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_RESIZE_MODE:
|
|
av1enc->aom_cfg.rc_resize_mode = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_RESIZE_DENOMINATOR:
|
|
av1enc->aom_cfg.rc_resize_denominator = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_RESIZE_KF_DENOMINATOR:
|
|
av1enc->aom_cfg.rc_resize_kf_denominator = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_SUPERRES_MODE:
|
|
av1enc->aom_cfg.rc_superres_mode = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_SUPERRES_DENOMINATOR:
|
|
av1enc->aom_cfg.rc_superres_denominator = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_SUPERRES_KF_DENOMINATOR:
|
|
av1enc->aom_cfg.rc_superres_kf_denominator = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_SUPERRES_QTHRESH:
|
|
av1enc->aom_cfg.rc_superres_qthresh = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_SUPERRES_KF_QTHRESH:
|
|
av1enc->aom_cfg.rc_superres_kf_qthresh = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_END_USAGE:
|
|
av1enc->aom_cfg.rc_end_usage = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_TARGET_BITRATE:
|
|
av1enc->aom_cfg.rc_target_bitrate = g_value_get_uint (value);
|
|
av1enc->target_bitrate_set = TRUE;
|
|
global = TRUE;
|
|
break;
|
|
case PROP_MIN_QUANTIZER:
|
|
av1enc->aom_cfg.rc_min_quantizer = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_MAX_QUANTIZER:
|
|
av1enc->aom_cfg.rc_max_quantizer = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_UNDERSHOOT_PCT:
|
|
av1enc->aom_cfg.rc_undershoot_pct = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_OVERSHOOT_PCT:
|
|
av1enc->aom_cfg.rc_overshoot_pct = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_BUF_SZ:
|
|
av1enc->aom_cfg.rc_buf_sz = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_BUF_INITIAL_SZ:
|
|
av1enc->aom_cfg.rc_buf_initial_sz = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_BUF_OPTIMAL_SZ:
|
|
av1enc->aom_cfg.rc_buf_optimal_sz = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_THREADS:
|
|
av1enc->threads = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
#ifdef AOM_CTRL_AV1E_SET_ROW_MT
|
|
case PROP_ROW_MT:
|
|
av1enc->row_mt = g_value_get_boolean (value);
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_ROW_MT,
|
|
(av1enc->row_mt ? 1 : 0));
|
|
break;
|
|
#endif
|
|
case PROP_TILE_COLUMNS:
|
|
av1enc->tile_columns = g_value_get_uint (value);
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_TILE_COLUMNS,
|
|
av1enc->tile_columns);
|
|
break;
|
|
case PROP_TILE_ROWS:
|
|
av1enc->tile_rows = g_value_get_uint (value);
|
|
GST_AV1_ENC_APPLY_CODEC_CONTROL (av1enc, AV1E_SET_TILE_ROWS,
|
|
av1enc->tile_rows);
|
|
break;
|
|
case PROP_KF_MODE:
|
|
av1enc->aom_cfg.kf_mode = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_ENC_PASS:
|
|
av1enc->aom_cfg.g_pass = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_USAGE_PROFILE:
|
|
av1enc->aom_cfg.g_usage = g_value_get_enum (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_LAG_IN_FRAMES:
|
|
av1enc->aom_cfg.g_lag_in_frames = g_value_get_uint (value);
|
|
global = TRUE;
|
|
break;
|
|
case PROP_KEYFRAME_MAX_DIST:
|
|
av1enc->aom_cfg.kf_max_dist = g_value_get_int (value);
|
|
global = TRUE;
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
|
|
if (global &&av1enc->encoder_inited) {
|
|
status = aom_codec_enc_config_set (&av1enc->encoder, &av1enc->aom_cfg);
|
|
GST_DEBUG_OBJECT (av1enc, "Set %s encoder configuration, ret = %s",
|
|
pspec->name, gst_av1_get_error_name (status));
|
|
}
|
|
|
|
g_mutex_unlock (&av1enc->encoder_lock);
|
|
GST_OBJECT_UNLOCK (av1enc);
|
|
}
|
|
|
|
static void
|
|
gst_av1_enc_get_property (GObject * object, guint prop_id, GValue * value,
|
|
GParamSpec * pspec)
|
|
{
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (object);
|
|
|
|
GST_OBJECT_LOCK (av1enc);
|
|
|
|
switch (prop_id) {
|
|
case PROP_CPU_USED:
|
|
g_value_set_int (value, av1enc->cpu_used);
|
|
break;
|
|
case PROP_DROP_FRAME:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_dropframe_thresh);
|
|
break;
|
|
case PROP_RESIZE_MODE:
|
|
g_value_set_enum (value, av1enc->aom_cfg.rc_resize_mode);
|
|
break;
|
|
case PROP_RESIZE_DENOMINATOR:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_resize_denominator);
|
|
break;
|
|
case PROP_RESIZE_KF_DENOMINATOR:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_resize_kf_denominator);
|
|
break;
|
|
case PROP_SUPERRES_MODE:
|
|
g_value_set_enum (value, av1enc->aom_cfg.rc_superres_mode);
|
|
break;
|
|
case PROP_SUPERRES_DENOMINATOR:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_superres_denominator);
|
|
break;
|
|
case PROP_SUPERRES_KF_DENOMINATOR:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_superres_kf_denominator);
|
|
break;
|
|
case PROP_SUPERRES_QTHRESH:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_superres_qthresh);
|
|
break;
|
|
case PROP_SUPERRES_KF_QTHRESH:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_superres_kf_qthresh);
|
|
break;
|
|
case PROP_END_USAGE:
|
|
g_value_set_enum (value, av1enc->aom_cfg.rc_end_usage);
|
|
break;
|
|
case PROP_TARGET_BITRATE:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_target_bitrate);
|
|
break;
|
|
case PROP_MIN_QUANTIZER:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_min_quantizer);
|
|
break;
|
|
case PROP_MAX_QUANTIZER:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_max_quantizer);
|
|
break;
|
|
case PROP_UNDERSHOOT_PCT:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_undershoot_pct);
|
|
break;
|
|
case PROP_OVERSHOOT_PCT:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_overshoot_pct);
|
|
break;
|
|
case PROP_BUF_SZ:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_buf_sz);
|
|
break;
|
|
case PROP_BUF_INITIAL_SZ:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_buf_initial_sz);
|
|
break;
|
|
case PROP_BUF_OPTIMAL_SZ:
|
|
g_value_set_uint (value, av1enc->aom_cfg.rc_buf_optimal_sz);
|
|
break;
|
|
case PROP_THREADS:
|
|
g_value_set_uint (value, av1enc->threads);
|
|
break;
|
|
#ifdef AOM_CTRL_AV1E_SET_ROW_MT
|
|
case PROP_ROW_MT:
|
|
g_value_set_boolean (value, av1enc->row_mt);
|
|
break;
|
|
#endif
|
|
case PROP_TILE_COLUMNS:
|
|
g_value_set_uint (value, av1enc->tile_columns);
|
|
break;
|
|
case PROP_TILE_ROWS:
|
|
g_value_set_uint (value, av1enc->tile_rows);
|
|
break;
|
|
case PROP_KF_MODE:
|
|
g_value_set_enum (value, av1enc->aom_cfg.kf_mode);
|
|
break;
|
|
case PROP_ENC_PASS:
|
|
g_value_set_enum (value, av1enc->aom_cfg.g_pass);
|
|
break;
|
|
case PROP_USAGE_PROFILE:
|
|
g_value_set_enum (value, av1enc->aom_cfg.g_usage);
|
|
break;
|
|
case PROP_LAG_IN_FRAMES:
|
|
g_value_set_uint (value, av1enc->aom_cfg.g_lag_in_frames);
|
|
break;
|
|
case PROP_KEYFRAME_MAX_DIST:
|
|
g_value_set_int (value, av1enc->aom_cfg.kf_max_dist);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (av1enc);
|
|
}
|
|
|
|
static gboolean
|
|
gst_av1_enc_start (GstVideoEncoder * encoder)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_av1_enc_stop (GstVideoEncoder * encoder)
|
|
{
|
|
GstAV1Enc *av1enc = GST_AV1_ENC_CAST (encoder);
|
|
|
|
if (av1enc->input_state) {
|
|
gst_video_codec_state_unref (av1enc->input_state);
|
|
}
|
|
av1enc->input_state = NULL;
|
|
|
|
gst_av1_enc_destroy_encoder (av1enc);
|
|
|
|
return TRUE;
|
|
}
|