2 * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
20 * SECTION:element-mulawdec
22 * This element decodes mulaw audio. Mulaw coding is also known as G.711.
30 #include "mulaw-decode.h"
31 #include "mulaw-conversion.h"
33 #if G_BYTE_ORDER == G_LITTLE_ENDIAN
34 #define INT_FORMAT "S16LE"
36 #define INT_FORMAT "S16BE"
39 extern GstStaticPadTemplate mulaw_dec_src_factory;
40 extern GstStaticPadTemplate mulaw_dec_sink_factory;
42 /* Stereo signals and args */
54 static GstStateChangeReturn
55 gst_mulawdec_change_state (GstElement * element, GstStateChange transition);
57 static gboolean gst_mulawdec_event (GstPad * pad, GstEvent * event);
58 static GstFlowReturn gst_mulawdec_chain (GstPad * pad, GstBuffer * buffer);
60 #define gst_mulawdec_parent_class parent_class
61 G_DEFINE_TYPE (GstMuLawDec, gst_mulawdec, GST_TYPE_ELEMENT);
64 mulawdec_setcaps (GstMuLawDec * mulawdec, GstCaps * caps)
66 GstStructure *structure;
71 structure = gst_caps_get_structure (caps, 0);
72 ret = gst_structure_get_int (structure, "rate", &rate);
73 ret = ret && gst_structure_get_int (structure, "channels", &channels);
77 outcaps = gst_caps_new_simple ("audio/x-raw",
78 "format", G_TYPE_STRING, INT_FORMAT,
79 "rate", G_TYPE_INT, rate, "channels", G_TYPE_INT, channels, NULL);
80 ret = gst_pad_set_caps (mulawdec->srcpad, outcaps);
81 gst_caps_unref (outcaps);
84 GST_DEBUG_OBJECT (mulawdec, "rate=%d, channels=%d", rate, channels);
85 mulawdec->rate = rate;
86 mulawdec->channels = channels;
92 mulawdec_getcaps (GstPad * pad, GstCaps * filter)
94 GstMuLawDec *mulawdec;
96 GstCaps *othercaps, *result;
101 mulawdec = GST_MULAWDEC (GST_PAD_PARENT (pad));
103 /* figure out the name of the caps we are going to return */
104 if (pad == mulawdec->srcpad) {
105 name = "audio/x-raw";
106 otherpad = mulawdec->sinkpad;
108 name = "audio/x-mulaw";
109 otherpad = mulawdec->srcpad;
111 /* get caps from the peer, this can return NULL when there is no peer */
112 othercaps = gst_pad_peer_query_caps (otherpad, filter);
114 /* get the template caps to make sure we return something acceptable */
115 templ = gst_pad_get_pad_template_caps (pad);
118 /* there was a peer */
119 othercaps = gst_caps_make_writable (othercaps);
121 /* go through the caps and remove the fields we don't want */
122 for (i = 0; i < gst_caps_get_size (othercaps); i++) {
123 GstStructure *structure;
125 structure = gst_caps_get_structure (othercaps, i);
127 /* adjust the name */
128 gst_structure_set_name (structure, name);
130 if (pad == mulawdec->sinkpad) {
131 /* remove the fields we don't want */
132 gst_structure_remove_fields (structure, "format", NULL);
134 /* add fixed fields */
135 gst_structure_set (structure, "format", G_TYPE_STRING, INT_FORMAT,
139 /* filter against the allowed caps of the pad to return our result */
140 result = gst_caps_intersect (othercaps, templ);
141 gst_caps_unref (othercaps);
143 /* there was no peer, return the template caps */
144 result = gst_caps_copy (templ);
150 gst_mulawdec_query (GstPad * pad, GstQuery * query)
154 switch (GST_QUERY_TYPE (query)) {
157 GstCaps *filter, *caps;
159 gst_query_parse_caps (query, &filter);
160 caps = mulawdec_getcaps (pad, filter);
161 gst_query_set_caps_result (query, caps);
162 gst_caps_unref (caps);
168 res = gst_pad_query_default (pad, query);
175 gst_mulawdec_class_init (GstMuLawDecClass * klass)
177 GstElementClass *element_class = (GstElementClass *) klass;
179 gst_element_class_add_pad_template (element_class,
180 gst_static_pad_template_get (&mulaw_dec_src_factory));
181 gst_element_class_add_pad_template (element_class,
182 gst_static_pad_template_get (&mulaw_dec_sink_factory));
184 gst_element_class_set_details_simple (element_class, "Mu Law audio decoder",
185 "Codec/Decoder/Audio",
186 "Convert 8bit mu law to 16bit PCM",
187 "Zaheer Abbas Merali <zaheerabbas at merali dot org>");
189 element_class->change_state = GST_DEBUG_FUNCPTR (gst_mulawdec_change_state);
193 gst_mulawdec_init (GstMuLawDec * mulawdec)
196 gst_pad_new_from_static_template (&mulaw_dec_sink_factory, "sink");
197 gst_pad_set_query_function (mulawdec->sinkpad, gst_mulawdec_query);
198 gst_pad_set_event_function (mulawdec->sinkpad, gst_mulawdec_event);
199 gst_pad_set_chain_function (mulawdec->sinkpad, gst_mulawdec_chain);
200 gst_element_add_pad (GST_ELEMENT (mulawdec), mulawdec->sinkpad);
203 gst_pad_new_from_static_template (&mulaw_dec_src_factory, "src");
204 gst_pad_set_query_function (mulawdec->srcpad, gst_mulawdec_query);
205 gst_element_add_pad (GST_ELEMENT (mulawdec), mulawdec->srcpad);
209 gst_mulawdec_event (GstPad * pad, GstEvent * event)
211 GstMuLawDec *mulawdec;
214 mulawdec = GST_MULAWDEC (GST_PAD_PARENT (pad));
216 switch (GST_EVENT_TYPE (event)) {
221 gst_event_parse_caps (event, &caps);
222 mulawdec_setcaps (mulawdec, caps);
223 gst_event_unref (event);
229 res = gst_pad_event_default (pad, event);
236 gst_mulawdec_chain (GstPad * pad, GstBuffer * buffer)
238 GstMuLawDec *mulawdec;
241 gsize mulaw_size, linear_size;
245 mulawdec = GST_MULAWDEC (GST_PAD_PARENT (pad));
247 if (G_UNLIKELY (mulawdec->rate == 0))
250 mulaw_data = gst_buffer_map (buffer, &mulaw_size, NULL, GST_MAP_READ);
252 linear_size = mulaw_size * 2;
254 outbuf = gst_buffer_new_allocate (NULL, linear_size, 0);
255 linear_data = gst_buffer_map (outbuf, NULL, NULL, GST_MAP_WRITE);
257 /* copy discont flag */
258 if (GST_BUFFER_FLAG_IS_SET (buffer, GST_BUFFER_FLAG_DISCONT))
259 GST_BUFFER_FLAG_SET (outbuf, GST_BUFFER_FLAG_DISCONT);
261 GST_BUFFER_TIMESTAMP (outbuf) = GST_BUFFER_TIMESTAMP (buffer);
262 if (GST_BUFFER_DURATION (outbuf) == GST_CLOCK_TIME_NONE)
263 GST_BUFFER_DURATION (outbuf) = gst_util_uint64_scale_int (GST_SECOND,
264 linear_size, 2 * mulawdec->rate * mulawdec->channels);
266 GST_BUFFER_DURATION (outbuf) = GST_BUFFER_DURATION (buffer);
268 mulaw_decode (mulaw_data, linear_data, mulaw_size);
270 gst_buffer_unmap (outbuf, linear_data, -1);
271 gst_buffer_unmap (buffer, mulaw_data, -1);
272 gst_buffer_unref (buffer);
274 ret = gst_pad_push (mulawdec->srcpad, outbuf);
281 GST_WARNING_OBJECT (mulawdec, "no input format set: not-negotiated");
282 gst_buffer_unref (buffer);
283 return GST_FLOW_NOT_NEGOTIATED;
287 static GstStateChangeReturn
288 gst_mulawdec_change_state (GstElement * element, GstStateChange transition)
290 GstStateChangeReturn ret;
291 GstMuLawDec *dec = GST_MULAWDEC (element);
293 switch (transition) {
298 ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
299 if (ret != GST_STATE_CHANGE_SUCCESS)
302 switch (transition) {
303 case GST_STATE_CHANGE_PAUSED_TO_READY: