summaryrefslogtreecommitdiffstats
path: root/ext/gsm/gstgsmenc.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/gsm/gstgsmenc.c')
-rw-r--r--ext/gsm/gstgsmenc.c165
1 files changed, 99 insertions, 66 deletions
diff --git a/ext/gsm/gstgsmenc.c b/ext/gsm/gstgsmenc.c
index 6a8f24ee..43d7f2db 100644
--- a/ext/gsm/gstgsmenc.c
+++ b/ext/gsm/gstgsmenc.c
@@ -56,6 +56,9 @@ static GstFlowReturn gst_gsmenc_chain (GstPad * pad, GstBuffer * buf);
static GstElementClass *parent_class = NULL;
static guint gst_gsmenc_signals[LAST_SIGNAL] = { 0 };
+static gboolean gst_gsmenc_sink_event (GstPad * pad, GstEvent * event);
+
+
GType
gst_gsmenc_get_type (void)
{
@@ -137,6 +140,7 @@ gst_gsmenc_init (GstGSMEnc * gsmenc)
(&gsmenc_sink_template), "sink");
gst_element_add_pad (GST_ELEMENT (gsmenc), gsmenc->sinkpad);
gst_pad_set_chain_function (gsmenc->sinkpad, gst_gsmenc_chain);
+ gst_pad_set_event_function (gsmenc->sinkpad, gst_gsmenc_sink_event);
gsmenc->srcpad =
gst_pad_new_from_template (gst_static_pad_template_get
@@ -146,98 +150,127 @@ gst_gsmenc_init (GstGSMEnc * gsmenc)
gsmenc->state = gsm_create ();
gsmenc->bufsize = 0;
gsmenc->next_ts = 0;
+ gsmenc->firstBuf = TRUE;
}
-static GstFlowReturn
-gst_gsmenc_chain (GstPad * pad, GstBuffer * buf)
+static gboolean
+gst_gsmenc_sink_event (GstPad * pad, GstEvent * event)
{
+
GstGSMEnc *gsmenc;
gsmenc = GST_GSMENC (GST_OBJECT_PARENT (pad));
-
- if (GST_IS_EVENT (buf)) {
- GstEvent *event = GST_EVENT (buf);
-
- switch (GST_EVENT_TYPE (event)) {
- case GST_EVENT_EOS:{
- gst_pad_push_event (gsmenc->srcpad, gst_event_new_eos ());
- gst_pad_push (gsmenc->srcpad, buf);
- break;
- }
- case GST_EVENT_NEWSEGMENT:{
- /* drop the discontinuity */
- break;
- }
- default:{
- gst_pad_event_default (pad, event);
- break;
- }
+ gboolean ret = TRUE;
+
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_EOS:
+ {
+ GST_STREAM_LOCK (pad);
+ ret = gst_pad_push_event (gsmenc->srcpad, event);
+ GST_STREAM_UNLOCK (pad);
+ break;
}
- return GST_FLOW_OK;
- } else if (GST_IS_BUFFER (buf)) {
- gsm_signal *data;
- guint size;
- GstCaps *tempcaps = NULL;
+ case GST_EVENT_NEWSEGMENT:
+ {
+ /* drop the discontinuity */
+ gst_event_unref (event);
+ break;
+ }
+ default:
+ {
+ gst_pad_event_default (pad, event);
+ break;
+ }
+ }
+ return ret;
+
+}
+
+static GstFlowReturn
+gst_gsmenc_chain (GstPad * pad, GstBuffer * buf)
+{
+ GstGSMEnc *gsmenc;
+ gsm_signal *data;
+ guint size;
+ GstFlowReturn ret = GST_FLOW_OK;
- data = (gsm_signal *) GST_BUFFER_DATA (buf);
- size = GST_BUFFER_SIZE (buf) / sizeof (gsm_signal);
+ gsmenc = GST_GSMENC (gst_pad_get_parent (pad));
- if (gsmenc->bufsize && (gsmenc->bufsize + size >= 160)) {
- GstBuffer *outbuf;
+ data = (gsm_signal *) GST_BUFFER_DATA (buf);
+ size = GST_BUFFER_SIZE (buf) / sizeof (gsm_signal);
- memcpy (gsmenc->buffer + gsmenc->bufsize, data,
- (160 - gsmenc->bufsize) * sizeof (gsm_signal));
+ if (gsmenc->bufsize && (gsmenc->bufsize + size >= 160)) {
+ GstBuffer *outbuf;
- outbuf = gst_buffer_new_and_alloc (33 * sizeof (gsm_byte));
- GST_BUFFER_TIMESTAMP (outbuf) = gsmenc->next_ts;
- GST_BUFFER_DURATION (outbuf) = 20 * GST_MSECOND;
- gsmenc->next_ts += 20 * GST_MSECOND;
+ memcpy (gsmenc->buffer + gsmenc->bufsize, data,
+ (160 - gsmenc->bufsize) * sizeof (gsm_signal));
- gsm_encode (gsmenc->state, gsmenc->buffer,
- (gsm_byte *) GST_BUFFER_DATA (outbuf));
+ outbuf = gst_buffer_new_and_alloc (33 * sizeof (gsm_byte));
+ GST_BUFFER_TIMESTAMP (outbuf) = gsmenc->next_ts;
+ GST_BUFFER_DURATION (outbuf) = 20 * GST_MSECOND;
+ gsmenc->next_ts += 20 * GST_MSECOND;
- tempcaps = gst_caps_new_simple ("audio/x-gsm",
- "rate", G_TYPE_INT, 8000, "channels", G_TYPE_INT, 1, NULL);
+ gsm_encode (gsmenc->state, gsmenc->buffer,
+ (gsm_byte *) GST_BUFFER_DATA (outbuf));
- gst_buffer_set_caps (outbuf, tempcaps);
- gst_pad_push (gsmenc->srcpad, outbuf);
+ gst_buffer_set_caps (outbuf, GST_PAD_CAPS (gsmenc->srcpad));
- size -= (160 - gsmenc->bufsize);
- data += (160 - gsmenc->bufsize);
- gsmenc->bufsize = 0;
+ if (gsmenc->firstBuf) {
+ gst_pad_push_event (gsmenc->srcpad,
+ gst_event_new_newsegment (FALSE, 1.0, GST_FORMAT_DEFAULT,
+ 0, GST_CLOCK_TIME_NONE, 0));
+ gsmenc->firstBuf = FALSE;
}
- while (size >= 160) {
- GstBuffer *outbuf;
+ ret = gst_pad_push (gsmenc->srcpad, outbuf);
+
+ if (ret != GST_FLOW_OK) {
+ gst_buffer_unref (outbuf);
+ goto error;
+ }
- outbuf = gst_buffer_new_and_alloc (33 * sizeof (gsm_byte));
- GST_BUFFER_TIMESTAMP (outbuf) = gsmenc->next_ts;
- GST_BUFFER_DURATION (outbuf) = 20 * GST_MSECOND;
- gsmenc->next_ts += 20 * GST_MSECOND;
+ size -= (160 - gsmenc->bufsize);
+ data += (160 - gsmenc->bufsize);
+ gsmenc->bufsize = 0;
- gsm_encode (gsmenc->state, data, (gsm_byte *) GST_BUFFER_DATA (outbuf));
+ }
- /* I was wondering that gst_buffer_set_caps (outbuf, GST_PAD_CAPS (gsmenc->srcpad));
- * could work, but it doens't work */
+ while (size >= 160) {
+ GstBuffer *outbuf;
- tempcaps = gst_caps_new_simple ("audio/x-gsm",
- "rate", G_TYPE_INT, 8000, "channels", G_TYPE_INT, 1, NULL);
+ outbuf = gst_buffer_new_and_alloc (33 * sizeof (gsm_byte));
+ GST_BUFFER_TIMESTAMP (outbuf) = gsmenc->next_ts;
+ GST_BUFFER_DURATION (outbuf) = 20 * GST_MSECOND;
+ gsmenc->next_ts += 20 * GST_MSECOND;
- gst_buffer_set_caps (outbuf, tempcaps);
- gst_pad_push (gsmenc->srcpad, outbuf);
+ gsm_encode (gsmenc->state, data, (gsm_byte *) GST_BUFFER_DATA (outbuf));
- size -= 160;
- data += 160;
+ gst_buffer_set_caps (outbuf, GST_PAD_CAPS (gsmenc->srcpad));
+ if (gsmenc->firstBuf) {
+ gst_pad_push_event (gsmenc->srcpad,
+ gst_event_new_newsegment (FALSE, 1.0, GST_FORMAT_DEFAULT,
+ 0, GST_CLOCK_TIME_NONE, 0));
+ gsmenc->firstBuf = FALSE;
}
+ ret = gst_pad_push (gsmenc->srcpad, outbuf);
- if (size) {
- memcpy (gsmenc->buffer + gsmenc->bufsize, data,
- size * sizeof (gsm_signal));
- gsmenc->bufsize += size;
+ if (ret != GST_FLOW_OK) {
+ gst_buffer_unref (outbuf);
+ goto error;
}
- /*gst_buffer_unref (buf); */
- return GST_FLOW_OK;
+ size -= 160;
+ data += 160;
+ }
+
+ if (size) {
+ memcpy (gsmenc->buffer + gsmenc->bufsize, data, size * sizeof (gsm_signal));
+ gsmenc->bufsize += size;
}
- return GST_FLOW_OK;
+
+error:
+
+ gst_object_unref (gsmenc);
+ return ret;
+
}