xref: /OK3568_Linux_fs/external/gstreamer-rockchip/gst/rockchipmpp/gstmppvp8enc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright 2020 Rockchip Electronics Co., Ltd
3*4882a593Smuzhiyun  *     Author: Jeffy Chen <jeffy.chen@rock-chips.com>
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Copyright 2021 Rockchip Electronics Co., Ltd
6*4882a593Smuzhiyun  *     Author: Jeffy Chen <jeffy.chen@rock-chips.com>
7*4882a593Smuzhiyun  *
8*4882a593Smuzhiyun  * This library is free software; you can redistribute it and/or
9*4882a593Smuzhiyun  * modify it under the terms of the GNU Library General Public
10*4882a593Smuzhiyun  * License as published by the Free Software Foundation; either
11*4882a593Smuzhiyun  * version 2 of the License, or (at your option) any later version.
12*4882a593Smuzhiyun  *
13*4882a593Smuzhiyun  * This library is distributed in the hope that it will be useful,
14*4882a593Smuzhiyun  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15*4882a593Smuzhiyun  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16*4882a593Smuzhiyun  * Library General Public License for more details.
17*4882a593Smuzhiyun  *
18*4882a593Smuzhiyun  * You should have received a copy of the GNU Library General Public
19*4882a593Smuzhiyun  * License along with this library; if not, write to the
20*4882a593Smuzhiyun  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
21*4882a593Smuzhiyun  * Boston, MA 02110-1301, USA.
22*4882a593Smuzhiyun  *
23*4882a593Smuzhiyun  */
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #ifdef HAVE_CONFIG_H
26*4882a593Smuzhiyun #include "config.h"
27*4882a593Smuzhiyun #endif
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun #include <string.h>
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun #include "gstmppvp8enc.h"
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun #define GST_MPP_VP8_ENC(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), \
34*4882a593Smuzhiyun     GST_TYPE_MPP_VP8_ENC, GstMppVp8Enc))
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun #define GST_CAT_DEFAULT mpp_vp8_enc_debug
37*4882a593Smuzhiyun GST_DEBUG_CATEGORY (GST_CAT_DEFAULT);
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun struct _GstMppVp8Enc
40*4882a593Smuzhiyun {
41*4882a593Smuzhiyun   GstMppEnc parent;
42*4882a593Smuzhiyun 
43*4882a593Smuzhiyun   guint qp_init;
44*4882a593Smuzhiyun   guint qp_min;
45*4882a593Smuzhiyun   guint qp_max;
46*4882a593Smuzhiyun };
47*4882a593Smuzhiyun 
48*4882a593Smuzhiyun #define parent_class gst_mpp_vp8_enc_parent_class
49*4882a593Smuzhiyun G_DEFINE_TYPE (GstMppVp8Enc, gst_mpp_vp8_enc, GST_TYPE_MPP_ENC);
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun #define DEFAULT_PROP_QP_INIT 40
52*4882a593Smuzhiyun #define DEFAULT_PROP_QP_MIN 0
53*4882a593Smuzhiyun #define DEFAULT_PROP_QP_MAX 127
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun enum
56*4882a593Smuzhiyun {
57*4882a593Smuzhiyun   PROP_0,
58*4882a593Smuzhiyun   PROP_QP_INIT,
59*4882a593Smuzhiyun   PROP_QP_MIN,
60*4882a593Smuzhiyun   PROP_QP_MAX,
61*4882a593Smuzhiyun   PROP_LAST,
62*4882a593Smuzhiyun };
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun #define GST_MPP_VP8_ENC_SIZE_CAPS \
65*4882a593Smuzhiyun     "width  = (int) [ 128, MAX ], height = (int) [ 64, MAX ]"
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun static GstStaticPadTemplate gst_mpp_vp8_enc_src_template =
68*4882a593Smuzhiyun GST_STATIC_PAD_TEMPLATE ("src",
69*4882a593Smuzhiyun     GST_PAD_SRC,
70*4882a593Smuzhiyun     GST_PAD_ALWAYS,
71*4882a593Smuzhiyun     GST_STATIC_CAPS ("video/x-vp8, " GST_MPP_VP8_ENC_SIZE_CAPS));
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun static GstStaticPadTemplate gst_mpp_vp8_enc_sink_template =
74*4882a593Smuzhiyun GST_STATIC_PAD_TEMPLATE ("sink",
75*4882a593Smuzhiyun     GST_PAD_SINK,
76*4882a593Smuzhiyun     GST_PAD_ALWAYS,
77*4882a593Smuzhiyun     GST_STATIC_CAPS ("video/x-raw,"
78*4882a593Smuzhiyun         "format = (string) { " MPP_ENC_FORMATS " }, "
79*4882a593Smuzhiyun         GST_MPP_VP8_ENC_SIZE_CAPS));
80*4882a593Smuzhiyun 
81*4882a593Smuzhiyun static void
gst_mpp_vp8_enc_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)82*4882a593Smuzhiyun gst_mpp_vp8_enc_set_property (GObject * object,
83*4882a593Smuzhiyun     guint prop_id, const GValue * value, GParamSpec * pspec)
84*4882a593Smuzhiyun {
85*4882a593Smuzhiyun   GstVideoEncoder *encoder = GST_VIDEO_ENCODER (object);
86*4882a593Smuzhiyun   GstMppVp8Enc *self = GST_MPP_VP8_ENC (encoder);
87*4882a593Smuzhiyun   GstMppEnc *mppenc = GST_MPP_ENC (encoder);
88*4882a593Smuzhiyun 
89*4882a593Smuzhiyun   switch (prop_id) {
90*4882a593Smuzhiyun     case PROP_QP_INIT:{
91*4882a593Smuzhiyun       guint qp_init = g_value_get_uint (value);
92*4882a593Smuzhiyun       if (self->qp_init == qp_init)
93*4882a593Smuzhiyun         return;
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun       self->qp_init = qp_init;
96*4882a593Smuzhiyun       break;
97*4882a593Smuzhiyun     }
98*4882a593Smuzhiyun     case PROP_QP_MIN:{
99*4882a593Smuzhiyun       guint qp_min = g_value_get_uint (value);
100*4882a593Smuzhiyun       if (self->qp_min == qp_min)
101*4882a593Smuzhiyun         return;
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun       self->qp_min = qp_min;
104*4882a593Smuzhiyun       break;
105*4882a593Smuzhiyun     }
106*4882a593Smuzhiyun     case PROP_QP_MAX:{
107*4882a593Smuzhiyun       guint qp_max = g_value_get_uint (value);
108*4882a593Smuzhiyun       if (self->qp_max == qp_max)
109*4882a593Smuzhiyun         return;
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun       self->qp_max = qp_max;
112*4882a593Smuzhiyun       break;
113*4882a593Smuzhiyun     }
114*4882a593Smuzhiyun     default:
115*4882a593Smuzhiyun       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
116*4882a593Smuzhiyun       return;
117*4882a593Smuzhiyun   }
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun   mppenc->prop_dirty = TRUE;
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun static void
gst_mpp_vp8_enc_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)123*4882a593Smuzhiyun gst_mpp_vp8_enc_get_property (GObject * object,
124*4882a593Smuzhiyun     guint prop_id, GValue * value, GParamSpec * pspec)
125*4882a593Smuzhiyun {
126*4882a593Smuzhiyun   GstVideoEncoder *encoder = GST_VIDEO_ENCODER (object);
127*4882a593Smuzhiyun   GstMppVp8Enc *self = GST_MPP_VP8_ENC (encoder);
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun   switch (prop_id) {
130*4882a593Smuzhiyun     case PROP_QP_INIT:
131*4882a593Smuzhiyun       g_value_set_uint (value, self->qp_init);
132*4882a593Smuzhiyun       break;
133*4882a593Smuzhiyun     case PROP_QP_MIN:
134*4882a593Smuzhiyun       g_value_set_uint (value, self->qp_min);
135*4882a593Smuzhiyun       break;
136*4882a593Smuzhiyun     case PROP_QP_MAX:
137*4882a593Smuzhiyun       g_value_set_uint (value, self->qp_max);
138*4882a593Smuzhiyun       break;
139*4882a593Smuzhiyun     default:
140*4882a593Smuzhiyun       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
141*4882a593Smuzhiyun       break;
142*4882a593Smuzhiyun   }
143*4882a593Smuzhiyun }
144*4882a593Smuzhiyun 
145*4882a593Smuzhiyun static gboolean
gst_mpp_vp8_enc_apply_properties(GstVideoEncoder * encoder)146*4882a593Smuzhiyun gst_mpp_vp8_enc_apply_properties (GstVideoEncoder * encoder)
147*4882a593Smuzhiyun {
148*4882a593Smuzhiyun   GstMppVp8Enc *self = GST_MPP_VP8_ENC (encoder);
149*4882a593Smuzhiyun   GstMppEnc *mppenc = GST_MPP_ENC (encoder);
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun   if (!mppenc->prop_dirty)
152*4882a593Smuzhiyun     return TRUE;
153*4882a593Smuzhiyun 
154*4882a593Smuzhiyun   mpp_enc_cfg_set_s32 (mppenc->mpp_cfg, "vp8:qp_init", self->qp_init);
155*4882a593Smuzhiyun   mpp_enc_cfg_set_s32 (mppenc->mpp_cfg, "vp8:qp_max", self->qp_max);
156*4882a593Smuzhiyun   mpp_enc_cfg_set_s32 (mppenc->mpp_cfg, "vp8:qp_min", self->qp_min);
157*4882a593Smuzhiyun   mpp_enc_cfg_set_s32 (mppenc->mpp_cfg, "vp8:disable_ivf", 1);
158*4882a593Smuzhiyun 
159*4882a593Smuzhiyun   return gst_mpp_enc_apply_properties (encoder);
160*4882a593Smuzhiyun }
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun static gboolean
gst_mpp_vp8_enc_set_format(GstVideoEncoder * encoder,GstVideoCodecState * state)163*4882a593Smuzhiyun gst_mpp_vp8_enc_set_format (GstVideoEncoder * encoder,
164*4882a593Smuzhiyun     GstVideoCodecState * state)
165*4882a593Smuzhiyun {
166*4882a593Smuzhiyun   GstVideoEncoderClass *pclass = GST_VIDEO_ENCODER_CLASS (parent_class);
167*4882a593Smuzhiyun   GstCaps *caps;
168*4882a593Smuzhiyun 
169*4882a593Smuzhiyun   if (!pclass->set_format (encoder, state))
170*4882a593Smuzhiyun     return FALSE;
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun   if (!gst_mpp_vp8_enc_apply_properties (encoder))
173*4882a593Smuzhiyun     return FALSE;
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun   caps = gst_caps_new_empty_simple ("video/x-vp8");
176*4882a593Smuzhiyun   return gst_mpp_enc_set_src_caps (encoder, caps);
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun static GstFlowReturn
gst_mpp_vp8_enc_handle_frame(GstVideoEncoder * encoder,GstVideoCodecFrame * frame)180*4882a593Smuzhiyun gst_mpp_vp8_enc_handle_frame (GstVideoEncoder * encoder,
181*4882a593Smuzhiyun     GstVideoCodecFrame * frame)
182*4882a593Smuzhiyun {
183*4882a593Smuzhiyun   GstVideoEncoderClass *pclass = GST_VIDEO_ENCODER_CLASS (parent_class);
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun   if (G_UNLIKELY (!gst_mpp_vp8_enc_apply_properties (encoder))) {
186*4882a593Smuzhiyun     gst_video_codec_frame_unref (frame);
187*4882a593Smuzhiyun     return GST_FLOW_NOT_NEGOTIATED;
188*4882a593Smuzhiyun   }
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun   return pclass->handle_frame (encoder, frame);
191*4882a593Smuzhiyun }
192*4882a593Smuzhiyun 
193*4882a593Smuzhiyun static void
gst_mpp_vp8_enc_init(GstMppVp8Enc * self)194*4882a593Smuzhiyun gst_mpp_vp8_enc_init (GstMppVp8Enc * self)
195*4882a593Smuzhiyun {
196*4882a593Smuzhiyun   self->parent.mpp_type = MPP_VIDEO_CodingVP8;
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun   self->qp_init = DEFAULT_PROP_QP_INIT;
199*4882a593Smuzhiyun   self->qp_min = DEFAULT_PROP_QP_MIN;
200*4882a593Smuzhiyun   self->qp_max = DEFAULT_PROP_QP_MAX;
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun static void
gst_mpp_vp8_enc_class_init(GstMppVp8EncClass * klass)204*4882a593Smuzhiyun gst_mpp_vp8_enc_class_init (GstMppVp8EncClass * klass)
205*4882a593Smuzhiyun {
206*4882a593Smuzhiyun   GstVideoEncoderClass *encoder_class = GST_VIDEO_ENCODER_CLASS (klass);
207*4882a593Smuzhiyun   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
208*4882a593Smuzhiyun   GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun   GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT, "mppvp8enc", 0, "MPP VP8 encoder");
211*4882a593Smuzhiyun 
212*4882a593Smuzhiyun   encoder_class->set_format = GST_DEBUG_FUNCPTR (gst_mpp_vp8_enc_set_format);
213*4882a593Smuzhiyun   encoder_class->handle_frame =
214*4882a593Smuzhiyun       GST_DEBUG_FUNCPTR (gst_mpp_vp8_enc_handle_frame);
215*4882a593Smuzhiyun 
216*4882a593Smuzhiyun   gobject_class->set_property =
217*4882a593Smuzhiyun       GST_DEBUG_FUNCPTR (gst_mpp_vp8_enc_set_property);
218*4882a593Smuzhiyun   gobject_class->get_property =
219*4882a593Smuzhiyun       GST_DEBUG_FUNCPTR (gst_mpp_vp8_enc_get_property);
220*4882a593Smuzhiyun 
221*4882a593Smuzhiyun   g_object_class_install_property (gobject_class, PROP_QP_INIT,
222*4882a593Smuzhiyun       g_param_spec_uint ("qp-init", "Initial QP",
223*4882a593Smuzhiyun           "Initial QP (lower value means higher quality)",
224*4882a593Smuzhiyun           0, 127, DEFAULT_PROP_QP_INIT,
225*4882a593Smuzhiyun           G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun   g_object_class_install_property (gobject_class, PROP_QP_MIN,
228*4882a593Smuzhiyun       g_param_spec_uint ("qp-min", "Min QP",
229*4882a593Smuzhiyun           "Min QP", 0, 127, DEFAULT_PROP_QP_MIN,
230*4882a593Smuzhiyun           G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun   g_object_class_install_property (gobject_class, PROP_QP_MAX,
233*4882a593Smuzhiyun       g_param_spec_uint ("qp-max", "Max QP",
234*4882a593Smuzhiyun           "Max QP", 0, 127, DEFAULT_PROP_QP_MAX,
235*4882a593Smuzhiyun           G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
236*4882a593Smuzhiyun 
237*4882a593Smuzhiyun   gst_element_class_add_pad_template (element_class,
238*4882a593Smuzhiyun       gst_static_pad_template_get (&gst_mpp_vp8_enc_src_template));
239*4882a593Smuzhiyun 
240*4882a593Smuzhiyun   gst_element_class_add_pad_template (element_class,
241*4882a593Smuzhiyun       gst_static_pad_template_get (&gst_mpp_vp8_enc_sink_template));
242*4882a593Smuzhiyun 
243*4882a593Smuzhiyun   gst_element_class_set_static_metadata (element_class,
244*4882a593Smuzhiyun       "Rockchip Mpp VP8 Encoder", "Codec/Encoder/Video",
245*4882a593Smuzhiyun       "Encode video streams via Rockchip Mpp",
246*4882a593Smuzhiyun       "Jeffy Chen <jeffy.chen@rock-chips.com>");
247*4882a593Smuzhiyun }
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun gboolean
gst_mpp_vp8_enc_register(GstPlugin * plugin,guint rank)250*4882a593Smuzhiyun gst_mpp_vp8_enc_register (GstPlugin * plugin, guint rank)
251*4882a593Smuzhiyun {
252*4882a593Smuzhiyun   if (!gst_mpp_enc_supported (MPP_VIDEO_CodingVP8))
253*4882a593Smuzhiyun     return FALSE;
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun   if (g_getenv ("GST_MPP_VP8ENC_FAKE_VP8ENC"))
256*4882a593Smuzhiyun     gst_element_register (plugin, "vp8enc", GST_RANK_PRIMARY,
257*4882a593Smuzhiyun         gst_mpp_vp8_enc_get_type ());
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun   return gst_element_register (plugin, "mppvp8enc", rank,
260*4882a593Smuzhiyun       gst_mpp_vp8_enc_get_type ());
261*4882a593Smuzhiyun }
262