gstauparse.c 8.66 KB
Newer Older
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
1
/* GStreamer
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
 * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Library General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Library General Public License for more details.
 *
 * You should have received a copy of the GNU Library General Public
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

/* 2001/04/03 - Updated parseau to use caps nego
 *              Zaheer Merali <zaheer@grid9.net
 */

24
25
26
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
27
28
29
#include <stdlib.h>
#include <string.h>

30
#include <gstauparse.h>
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
31
32
33


/* elementfactory information */
34
static GstElementDetails gst_auparse_details = {
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
35
  ".au parser",
36
  "Codec/Parser",
37
  "LGPL",
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
38
39
40
41
42
43
44
  "Parse an .au file into raw audio",
  VERSION,
  "Erik Walthinsen <omega@cse.ogi.edu>",
  "(C) 1999",
};

static GstCaps*
45
au_type_find (GstBuffer *buf, gpointer private)
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
46
47
48
49
50
{
  GstCaps *new = NULL;
  gulong *head = (gulong *) GST_BUFFER_DATA (buf);

  if (*head == 0x2e736e64 || *head == 0x646e732e)
51
    new = gst_caps_new ("au_type_find", "audio/au", NULL);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
52
53
54
55
56
57

  return new;
}

/* typefactory for 'au' */
static GstTypeDefinition audefinition = {
58
  "auparse_audio/au",
59
  "audio/x-au",
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
60
  ".au",
61
  au_type_find,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
62
63
};

64
GST_PAD_TEMPLATE_FACTORY (sink_factory_templ,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
65
66
67
68
69
  "sink",
  GST_PAD_SINK,
  GST_PAD_ALWAYS,
  GST_CAPS_NEW (
    "auparse_sink",
70
    "audio/x-au",
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
71
72
73
74
75
    NULL
  )
)


76
GST_PAD_TEMPLATE_FACTORY (src_factory_templ,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
77
78
79
80
81
  "src",
  GST_PAD_SRC,
  GST_PAD_ALWAYS,
  GST_CAPS_NEW (
    "auparse_src",
82
83
    "audio/x-raw-int",
      "endianness", GST_PROPS_INT (G_BIG_ENDIAN),
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
84
85
86
87
88
89
90
91
92
93
94
95
96
97
      "signed",     GST_PROPS_LIST(
		      GST_PROPS_BOOLEAN (FALSE),
		      GST_PROPS_BOOLEAN (TRUE)
	  	    ),
      "width",      GST_PROPS_LIST(
		      GST_PROPS_INT (8),
		      GST_PROPS_INT (16)
		    ),
      "depth",      GST_PROPS_LIST(
		      GST_PROPS_INT (8),
		      GST_PROPS_INT (16)
		    ),
      "rate",       GST_PROPS_INT_RANGE (8000,48000),
      "channels",   GST_PROPS_INT_RANGE (1, 2)
98
99
100
101
102
103
  ),
  GST_CAPS_NEW (
    "auparse_src_alaw",
    "audio/x-alaw",
      "rate",       GST_PROPS_INT_RANGE (8000,48000),
      "channels",   GST_PROPS_INT_RANGE (1, 2)
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
104
105
106
  )
)

107
/* AuParse signals and args */
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
108
109
110
111
112
113
114
115
116
117
enum {
  /* FILL ME */
  LAST_SIGNAL
};

enum {
  ARG_0,
  /* FILL ME */
};

118
119
static void 	gst_auparse_class_init		(GstAuParseClass *klass);
static void 	gst_auparse_init		(GstAuParse *auparse);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
120

121
static void 	gst_auparse_chain		(GstPad *pad,GstBuffer *buf);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
122
123

static GstElementClass *parent_class = NULL;
124
/*static guint gst_auparse_signals[LAST_SIGNAL] = { 0 }; */
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
125
126

