Speexenc ported to 0.9.

Original commit message from CVS:
Speexenc ported to 0.9.
This commit is contained in:
Edgard Lima 2005-10-10 19:57:40 +00:00
parent 9566dd506b
commit ddecb1d34a
5 changed files with 245 additions and 194 deletions

View file

@ -1,3 +1,11 @@
2005-10-10 Edgard Lima <edgard.lima@indt.org.br>
* PORTED_09:
* ext/speex/Makefile.am:
* ext/speex/gstspeex.c:
* ext/speex/gstspeexenc.c:
Speexenc ported to 0.9
2005-10-10 Wim Taymans <wim@fluendo.com> 2005-10-10 Wim Taymans <wim@fluendo.com>
* sys/oss/gstosssink.c: (gst_oss_sink_class_init), * sys/oss/gstosssink.c: (gst_oss_sink_class_init),

View file

@ -1,6 +1,7 @@
When porting a plugin start with 0.8 CVS head, not the old code in this module. There are many bugfixes which have gone into 0.8 which you want to keep. When porting a plugin start with 0.8 CVS head, not the old code in this module. There are many bugfixes which have gone into 0.8 which you want to keep.
List of ported plugins (update when you commit a ported plugin): List of ported plugins (update when you commit a ported plugin):
speexenc (alima)
auparse (alima) auparse (alima)
effectv (wim) effectv (wim)
mad (wim) mad (wim)

View file

@ -1,9 +1,9 @@
plugin_LTLIBRARIES = libgstspeex.la plugin_LTLIBRARIES = libgstspeex.la
libgstspeex_la_SOURCES = gstspeex.c gstspeexdec.c libgstspeex_la_SOURCES = gstspeex.c gstspeexdec.c gstspeexenc.c
#gstspeexenc.c libgstspeex_la_CFLAGS = $(GST_CFLAGS) $(SPEEX_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS)
libgstspeex_la_CFLAGS = $(GST_CFLAGS) $(SPEEX_CFLAGS) libgstspeex_la_LIBADD = $(GST_LIBS) $(SPEEX_LIBS) $(GST_PLUGINS_BASE_LIBS) \
libgstspeex_la_LIBADD = $(GST_LIBS) $(SPEEX_LIBS) -lgsttagedit-@GST_MAJORMINOR@
libgstspeex_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS) libgstspeex_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS) $(GST_BASE_LIBS)
noinst_HEADERS = gstspeexenc.h gstspeexdec.h noinst_HEADERS = gstspeexenc.h gstspeexdec.h

View file

