gstauparse.c 8.71 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
{
  GstCaps *new = NULL;
  gulong *head = (gulong *) GST_BUFFER_DATA (buf);

50
51
52
  if (GST_BUFFER_SIZE (buf) < 4)
    return NULL;

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

  return new;
}

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

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


79
GST_PAD_TEMPLATE_FACTORY (src_factory_templ,
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
80
81
82
83
84
  "src",
  GST_PAD_SRC,
  GST_PAD_ALWAYS,
  GST_CAPS_NEW (
    "auparse_src",
85
86
    "audio/x-raw-int",
      "endianness", GST_PROPS_INT (G_BIG_ENDIAN),
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
87
88
89
90
91
92
93
94
95
96
97
98
99
100
      "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)
101
102
103
104
105
106
  ),
  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
107
108
109
  )
)

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

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

121
122
static void 	gst_auparse_class_init		(GstAuParseClass *klass);
static void 	gst_auparse_init		(GstAuParse *auparse);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
123

124
static void 	gst_auparse_chain		(GstPad *pad,GstBuffer *buf);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
125
126

static GstElementClass *parent_class = NULL;
127
/*static guint gst_auparse_signals[LAST_SIGNAL] = { 0 }; */
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
128
129

GType
130
gst_auparse_get_type (void) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
131
{
132
  static GType auparse_type = 0;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
133

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

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

  gstelement_class = (GstElementClass*) klass;

  parent_class = g_type_class_ref (GST_TYPE_ELEMENT);
}

static void 
161
gst_auparse_init (GstAuParse *auparse) 
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
162
{
163
  auparse->sinkpad = gst_pad_new_from_template (
164
		  GST_PAD_TEMPLATE_GET (sink_factory_templ), "sink");
165
166
  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
167

168
  auparse->srcpad = gst_pad_new_from_template (
169
		  GST_PAD_TEMPLATE_GET (src_factory_templ), "src");
170
  gst_element_add_pad (GST_ELEMENT (auparse), auparse->srcpad);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
171

172
173
174
175
176
  auparse->offset = 0;
  auparse->size = 0;
  auparse->encoding = 0;
  auparse->frequency = 0;
  auparse->channels = 0;
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
177
178
179
}

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

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

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

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

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

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

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

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

272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
    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));
    }
287

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

    newbuf = gst_buffer_new ();
295
296
297
    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
298
299
300

    gst_buffer_unref (buf);

301
    gst_pad_push (auparse->srcpad, newbuf);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
302
303
304
    return;
  }

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


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

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

322
323
  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
324

325
  type = gst_type_factory_new (&audefinition);
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
326
327
328
329
330
331
332
333
334
335

  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,
336
  "auparse",
Andy Wingo Wingo's avatar
Andy Wingo Wingo committed
337
338
339
  plugin_init
};