GType
127
gst_auparse_get_type (void) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
128
{
129
  static GType auparse_type = 0;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
130

131
132
133
  if (!auparse_type) {
    static const GTypeInfo auparse_info = {
      sizeof(GstAuParseClass),      NULL,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
134
      NULL,
135
      (GClassInitFunc) gst_auparse_class_init,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
136
137
      NULL,
      NULL,
138
      sizeof(GstAuParse),
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
139
      0,
140
      (GInstanceInitFunc) gst_auparse_init,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
141
    };
142
    auparse_type = g_type_register_static (GST_TYPE_ELEMENT, "GstAuParse", &auparse_info, 0);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
143
  }
144
  return auparse_type;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
145
146
147
}

static void
148
gst_auparse_class_init (GstAuParseClass *klass) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
149
150
151
152
153
154
155
156
157
{
  GstElementClass *gstelement_class;

  gstelement_class = (GstElementClass*) klass;

  parent_class = g_type_class_ref (GST_TYPE_ELEMENT);
}

static void 
158
gst_auparse_init (GstAuParse *auparse) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
159
{
160
  auparse->sinkpad = gst_pad_new_from_template (
161
		  GST_PAD_TEMPLATE_GET (sink_factory_templ), "sink");
162
163
  gst_element_add_pad (GST_ELEMENT (auparse), auparse->sinkpad);
  gst_pad_set_chain_function (auparse->sinkpad, gst_auparse_chain);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
164

165
  auparse->srcpad = gst_pad_new_from_template (
166
		  GST_PAD_TEMPLATE_GET (src_factory_templ), "src");
167
  gst_element_add_pad (GST_ELEMENT (auparse), auparse->srcpad);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
168

169
170
171
172
173
  auparse->offset = 0;
  auparse->size = 0;
  auparse->encoding = 0;
  auparse->frequency = 0;
  auparse->channels = 0;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
174
175
176
}