@ -26,11 +26,11 @@
static gboolean static gboolean
plugin_init (GstPlugin * plugin) plugin_init (GstPlugin * plugin)
{ {
#if 0
if (!gst_element_register (plugin, "speexenc", GST_RANK_NONE, if (!gst_element_register (plugin, "speexenc", GST_RANK_NONE,
GST_TYPE_SPEEXENC)) GST_TYPE_SPEEXENC))
return FALSE; return FALSE;
#endif
if (!gst_element_register (plugin, "speexdec", GST_RANK_PRIMARY, if (!gst_element_register (plugin, "speexdec", GST_RANK_PRIMARY,
GST_TYPE_SPEEXDEC)) GST_TYPE_SPEEXDEC))
return FALSE; return FALSE;

View file

@ -75,6 +75,7 @@ enum
ARG_LAST_MESSAGE ARG_LAST_MESSAGE
}; };
#if 0
static const GstFormat * static const GstFormat *
gst_speexenc_get_formats (GstPad * pad) gst_speexenc_get_formats (GstPad * pad)
{ {
@ -92,12 +93,14 @@ gst_speexenc_get_formats (GstPad * pad)
return (GST_PAD_IS_SRC (pad) ? src_formats : sink_formats); return (GST_PAD_IS_SRC (pad) ? src_formats : sink_formats);
} }
#endif
static void gst_speexenc_base_init (gpointer g_class); static void gst_speexenc_base_init (gpointer g_class);
static void gst_speexenc_class_init (GstSpeexEncClass * klass); static void gst_speexenc_class_init (GstSpeexEncClass * klass);
static void gst_speexenc_init (GstSpeexEnc * speexenc); static void gst_speexenc_init (GstSpeexEnc * speexenc);
static void gst_speexenc_chain (GstPad * pad, GstData * _data); static gboolean gst_speexenc_sinkevent (GstPad * pad, GstEvent * event);
static GstFlowReturn gst_speexenc_chain (GstPad * pad, GstBuffer * buf);
static gboolean gst_speexenc_setup (GstSpeexEnc * speexenc); static gboolean gst_speexenc_setup (GstSpeexEnc * speexenc);
static void gst_speexenc_get_property (GObject * object, guint prop_id, static void gst_speexenc_get_property (GObject * object, guint prop_id,
@ -232,13 +235,13 @@ gst_speexenc_class_init (GstSpeexEncClass * klass)
gstelement_class->change_state = gst_speexenc_change_state; gstelement_class->change_state = gst_speexenc_change_state;
} }
static GstPadLinkReturn static gboolean
gst_speexenc_sinkconnect (GstPad * pad, const GstCaps * caps) gst_speexenc_sink_setcaps (GstPad * pad, GstCaps * caps)
{ {
GstSpeexEnc *speexenc; GstSpeexEnc *speexenc;
GstStructure *structure; GstStructure *structure;
speexenc = GST_SPEEXENC (gst_pad_get_parent (pad)); speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
speexenc->setup = FALSE; speexenc->setup = FALSE;
structure = gst_caps_get_structure (caps, 0); structure = gst_caps_get_structure (caps, 0);
@ -248,9 +251,9 @@ gst_speexenc_sinkconnect (GstPad * pad, const GstCaps * caps)
gst_speexenc_setup (speexenc); gst_speexenc_setup (speexenc);
if (speexenc->setup) if (speexenc->setup)
return GST_PAD_LINK_OK; return TRUE;
return GST_PAD_LINK_REFUSED; return FALSE;
} }
static gboolean static gboolean
@ -261,7 +264,7 @@ gst_speexenc_convert_src (GstPad * pad, GstFormat src_format, gint64 src_value,
GstSpeexEnc *speexenc; GstSpeexEnc *speexenc;
gint64 avg; gint64 avg;
speexenc = GST_SPEEXENC (gst_pad_get_parent (pad)); speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
if (speexenc->samples_in == 0 || if (speexenc->samples_in == 0 ||
speexenc->bytes_out == 0 || speexenc->rate == 0) speexenc->bytes_out == 0 || speexenc->rate == 0)
@ -303,7 +306,7 @@ gst_speexenc_convert_sink (GstPad * pad, GstFormat src_format,
gint bytes_per_sample; gint bytes_per_sample;
GstSpeexEnc *speexenc; GstSpeexEnc *speexenc;
speexenc = GST_SPEEXENC (gst_pad_get_parent (pad)); speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
bytes_per_sample = speexenc->channels * 2; bytes_per_sample = speexenc->channels * 2;
@ -364,7 +367,6 @@ static const GstQueryType *
gst_speexenc_get_query_types (GstPad * pad) gst_speexenc_get_query_types (GstPad * pad)
{ {
static const GstQueryType gst_speexenc_src_query_types[] = { static const GstQueryType gst_speexenc_src_query_types[] = {
GST_QUERY_TOTAL,
GST_QUERY_POSITION, GST_QUERY_POSITION,
0 0
}; };
@ -373,17 +375,17 @@ gst_speexenc_get_query_types (GstPad * pad)
} }
static gboolean static gboolean
gst_speexenc_src_query (GstPad * pad, GstQueryType type, gst_speexenc_src_query (GstPad * pad, GstQuery * query)
GstFormat * format, gint64 * value)
{ {
gboolean res = TRUE; gboolean res = TRUE;
GstSpeexEnc *speexenc; GstSpeexEnc *speexenc;
speexenc = GST_SPEEXENC (gst_pad_get_parent (pad)); speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
switch (type) { switch (GST_QUERY_TYPE (query)) {
case GST_QUERY_TOTAL: case GST_QUERY_POSITION:
{ {
#if 0
switch (*format) { switch (*format) {
case GST_FORMAT_BYTES: case GST_FORMAT_BYTES:
case GST_FORMAT_TIME: case GST_FORMAT_TIME:
@ -416,27 +418,60 @@ gst_speexenc_src_query (GstPad * pad, GstQueryType type,
} }
break; break;
} }
default: }
#endif
res = FALSE; res = FALSE;
break; break;
} }
break; case GST_QUERY_CONVERT:
}
case GST_QUERY_POSITION:
switch (*format) {
default:
{ {
/* we only know about our samples, convert to requested format */ GstFormat src_fmt, dest_fmt;
res = gst_pad_convert (pad, gint64 src_val, dest_val;
GST_FORMAT_BYTES, speexenc->bytes_out, format, value);
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
if (!(res = gst_speexenc_convert_src (pad, src_fmt, src_val, &dest_fmt,
&dest_val)))
goto error;
gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
break; break;
} }
}
break;
default: default:
res = FALSE; res = FALSE;
break; break;
} }
error:
return res;
}
static gboolean
gst_speexenc_sink_query (GstPad * pad, GstQuery * query)
{
gboolean res = TRUE;
GstSpeexEnc *speexenc;
speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
switch (GST_QUERY_TYPE (query)) {
case GST_QUERY_CONVERT:
{
GstFormat src_fmt, dest_fmt;
gint64 src_val, dest_val;
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
if (!(res =
gst_speexenc_convert_sink (pad, src_fmt, src_val, &dest_fmt,
&dest_val)))
goto error;
gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
break;
}
default:
res = FALSE;
break;
}
error:
return res; return res;
} }
@ -446,12 +481,11 @@ gst_speexenc_init (GstSpeexEnc * speexenc)
speexenc->sinkpad = speexenc->sinkpad =
gst_pad_new_from_template (gst_speexenc_sink_template, "sink"); gst_pad_new_from_template (gst_speexenc_sink_template, "sink");
gst_element_add_pad (GST_ELEMENT (speexenc), speexenc->sinkpad); gst_element_add_pad (GST_ELEMENT (speexenc), speexenc->sinkpad);
gst_pad_set_event_function (speexenc->sinkpad, gst_speexenc_sinkevent);
gst_pad_set_chain_function (speexenc->sinkpad, gst_speexenc_chain); gst_pad_set_chain_function (speexenc->sinkpad, gst_speexenc_chain);
gst_pad_set_link_function (speexenc->sinkpad, gst_speexenc_sinkconnect); gst_pad_set_setcaps_function (speexenc->sinkpad, gst_speexenc_sink_setcaps);
gst_pad_set_convert_function (speexenc->sinkpad, gst_pad_set_query_function (speexenc->sinkpad,
GST_DEBUG_FUNCPTR (gst_speexenc_convert_sink)); GST_DEBUG_FUNCPTR (gst_speexenc_sink_query));
gst_pad_set_formats_function (speexenc->sinkpad,
GST_DEBUG_FUNCPTR (gst_speexenc_get_formats));
speexenc->srcpad = speexenc->srcpad =
gst_pad_new_from_template (gst_speexenc_src_template, "src"); gst_pad_new_from_template (gst_speexenc_src_template, "src");
@ -459,10 +493,6 @@ gst_speexenc_init (GstSpeexEnc * speexenc)
GST_DEBUG_FUNCPTR (gst_speexenc_src_query)); GST_DEBUG_FUNCPTR (gst_speexenc_src_query));
gst_pad_set_query_type_function (speexenc->srcpad, gst_pad_set_query_type_function (speexenc->srcpad,
GST_DEBUG_FUNCPTR (gst_speexenc_get_query_types)); GST_DEBUG_FUNCPTR (gst_speexenc_get_query_types));
gst_pad_set_convert_function (speexenc->srcpad,
GST_DEBUG_FUNCPTR (gst_speexenc_convert_src));
gst_pad_set_formats_function (speexenc->srcpad,
GST_DEBUG_FUNCPTR (gst_speexenc_get_formats));
gst_element_add_pad (GST_ELEMENT (speexenc), speexenc->srcpad); gst_element_add_pad (GST_ELEMENT (speexenc), speexenc->srcpad);
speexenc->channels = -1; speexenc->channels = -1;
@ -478,14 +508,9 @@ gst_speexenc_init (GstSpeexEnc * speexenc)
speexenc->nframes = DEFAULT_NFRAMES; speexenc->nframes = DEFAULT_NFRAMES;
speexenc->setup = FALSE; speexenc->setup = FALSE;
speexenc->eos = FALSE;
speexenc->header_sent = FALSE; speexenc->header_sent = FALSE;
speexenc->tags = gst_tag_list_new ();
speexenc->adapter = gst_adapter_new (); speexenc->adapter = gst_adapter_new ();
/* we're chained and we can deal with events */
GST_FLAG_SET (speexenc, GST_ELEMENT_EVENT_AWARE);
} }
@ -658,6 +683,7 @@ gst_speexenc_setup (GstSpeexEnc * speexenc)
speexenc->speex_mode = (SpeexMode *) & speex_nb_mode; speexenc->speex_mode = (SpeexMode *) & speex_nb_mode;
break; break;
case GST_SPEEXENC_MODE_AUTO: case GST_SPEEXENC_MODE_AUTO:
/* fall through */
default: default:
break; break;
} }
@ -796,77 +822,88 @@ gst_speexenc_push_buffer (GstSpeexEnc * speexenc, GstBuffer * buffer)
speexenc->bytes_out += GST_BUFFER_SIZE (buffer); speexenc->bytes_out += GST_BUFFER_SIZE (buffer);
if (GST_PAD_IS_USABLE (speexenc->srcpad)) { if (GST_PAD_IS_USABLE (speexenc->srcpad)) {
gst_pad_push (speexenc->srcpad, GST_DATA (buffer)); gst_pad_push (speexenc->srcpad, buffer);
} else { } else {
gst_buffer_unref (buffer); gst_buffer_unref (buffer);
} }
} }
static void static GstCaps *
gst_speexenc_set_header_on_caps (GstCaps * caps, GstBuffer * buf1, gst_speexenc_set_header_on_caps (GstCaps * caps, GstBuffer * buf1,
GstBuffer * buf2) GstBuffer * buf2)
{ {
caps = gst_caps_make_writable (caps);
GstStructure *structure = gst_caps_get_structure (caps, 0); GstStructure *structure = gst_caps_get_structure (caps, 0);
GValue list = { 0 }; GValue list = { 0 };
GValue value = { 0 }; GValue value = { 0 };
/* mark buffers */ /* mark buffers */
GST_BUFFER_FLAG_SET (buf1, GST_BUFFER_IN_CAPS); GST_BUFFER_FLAG_SET (buf1, GST_BUFFER_FLAG_IN_CAPS);
GST_BUFFER_FLAG_SET (buf2, GST_BUFFER_IN_CAPS); GST_BUFFER_FLAG_SET (buf2, GST_BUFFER_FLAG_IN_CAPS);
/* put buffers in a fixed list */ /* put buffers in a fixed list */
g_value_init (&list, GST_TYPE_FIXED_LIST); g_value_init (&list, GST_TYPE_ARRAY);
g_value_init (&value, GST_TYPE_BUFFER); g_value_init (&value, GST_TYPE_BUFFER);
g_value_set_boxed (&value, buf1); gst_value_set_buffer (&value, buf1);
gst_value_list_append_value (&list, &value); gst_value_list_append_value (&list, &value);
g_value_unset (&value); g_value_unset (&value);
g_value_init (&value, GST_TYPE_BUFFER); g_value_init (&value, GST_TYPE_BUFFER);
g_value_set_boxed (&value, buf2); gst_value_set_buffer (&value, buf2);
gst_value_list_append_value (&list, &value); gst_value_list_append_value (&list, &value);
gst_structure_set_value (structure, "streamheader", &list); gst_structure_set_value (structure, "streamheader", &list);
g_value_unset (&value); g_value_unset (&value);
g_value_unset (&list); g_value_unset (&list);
return caps;
} }
static void
gst_speexenc_chain (GstPad * pad, GstData * _data) static gboolean
gst_speexenc_sinkevent (GstPad * pad, GstEvent * event)
{ {
GstBuffer *buf = GST_BUFFER (_data); gboolean res = TRUE;
GstSpeexEnc *speexenc; GstSpeexEnc *speexenc;
g_return_if_fail (pad != NULL); speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (buf != NULL);
speexenc = GST_SPEEXENC (gst_pad_get_parent (pad));
if (GST_IS_EVENT (buf)) {
GstEvent *event = GST_EVENT (buf);
switch (GST_EVENT_TYPE (event)) { switch (GST_EVENT_TYPE (event)) {
case GST_EVENT_EOS: case GST_EVENT_EOS:
speexenc->eos = TRUE; speexenc->eos = TRUE;
gst_event_unref (event); res = gst_pad_event_default (pad, event);
break; break;
case GST_EVENT_TAG: case GST_EVENT_TAG:
{
GstTagList *list;
gst_event_parse_tag (event, &list);
if (speexenc->tags) { if (speexenc->tags) {
gst_tag_list_insert (speexenc->tags, gst_event_tag_get_list (event), gst_tag_list_insert (speexenc->tags, list,
gst_tag_setter_get_merge_mode (GST_TAG_SETTER (speexenc))); gst_tag_setter_get_merge_mode (GST_TAG_SETTER (speexenc)));
} else { } else {
g_assert_not_reached (); g_assert_not_reached ();
} }
gst_pad_event_default (pad, event); res = gst_pad_event_default (pad, event);
return; break;
default:
gst_pad_event_default (pad, event);
return;
} }
} else { default:
res = gst_pad_event_default (pad, event);
break;
}
return res;
}
static GstFlowReturn
gst_speexenc_chain (GstPad * pad, GstBuffer * buf)
{
GstSpeexEnc *speexenc;
speexenc = GST_SPEEXENC (GST_PAD_PARENT (pad));
if (!speexenc->setup) { if (!speexenc->setup) {
gst_buffer_unref (buf); gst_buffer_unref (buf);
GST_ELEMENT_ERROR (speexenc, CORE, NEGOTIATION, (NULL), GST_ELEMENT_ERROR (speexenc, CORE, NEGOTIATION, (NULL),
("encoder not initialized (input is not audio?)")); ("encoder not initialized (input is not audio?)"));
return; return GST_FLOW_UNEXPECTED;
} }
if (!speexenc->header_sent) { if (!speexenc->header_sent) {
@ -894,11 +931,14 @@ gst_speexenc_chain (GstPad * pad, GstData * _data)
/* mark and put on caps */ /* mark and put on caps */
caps = gst_pad_get_caps (speexenc->srcpad); caps = gst_pad_get_caps (speexenc->srcpad);
gst_speexenc_set_header_on_caps (caps, buf1, buf2); caps = gst_speexenc_set_header_on_caps (caps, buf1, buf2);
/* negotiate with these caps */ /* negotiate with these caps */
GST_DEBUG ("here are the caps: %" GST_PTR_FORMAT, caps); GST_DEBUG ("here are the caps: %" GST_PTR_FORMAT, caps);
gst_pad_try_set_caps (speexenc->srcpad, caps); gst_pad_set_caps (speexenc->srcpad, caps);
gst_buffer_set_caps (buf1, caps);
gst_buffer_set_caps (buf2, caps);
/* push out buffers */ /* push out buffers */
gst_speexenc_push_buffer (speexenc, buf1); gst_speexenc_push_buffer (speexenc, buf1);
@ -944,12 +984,13 @@ gst_speexenc_chain (GstPad * pad, GstData * _data)
speex_bits_insert_terminator (&speexenc->bits); speex_bits_insert_terminator (&speexenc->bits);
outsize = speex_bits_nbytes (&speexenc->bits); outsize = speex_bits_nbytes (&speexenc->bits);
outbuf =
gst_pad_alloc_buffer (speexenc->srcpad, GST_BUFFER_OFFSET_NONE, gst_pad_alloc_buffer (speexenc->srcpad,
outsize); GST_BUFFER_OFFSET_NONE, outsize, GST_PAD_CAPS (speexenc->srcpad),
&outbuf);
written = written =
speex_bits_write (&speexenc->bits, GST_BUFFER_DATA (outbuf), speex_bits_write (&speexenc->bits, GST_BUFFER_DATA (outbuf), outsize);
outsize);
g_assert (written == outsize); g_assert (written == outsize);
speex_bits_reset (&speexenc->bits); speex_bits_reset (&speexenc->bits);
@ -964,13 +1005,8 @@ gst_speexenc_chain (GstPad * pad, GstData * _data)
gst_speexenc_push_buffer (speexenc, outbuf); gst_speexenc_push_buffer (speexenc, outbuf);
} }
} }
}
if (speexenc->eos) { return GST_FLOW_OK;
/* clean up and exit. */
gst_pad_push (speexenc->srcpad, GST_DATA (gst_event_new (GST_EVENT_EOS)));
gst_element_set_eos (GST_ELEMENT (speexenc));
}
} }
static void static void
@ -1062,31 +1098,37 @@ static GstStateChangeReturn
gst_speexenc_change_state (GstElement * element, GstStateChange transition) gst_speexenc_change_state (GstElement * element, GstStateChange transition)
{ {
GstSpeexEnc *speexenc = GST_SPEEXENC (element); GstSpeexEnc *speexenc = GST_SPEEXENC (element);
GstStateChangeReturn res;
switch (transition) { switch (transition) {
case GST_STATE_CHANGE_NULL_TO_READY: case GST_STATE_CHANGE_NULL_TO_READY:
speexenc->tags = gst_tag_list_new ();
break; break;
case GST_STATE_CHANGE_READY_TO_PAUSED: case GST_STATE_CHANGE_READY_TO_PAUSED:
speexenc->eos = FALSE;
speexenc->frameno = 0; speexenc->frameno = 0;
speexenc->samples_in = 0; speexenc->samples_in = 0;
break; break;
case GST_STATE_CHANGE_PAUSED_TO_PLAYING: case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
/* fall through */
default:
break;
}
res = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
switch (transition) {
case GST_STATE_CHANGE_PLAYING_TO_PAUSED: case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
break; break;
case GST_STATE_CHANGE_PAUSED_TO_READY: case GST_STATE_CHANGE_PAUSED_TO_READY:
speexenc->setup = FALSE; speexenc->setup = FALSE;
speexenc->header_sent = FALSE; speexenc->header_sent = FALSE;
gst_tag_list_free (speexenc->tags);
speexenc->tags = gst_tag_list_new ();
break; break;
case GST_STATE_CHANGE_READY_TO_NULL: case GST_STATE_CHANGE_READY_TO_NULL:
gst_tag_list_free (speexenc->tags);
speexenc->tags = NULL;
default: default:
break; break;
} }
if (GST_ELEMENT_CLASS (parent_class)->change_state) return res;
return GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
return GST_STATE_CHANGE_SUCCESS;
} }