ext/theora/: Added cropping option to theora decoder.

Original commit message from CVS:
* ext/theora/theoradec.c: (gst_theora_dec_class_init),
(gst_theora_dec_init), (theora_get_formats),
(theora_dec_src_convert), (theora_dec_sink_convert),
(theora_dec_src_query), (theora_dec_src_event), (theora_dec_event),
(theora_dec_chain), (theora_dec_set_property),
(theora_dec_get_property):
* ext/theora/theoraenc.c: (gst_border_mode_get_type),
(gst_theora_enc_class_init), (gst_theora_enc_init),
(theora_enc_sink_link), (theora_enc_chain),
(theora_enc_set_property), (theora_enc_get_property):
Added cropping option to theora decoder.
Added border option to theora encoder.
This commit is contained in:
Wim Taymans 2004-07-30 10:18:42 +00:00
parent 05584c6b22
commit fe21b56332
3 changed files with 197 additions and 21 deletions

View file

@ -1,3 +1,18 @@
2004-07-30 Wim Taymans <wim@fluendo.com>
* ext/theora/theoradec.c: (gst_theora_dec_class_init),
(gst_theora_dec_init), (theora_get_formats),
(theora_dec_src_convert), (theora_dec_sink_convert),
(theora_dec_src_query), (theora_dec_src_event), (theora_dec_event),
(theora_dec_chain), (theora_dec_set_property),
(theora_dec_get_property):
* ext/theora/theoraenc.c: (gst_border_mode_get_type),
(gst_theora_enc_class_init), (gst_theora_enc_init),
(theora_enc_sink_link), (theora_enc_chain),
(theora_enc_set_property), (theora_enc_get_property):
Added cropping option to theora decoder.
Added border option to theora encoder.
2004-07-30 Zaheer Abbas Merali <zaheerabbas at merali dot org> 2004-07-30 Zaheer Abbas Merali <zaheerabbas at merali dot org>
* ext/libpng/gstpngenc.c: (gst_pngenc_class_init), * ext/libpng/gstpngenc.c: (gst_pngenc_class_init),

View file

@ -60,6 +60,8 @@ struct _GstTheoraDec
gboolean need_keyframe; gboolean need_keyframe;
gint width, height; gint width, height;
gint offset_x, offset_y; gint offset_x, offset_y;
gboolean crop;
}; };
struct _GstTheoraDecClass struct _GstTheoraDecClass
@ -67,6 +69,13 @@ struct _GstTheoraDecClass
GstElementClass parent_class; GstElementClass parent_class;
}; };
#define THEORA_DEF_CROP TRUE
enum
{
ARG_0,
ARG_CROP
};
static GstElementDetails theora_dec_details = { static GstElementDetails theora_dec_details = {
"TheoraDec", "TheoraDec",
"Codec/Decoder/Video", "Codec/Decoder/Video",
@ -94,6 +103,11 @@ GST_STATIC_PAD_TEMPLATE ("sink",
GST_BOILERPLATE (GstTheoraDec, gst_theora_dec, GstElement, GST_TYPE_ELEMENT); GST_BOILERPLATE (GstTheoraDec, gst_theora_dec, GstElement, GST_TYPE_ELEMENT);
static void theora_dec_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec);
static void theora_dec_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec);
static void theora_dec_chain (GstPad * pad, GstData * data); static void theora_dec_chain (GstPad * pad, GstData * data);
static GstElementStateReturn theora_dec_change_state (GstElement * element); static GstElementStateReturn theora_dec_change_state (GstElement * element);
static gboolean theora_dec_src_event (GstPad * pad, GstEvent * event); static gboolean theora_dec_src_event (GstPad * pad, GstEvent * event);
@ -125,8 +139,17 @@ gst_theora_dec_base_init (gpointer g_class)
static void static void
gst_theora_dec_class_init (GstTheoraDecClass * klass) gst_theora_dec_class_init (GstTheoraDecClass * klass)
{ {
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass); GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass);
gobject_class->set_property = theora_dec_set_property;
gobject_class->get_property = theora_dec_get_property;
g_object_class_install_property (gobject_class, ARG_CROP,
g_param_spec_boolean ("crop", "Crop",
"Crop the image to the visible region", THEORA_DEF_CROP,
(GParamFlags) G_PARAM_READWRITE));
gstelement_class->change_state = theora_dec_change_state; gstelement_class->change_state = theora_dec_change_state;
GST_DEBUG_CATEGORY_INIT (theoradec_debug, "theoradec", 0, "Theora decoder"); GST_DEBUG_CATEGORY_INIT (theoradec_debug, "theoradec", 0, "Theora decoder");
@ -157,6 +180,8 @@ gst_theora_dec_init (GstTheoraDec * dec)
gst_element_add_pad (GST_ELEMENT (dec), dec->srcpad); gst_element_add_pad (GST_ELEMENT (dec), dec->srcpad);
GST_FLAG_SET (dec, GST_ELEMENT_EVENT_AWARE); GST_FLAG_SET (dec, GST_ELEMENT_EVENT_AWARE);
dec->crop = THEORA_DEF_CROP;
} }
/* FIXME: copy from libtheora, theora should somehow make this available for seeking */ /* FIXME: copy from libtheora, theora should somehow make this available for seeking */
@ -565,6 +590,7 @@ theora_dec_chain (GstPad * pad, GstData * data)
dec->info.frame_width, dec->info.frame_height, dec->info.frame_width, dec->info.frame_height,
dec->info.offset_x, dec->info.offset_y); dec->info.offset_x, dec->info.offset_y);
if (dec->crop) {
/* add black borders to make width/height/offsets even. we need this because /* add black borders to make width/height/offsets even. we need this because
* we cannot express an offset to the peer plugin. */ * we cannot express an offset to the peer plugin. */
dec->width = dec->width =
@ -573,6 +599,13 @@ theora_dec_chain (GstPad * pad, GstData * data)
ROUND_UP_2 (dec->info.frame_height + (dec->info.offset_y & 1)); ROUND_UP_2 (dec->info.frame_height + (dec->info.offset_y & 1));
dec->offset_x = dec->info.offset_x & ~1; dec->offset_x = dec->info.offset_x & ~1;
dec->offset_y = dec->info.offset_y & ~1; dec->offset_y = dec->info.offset_y & ~1;
} else {
/* no cropping, use the encoded dimensions */
dec->width = dec->info.width;
dec->height = dec->info.height;
dec->offset_x = 0;
dec->offset_y = 0;
}
GST_DEBUG_OBJECT (dec, "after fixup frame dimension %dx%d, offset %d:%d", GST_DEBUG_OBJECT (dec, "after fixup frame dimension %dx%d, offset %d:%d",
dec->width, dec->height, dec->offset_x, dec->offset_y); dec->width, dec->height, dec->offset_x, dec->offset_y);
@ -726,3 +759,35 @@ theora_dec_change_state (GstElement * element)
return parent_class->change_state (element); return parent_class->change_state (element);
} }
static void
theora_dec_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec)
{
GstTheoraDec *dec = GST_THEORA_DEC (object);
switch (prop_id) {
case ARG_CROP:
dec->crop = g_value_get_boolean (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
theora_dec_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec)
{
GstTheoraDec *dec = GST_THEORA_DEC (object);
switch (prop_id) {
case ARG_CROP:
g_value_set_boolean (value, dec->crop);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}

View file

@ -43,6 +43,33 @@ GST_DEBUG_CATEGORY (theoraenc_debug);
typedef struct _GstTheoraEnc GstTheoraEnc; typedef struct _GstTheoraEnc GstTheoraEnc;
typedef struct _GstTheoraEncClass GstTheoraEncClass; typedef struct _GstTheoraEncClass GstTheoraEncClass;
typedef enum
{
BORDER_NONE,
BORDER_BLACK,
BORDER_MIRROR
}
GstTheoraEncBorderMode;
#define GST_TYPE_BORDER_MODE (gst_border_mode_get_type())
static GType
gst_border_mode_get_type (void)
{
static GType border_mode_type = 0;
static GEnumValue border_mode[] = {
{BORDER_NONE, "BORDER_NONE", "No Border"},
{BORDER_BLACK, "BORDER_BLACK", "Black Border"},
{BORDER_MIRROR, "BORDER_MIRROR", "Mirror image in borders"},
{0, NULL, NULL},
};
if (!border_mode_type) {
border_mode_type =
g_enum_register_static ("GstTheoraEncBorderMode", border_mode);
}
return border_mode_type;
}
struct _GstTheoraEnc struct _GstTheoraEnc
{ {
GstElement element; GstElement element;
@ -57,6 +84,7 @@ struct _GstTheoraEnc
theora_comment comment; theora_comment comment;
gboolean center; gboolean center;
GstTheoraEncBorderMode border;
gint video_bitrate; /* bitrate target for Theora video */ gint video_bitrate; /* bitrate target for Theora video */
gint video_quality; /* Theora quality selector 0 = low, 63 = high */ gint video_quality; /* Theora quality selector 0 = low, 63 = high */
@ -88,6 +116,7 @@ struct _GstTheoraEncClass
#define ROUND_UP_8(x) (((x) + 7) & ~7) #define ROUND_UP_8(x) (((x) + 7) & ~7)
#define THEORA_DEF_CENTER TRUE #define THEORA_DEF_CENTER TRUE
#define THEORA_DEF_BORDER BORDER_BLACK
#define THEORA_DEF_BITRATE 0 #define THEORA_DEF_BITRATE 0
#define THEORA_DEF_QUALITY 16 #define THEORA_DEF_QUALITY 16
#define THEORA_DEF_QUICK TRUE #define THEORA_DEF_QUICK TRUE
@ -102,6 +131,7 @@ enum
{ {
ARG_0, ARG_0,
ARG_CENTER, ARG_CENTER,
ARG_BORDER,
ARG_BITRATE, ARG_BITRATE,
ARG_QUALITY, ARG_QUALITY,
ARG_QUICK, ARG_QUICK,
@ -174,6 +204,11 @@ gst_theora_enc_class_init (GstTheoraEncClass * klass)
g_param_spec_boolean ("center", "Center", g_param_spec_boolean ("center", "Center",
"Center image when sizes not multiple of 16", THEORA_DEF_CENTER, "Center image when sizes not multiple of 16", THEORA_DEF_CENTER,
(GParamFlags) G_PARAM_READWRITE)); (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_BORDER,
g_param_spec_enum ("border", "Border",
"Border color to add when sizes not multiple of 16",
GST_TYPE_BORDER_MODE, THEORA_DEF_BORDER,
(GParamFlags) G_PARAM_READWRITE));
/* general encoding stream options */ /* general encoding stream options */
g_object_class_install_property (gobject_class, ARG_BITRATE, g_object_class_install_property (gobject_class, ARG_BITRATE,
g_param_spec_int ("bitrate", "Bitrate", "Compressed video bitrate (kbps)", g_param_spec_int ("bitrate", "Bitrate", "Compressed video bitrate (kbps)",
@ -229,6 +264,9 @@ gst_theora_enc_init (GstTheoraEnc * enc)
gst_pad_use_explicit_caps (enc->srcpad); gst_pad_use_explicit_caps (enc->srcpad);
gst_element_add_pad (GST_ELEMENT (enc), enc->srcpad); gst_element_add_pad (GST_ELEMENT (enc), enc->srcpad);
enc->center = THEORA_DEF_CENTER;
enc->border = THEORA_DEF_BORDER;
enc->video_bitrate = THEORA_DEF_BITRATE; enc->video_bitrate = THEORA_DEF_BITRATE;
enc->video_quality = THEORA_DEF_QUALITY; enc->video_quality = THEORA_DEF_QUALITY;
enc->quick = THEORA_DEF_QUICK; enc->quick = THEORA_DEF_QUICK;
@ -494,6 +532,7 @@ theora_enc_chain (GstPad * pad, GstData * data)
gint dst_y_stride, dst_uv_stride; gint dst_y_stride, dst_uv_stride;
gint width, height; gint width, height;
gint cwidth, cheight; gint cwidth, cheight;
gint offset_x, right_x, right_border;
/* source width/height */ /* source width/height */
width = enc->width; width = enc->width;
@ -513,33 +552,74 @@ theora_enc_chain (GstPad * pad, GstData * data)
newbuf = gst_pad_alloc_buffer (enc->srcpad, newbuf = gst_pad_alloc_buffer (enc->srcpad,
GST_BUFFER_OFFSET_NONE, y_size * 3 / 2); GST_BUFFER_OFFSET_NONE, y_size * 3 / 2);
yuv.y = GST_BUFFER_DATA (newbuf); dest_y = yuv.y = GST_BUFFER_DATA (newbuf);
yuv.u = yuv.y + y_size; dest_u = yuv.u = yuv.y + y_size;
yuv.v = yuv.u + y_size / 4; dest_v = yuv.v = yuv.u + y_size / 4;
/* center if needed */
dest_y = yuv.y + enc->offset_x + enc->offset_y * dst_y_stride;
dest_u =
yuv.u + (enc->offset_x / 2) + (enc->offset_y / 2) * dst_uv_stride;
dest_v =
yuv.v + (enc->offset_x / 2) + (enc->offset_y / 2) * dst_uv_stride;
src_y = GST_BUFFER_DATA (buf); src_y = GST_BUFFER_DATA (buf);
src_u = src_y + src_y_stride * ROUND_UP_2 (height); src_u = src_y + src_y_stride * ROUND_UP_2 (height);
src_v = src_u + src_uv_stride * ROUND_UP_2 (height) / 2; src_v = src_u + src_uv_stride * ROUND_UP_2 (height) / 2;
if (enc->border != BORDER_NONE) {
/* fill top border */
for (i = 0; i < enc->offset_y; i++) {
memset (dest_y, 0, dst_y_stride);
dest_y += dst_y_stride;
}
} else {
dest_y += dst_y_stride * enc->offset_y;
}
offset_x = enc->offset_x;
right_x = width + enc->offset_x;
right_border = dst_y_stride - right_x;
/* copy Y plane */ /* copy Y plane */
for (i = 0; i < height; i++) { for (i = 0; i < height; i++) {
memcpy (dest_y, src_y, width); memcpy (dest_y + offset_x, src_y, width);
if (enc->border != BORDER_NONE) {
memset (dest_y, 0, offset_x);
memset (dest_y + right_x, 0, right_border);
}
dest_y += dst_y_stride; dest_y += dst_y_stride;
src_y += src_y_stride; src_y += src_y_stride;
} }
if (enc->border != BORDER_NONE) {
/* fill bottom border */
for (i = height + enc->offset_y; i < enc->info.height; i++) {
memset (dest_y, 0, dst_y_stride);
dest_y += dst_y_stride;
}
/* fill top border chroma */
for (i = 0; i < enc->offset_y / 2; i++) {
memset (dest_u, 128, dst_uv_stride);
memset (dest_v, 128, dst_uv_stride);
dest_u += dst_uv_stride;
dest_v += dst_uv_stride;
}
} else {
dest_u += dst_uv_stride * enc->offset_y / 2;
dest_v += dst_uv_stride * enc->offset_y / 2;
}
offset_x = enc->offset_x / 2;
right_x = cwidth + offset_x;
right_border = dst_uv_stride - right_x;
/* copy UV planes */ /* copy UV planes */
for (i = 0; i < cheight; i++) { for (i = 0; i < cheight; i++) {
memcpy (dest_u, src_u, cwidth); memcpy (dest_v + offset_x, src_v, cwidth);
memcpy (dest_v, src_v, cwidth); memcpy (dest_u + offset_x, src_u, cwidth);
if (enc->border != BORDER_NONE) {
memset (dest_u, 128, offset_x);
memset (dest_u + right_x, 128, right_border);
memset (dest_v, 128, offset_x);
memset (dest_v + right_x, 128, right_border);
}
dest_u += dst_uv_stride; dest_u += dst_uv_stride;
dest_v += dst_uv_stride; dest_v += dst_uv_stride;
@ -547,6 +627,16 @@ theora_enc_chain (GstPad * pad, GstData * data)
src_v += src_uv_stride; src_v += src_uv_stride;
} }
if (enc->border != BORDER_NONE) {
/* fill bottom border */
for (i = cheight + enc->offset_y / 2; i < enc->info_height / 2; i++) {
memset (dest_u, 128, dst_uv_stride);
memset (dest_v, 128, dst_uv_stride);
dest_u += dst_uv_stride;
dest_v += dst_uv_stride;
}
}
gst_buffer_unref (buf); gst_buffer_unref (buf);
buf = newbuf; buf = newbuf;
} }
@ -604,6 +694,9 @@ theora_enc_set_property (GObject * object, guint prop_id,
case ARG_CENTER: case ARG_CENTER:
enc->center = g_value_get_boolean (value); enc->center = g_value_get_boolean (value);
break; break;
case ARG_BORDER:
enc->border = g_value_get_enum (value);
break;
case ARG_BITRATE: case ARG_BITRATE:
enc->video_bitrate = g_value_get_int (value) * 1000; enc->video_bitrate = g_value_get_int (value) * 1000;
enc->video_quality = 0; enc->video_quality = 0;
@ -649,6 +742,9 @@ theora_enc_get_property (GObject * object, guint prop_id,
case ARG_CENTER: case ARG_CENTER:
g_value_set_boolean (value, enc->center); g_value_set_boolean (value, enc->center);
break; break;
case ARG_BORDER:
g_value_set_enum (value, enc->border);
break;
case ARG_BITRATE: case ARG_BITRATE:
g_value_set_int (value, enc->video_bitrate / 1000); g_value_set_int (value, enc->video_bitrate / 1000);
break; break;