static void 
177
gst_auparse_chain (GstPad *pad, GstBuffer *buf) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
178
{
179
  GstAuParse *auparse;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
180
181
182
  gchar *data;
  glong size;
  GstCaps* tempcaps;
183
184
  gint law, depth;
  gboolean sign;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
185
186
187
188
189

  g_return_if_fail (pad != NULL);
  g_return_if_fail (GST_IS_PAD (pad));
  g_return_if_fail (buf != NULL);

190
  auparse = GST_AUPARSE (gst_pad_get_parent (pad));
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
191
  
192
  GST_DEBUG ("gst_auparse_chain: got buffer in '%s'",
193
          gst_element_get_name (GST_ELEMENT (auparse)));
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
194
195
196
197
198

  data = GST_BUFFER_DATA (buf);
  size = GST_BUFFER_SIZE (buf);

  /* if we haven't seen any data yet... */
199
  if (auparse->size == 0) {
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
200
    GstBuffer *newbuf;
201
    guint32 *head = (guint32 *)data;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
202
203

    /* normal format is big endian (au is a Sparc format) */
204
    if (GUINT32_FROM_BE (*head) == 0x2e736e64) {
205
      head++;
206
      auparse->le = 0;
207
      auparse->offset 		= GUINT32_FROM_BE (*head);
208
      head++;
209
      auparse->size 		= GUINT32_FROM_BE (*head);
210
      head++;
211
      auparse->encoding 	= GUINT32_FROM_BE (*head);
212
      head++;
213
      auparse->frequency 	= GUINT32_FROM_BE (*head);
214
      head++;
215
      auparse->channels 	= GUINT32_FROM_BE (*head);
216
      head++;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
217

218
219
220
    /* and of course, someone had to invent a little endian
     * version.  Used by DEC systems. */
    } else if (GUINT32_FROM_LE (*head) == 0x0064732E) {
221
      auparse->le = 1;
222
223
      head++;
      auparse->le = 0;
224
      auparse->offset 		= GUINT32_FROM_LE (*head);
225
      head++;
226
      auparse->size 		= GUINT32_FROM_LE (*head);
227
      head++;
228
      auparse->encoding 	= GUINT32_FROM_LE (*head);
229
      head++;
230
      auparse->frequency 	= GUINT32_FROM_LE (*head);
231
      head++;
232
      auparse->channels 	= GUINT32_FROM_LE (*head);
233
      head++;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
234
235
236
237
238
239
240
241

    } else {
      g_warning ("help, dunno what I'm looking at!\n");
      gst_buffer_unref(buf);
      return;
    }

    g_print ("offset %ld, size %ld, encoding %ld, frequency %ld, channels %ld\n",
242
243
             auparse->offset,auparse->size,auparse->encoding,
             auparse->frequency,auparse->channels);
244
    GST_DEBUG ("offset %ld, size %ld, encoding %ld, frequency %ld, channels %ld",
245
246
             auparse->offset,auparse->size,auparse->encoding,
             auparse->frequency,auparse->channels);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
247
    
248
    switch (auparse->encoding) {
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
249
      case 1:
250
251
252
	law = 1;
	depth = 8;
	sign = FALSE;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
253
254
	break;
      case 2:
255
256
	law = 0;
	depth = 8;
257
	sign = FALSE;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
258
259
	break;
      case 3:
260
261
262
	law = 0;
	depth = 16;
	sign = TRUE;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
263
264
265
266
267
268
	break;
      default:
	g_warning ("help!, dont know how to deal with this format yet\n");
	return;
    }

269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
    if (law) {
      tempcaps = GST_CAPS_NEW ("auparse_src",
			       "audio/x-alaw",
				 "rate",     GST_PROPS_INT (auparse->frequency),
				 "channels", GST_PROPS_INT (auparse->channels));
    } else {
      tempcaps = GST_CAPS_NEW ("auparse_src",
			       "audio/x-raw-int",
      				 "endianness", GST_PROPS_INT (G_BIG_ENDIAN),
				 "rate",       GST_PROPS_INT (auparse->frequency),
				 "channels",   GST_PROPS_INT (auparse->channels),
				 "depth",      GST_PROPS_INT (depth),
				 "width",      GST_PROPS_INT (depth),
				 "signed",     GST_PROPS_BOOLEAN (sign));
    }
284

285
    if (gst_pad_try_set_caps (auparse->srcpad, tempcaps) <= 0) {
286
      gst_buffer_unref (buf);
287
      gst_element_error (GST_ELEMENT (auparse), "could not set audio caps");
288
289
      return;
    }
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
290
291

    newbuf = gst_buffer_new ();
292
293
294
    GST_BUFFER_DATA (newbuf) = (gpointer) malloc (size-(auparse->offset));
    memcpy (GST_BUFFER_DATA (newbuf), data+24, size-(auparse->offset));
    GST_BUFFER_SIZE (newbuf) = size-(auparse->offset);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
295
296
297

    gst_buffer_unref (buf);

298
    gst_pad_push (auparse->srcpad, newbuf);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
299
300
301
    return;
  }

302
  gst_pad_push (auparse->srcpad, buf);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
303
304
305
306
307
308
309
310
311
312
}


static gboolean
plugin_init (GModule *module, GstPlugin *plugin)
{
  GstElementFactory *factory;
  GstTypeFactory *type;

  /* create the plugin structure */
313
  /* create an elementfactory for the auparse element and list it */
314
  factory = gst_element_factory_new ("auparse", GST_TYPE_AUPARSE,
315
                                    &gst_auparse_details);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
316
  g_return_val_if_fail (factory != NULL, FALSE);
317
  gst_element_factory_set_rank (factory, GST_ELEMENT_RANK_SECONDARY);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
318

319
320
  gst_element_factory_add_pad_template (factory, GST_PAD_TEMPLATE_GET (sink_factory_templ));
  gst_element_factory_add_pad_template (factory, GST_PAD_TEMPLATE_GET (src_factory_templ));
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
321

322
  type = gst_type_factory_new (&audefinition);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
323
324
325
326
327
328
329
330
331
332

  gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (factory));
  gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (type));

  return TRUE;
}

GstPluginDesc plugin_desc = {
  GST_VERSION_MAJOR,
  GST_VERSION_MINOR,
333
  "auparse",
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
334
335
336
  plugin_init
};