1 /* GStreamer
2 * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
3 * 2005 Wim Taymans <wim@fluendo.com>
4 *
5 * gstaudiosrc.c: simple audio src base class
6 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Library General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Library General Public License for more details.
16 *
17 * You should have received a copy of the GNU Library General Public
18 * License along with this library; if not, write to the
19 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
20 * Boston, MA 02110-1301, USA.
21 */
22
23 /**
24 * SECTION:gstaudiosrc
25 * @title: GstAudioSrc
26 * @short_description: Simple base class for audio sources
27 * @see_also: #GstAudioBaseSrc, #GstAudioRingBuffer, #GstAudioSrc.
28 *
29 * This is the most simple base class for audio sources that only requires
30 * subclasses to implement a set of simple functions:
31 *
32 * * `open()` :Open the device.
33 * * `prepare()` :Configure the device with the specified format.
34 * * `read()` :Read samples from the device.
35 * * `reset()` :Unblock reads and flush the device.
36 * * `delay()` :Get the number of samples in the device but not yet read.
37 * * `unprepare()` :Undo operations done by prepare.
38 * * `close()` :Close the device.
39 *
40 * All scheduling of samples and timestamps is done in this base class
41 * together with #GstAudioBaseSrc using a default implementation of a
42 * #GstAudioRingBuffer that uses threads.
43 */
44 #ifdef HAVE_CONFIG_H
45 #include "config.h"
46 #endif
47
48 #include <string.h>
49
50 #include <gst/audio/audio.h>
51 #include "gstaudiosrc.h"
52 #include "gstaudioutilsprivate.h"
53
54 GST_DEBUG_CATEGORY_STATIC (gst_audio_src_debug);
55 #define GST_CAT_DEFAULT gst_audio_src_debug
56
57 #define GST_TYPE_AUDIO_SRC_RING_BUFFER \
58 (gst_audio_src_ring_buffer_get_type())
59 #define GST_AUDIO_SRC_RING_BUFFER(obj) \
60 (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_AUDIO_SRC_RING_BUFFER,GstAudioSrcRingBuffer))
61 #define GST_AUDIO_SRC_RING_BUFFER_CLASS(klass) \
62 (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_AUDIO_SRC_RING_BUFFER,GstAudioSrcRingBufferClass))
63 #define GST_AUDIO_SRC_RING_BUFFER_GET_CLASS(obj) \
64 (G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_AUDIO_SRC_RING_BUFFER, GstAudioSrcRingBufferClass))
65 #define GST_IS_AUDIO_SRC_RING_BUFFER(obj) \
66 (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_AUDIO_SRC_RING_BUFFER))
67 #define GST_IS_AUDIO_SRC_RING_BUFFER_CLASS(klass)\
68 (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_AUDIO_SRC_RING_BUFFER))
69
70 typedef struct _GstAudioSrcRingBuffer GstAudioSrcRingBuffer;
71 typedef struct _GstAudioSrcRingBufferClass GstAudioSrcRingBufferClass;
72
73 #define GST_AUDIO_SRC_RING_BUFFER_GET_COND(buf) (&(((GstAudioSrcRingBuffer *)buf)->cond))
74 #define GST_AUDIO_SRC_RING_BUFFER_WAIT(buf) (g_cond_wait (GST_AUDIO_SRC_RING_BUFFER_GET_COND (buf), GST_OBJECT_GET_LOCK (buf)))
75 #define GST_AUDIO_SRC_RING_BUFFER_SIGNAL(buf) (g_cond_signal (GST_AUDIO_SRC_RING_BUFFER_GET_COND (buf)))
76 #define GST_AUDIO_SRC_RING_BUFFER_BROADCAST(buf)(g_cond_broadcast (GST_AUDIO_SRC_RING_BUFFER_GET_COND (buf)))
77
78 struct _GstAudioSrcRingBuffer
79 {
80 GstAudioRingBuffer object;
81
82 gboolean running;
83 gint queuedseg;
84
85 GCond cond;
86 };
87
88 struct _GstAudioSrcRingBufferClass
89 {
90 GstAudioRingBufferClass parent_class;
91 };
92
93 static void gst_audio_src_ring_buffer_class_init (GstAudioSrcRingBufferClass *
94 klass);
95 static void gst_audio_src_ring_buffer_init (GstAudioSrcRingBuffer * ringbuffer,
96 GstAudioSrcRingBufferClass * klass);
97 static void gst_audio_src_ring_buffer_dispose (GObject * object);
98 static void gst_audio_src_ring_buffer_finalize (GObject * object);
99
100 static GstAudioRingBufferClass *ring_parent_class = NULL;
101
102 static gboolean gst_audio_src_ring_buffer_open_device (GstAudioRingBuffer *
103 buf);
104 static gboolean gst_audio_src_ring_buffer_close_device (GstAudioRingBuffer *
105 buf);
106 static gboolean gst_audio_src_ring_buffer_acquire (GstAudioRingBuffer * buf,
107 GstAudioRingBufferSpec * spec);
108 static gboolean gst_audio_src_ring_buffer_release (GstAudioRingBuffer * buf);
109 static gboolean gst_audio_src_ring_buffer_start (GstAudioRingBuffer * buf);
110 static gboolean gst_audio_src_ring_buffer_stop (GstAudioRingBuffer * buf);
111 static guint gst_audio_src_ring_buffer_delay (GstAudioRingBuffer * buf);
112
113 /* ringbuffer abstract base class */
114 static GType
gst_audio_src_ring_buffer_get_type(void)115 gst_audio_src_ring_buffer_get_type (void)
116 {
117 static GType ringbuffer_type = 0;
118
119 if (!ringbuffer_type) {
120 static const GTypeInfo ringbuffer_info = {
121 sizeof (GstAudioSrcRingBufferClass),
122 NULL,
123 NULL,
124 (GClassInitFunc) gst_audio_src_ring_buffer_class_init,
125 NULL,
126 NULL,
127 sizeof (GstAudioSrcRingBuffer),
128 0,
129 (GInstanceInitFunc) gst_audio_src_ring_buffer_init,
130 NULL
131 };
132
133 ringbuffer_type =
134 g_type_register_static (GST_TYPE_AUDIO_RING_BUFFER,
135 "GstAudioSrcRingBuffer", &ringbuffer_info, 0);
136 }
137 return ringbuffer_type;
138 }
139
140 static void
gst_audio_src_ring_buffer_class_init(GstAudioSrcRingBufferClass * klass)141 gst_audio_src_ring_buffer_class_init (GstAudioSrcRingBufferClass * klass)
142 {
143 GObjectClass *gobject_class;
144 GstAudioRingBufferClass *gstringbuffer_class;
145
146 gobject_class = (GObjectClass *) klass;
147 gstringbuffer_class = (GstAudioRingBufferClass *) klass;
148
149 ring_parent_class = g_type_class_peek_parent (klass);
150
151 gobject_class->dispose = gst_audio_src_ring_buffer_dispose;
152 gobject_class->finalize = gst_audio_src_ring_buffer_finalize;
153
154 gstringbuffer_class->open_device =
155 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_open_device);
156 gstringbuffer_class->close_device =
157 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_close_device);
158 gstringbuffer_class->acquire =
159 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_acquire);
160 gstringbuffer_class->release =
161 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_release);
162 gstringbuffer_class->start =
163 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_start);
164 gstringbuffer_class->resume =
165 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_start);
166 gstringbuffer_class->stop =
167 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_stop);
168
169 gstringbuffer_class->delay =
170 GST_DEBUG_FUNCPTR (gst_audio_src_ring_buffer_delay);
171 }
172
173 typedef guint (*ReadFunc)
174 (GstAudioSrc * src, gpointer data, guint length, GstClockTime * timestamp);
175
176 /* this internal thread does nothing else but read samples from the audio device.
177 * It will read each segment in the ringbuffer and will update the play
178 * pointer.
179 * The start/stop methods control the thread.
180 */
181 static void
audioringbuffer_thread_func(GstAudioRingBuffer * buf)182 audioringbuffer_thread_func (GstAudioRingBuffer * buf)
183 {
184 GstAudioSrc *src;
185 GstAudioSrcClass *csrc;
186 GstAudioSrcRingBuffer *abuf = GST_AUDIO_SRC_RING_BUFFER (buf);
187 ReadFunc readfunc;
188 GstMessage *message;
189 GValue val = { 0 };
190
191 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
192 csrc = GST_AUDIO_SRC_GET_CLASS (src);
193
194 GST_DEBUG_OBJECT (src, "enter thread");
195
196 if ((readfunc = csrc->read) == NULL)
197 goto no_function;
198
199 if (G_UNLIKELY (!__gst_audio_set_thread_priority ()))
200 GST_WARNING_OBJECT (src, "failed to set thread priority");
201
202 message = gst_message_new_stream_status (GST_OBJECT_CAST (buf),
203 GST_STREAM_STATUS_TYPE_ENTER, GST_ELEMENT_CAST (src));
204 g_value_init (&val, GST_TYPE_G_THREAD);
205 g_value_set_boxed (&val, g_thread_self ());
206 gst_message_set_stream_status_object (message, &val);
207 g_value_unset (&val);
208 GST_DEBUG_OBJECT (src, "posting ENTER stream status");
209 gst_element_post_message (GST_ELEMENT_CAST (src), message);
210
211 while (TRUE) {
212 gint left, len;
213 guint8 *readptr;
214 gint readseg;
215 GstClockTime timestamp = GST_CLOCK_TIME_NONE;
216
217 if (gst_audio_ring_buffer_prepare_read (buf, &readseg, &readptr, &len)) {
218 gint read;
219
220 left = len;
221 do {
222 read = readfunc (src, readptr, left, ×tamp);
223 GST_LOG_OBJECT (src, "transfered %d bytes of %d to segment %d", read,
224 left, readseg);
225 if (read < 0 || read > left) {
226 GST_WARNING_OBJECT (src,
227 "error reading data %d (reason: %s), skipping segment", read,
228 g_strerror (errno));
229 break;
230 }
231 left -= read;
232 readptr += read;
233
234 } while (left > 0 && g_atomic_int_get (&abuf->running));
235
236 /* Update timestamp on buffer if required */
237 gst_audio_ring_buffer_set_timestamp (buf, readseg, timestamp);
238
239 /* we read one segment */
240 gst_audio_ring_buffer_advance (buf, 1);
241 } else {
242 GST_OBJECT_LOCK (abuf);
243 if (!abuf->running)
244 goto stop_running;
245 if (G_UNLIKELY (g_atomic_int_get (&buf->state) ==
246 GST_AUDIO_RING_BUFFER_STATE_STARTED)) {
247 GST_OBJECT_UNLOCK (abuf);
248 continue;
249 }
250 GST_DEBUG_OBJECT (src, "signal wait");
251 GST_AUDIO_SRC_RING_BUFFER_SIGNAL (buf);
252 GST_DEBUG_OBJECT (src, "wait for action");
253 GST_AUDIO_SRC_RING_BUFFER_WAIT (buf);
254 GST_DEBUG_OBJECT (src, "got signal");
255 if (!abuf->running)
256 goto stop_running;
257 GST_DEBUG_OBJECT (src, "continue running");
258 GST_OBJECT_UNLOCK (abuf);
259 }
260 }
261
262 /* Will never be reached */
263 g_assert_not_reached ();
264 return;
265
266 /* ERROR */
267 no_function:
268 {
269 GST_DEBUG ("no write function, exit thread");
270 return;
271 }
272 stop_running:
273 {
274 GST_OBJECT_UNLOCK (abuf);
275 GST_DEBUG ("stop running, exit thread");
276 message = gst_message_new_stream_status (GST_OBJECT_CAST (buf),
277 GST_STREAM_STATUS_TYPE_LEAVE, GST_ELEMENT_CAST (src));
278 g_value_init (&val, GST_TYPE_G_THREAD);
279 g_value_set_boxed (&val, g_thread_self ());
280 gst_message_set_stream_status_object (message, &val);
281 g_value_unset (&val);
282 GST_DEBUG_OBJECT (src, "posting LEAVE stream status");
283 gst_element_post_message (GST_ELEMENT_CAST (src), message);
284 return;
285 }
286 }
287
288 static void
gst_audio_src_ring_buffer_init(GstAudioSrcRingBuffer * ringbuffer,GstAudioSrcRingBufferClass * g_class)289 gst_audio_src_ring_buffer_init (GstAudioSrcRingBuffer * ringbuffer,
290 GstAudioSrcRingBufferClass * g_class)
291 {
292 ringbuffer->running = FALSE;
293 ringbuffer->queuedseg = 0;
294
295 g_cond_init (&ringbuffer->cond);
296 }
297
298 static void
gst_audio_src_ring_buffer_dispose(GObject * object)299 gst_audio_src_ring_buffer_dispose (GObject * object)
300 {
301 GstAudioSrcRingBuffer *ringbuffer = GST_AUDIO_SRC_RING_BUFFER (object);
302
303 g_cond_clear (&ringbuffer->cond);
304
305 G_OBJECT_CLASS (ring_parent_class)->dispose (object);
306 }
307
308 static void
gst_audio_src_ring_buffer_finalize(GObject * object)309 gst_audio_src_ring_buffer_finalize (GObject * object)
310 {
311 G_OBJECT_CLASS (ring_parent_class)->finalize (object);
312 }
313
314 static gboolean
gst_audio_src_ring_buffer_open_device(GstAudioRingBuffer * buf)315 gst_audio_src_ring_buffer_open_device (GstAudioRingBuffer * buf)
316 {
317 GstAudioSrc *src;
318 GstAudioSrcClass *csrc;
319 gboolean result = TRUE;
320
321 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
322 csrc = GST_AUDIO_SRC_GET_CLASS (src);
323
324 if (csrc->open)
325 result = csrc->open (src);
326
327 if (!result)
328 goto could_not_open;
329
330 return result;
331
332 could_not_open:
333 {
334 return FALSE;
335 }
336 }
337
338 static gboolean
gst_audio_src_ring_buffer_close_device(GstAudioRingBuffer * buf)339 gst_audio_src_ring_buffer_close_device (GstAudioRingBuffer * buf)
340 {
341 GstAudioSrc *src;
342 GstAudioSrcClass *csrc;
343 gboolean result = TRUE;
344
345 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
346 csrc = GST_AUDIO_SRC_GET_CLASS (src);
347
348 if (csrc->close)
349 result = csrc->close (src);
350
351 if (!result)
352 goto could_not_open;
353
354 return result;
355
356 could_not_open:
357 {
358 return FALSE;
359 }
360 }
361
362 static gboolean
gst_audio_src_ring_buffer_acquire(GstAudioRingBuffer * buf,GstAudioRingBufferSpec * spec)363 gst_audio_src_ring_buffer_acquire (GstAudioRingBuffer * buf,
364 GstAudioRingBufferSpec * spec)
365 {
366 GstAudioSrc *src;
367 GstAudioSrcClass *csrc;
368 GstAudioSrcRingBuffer *abuf;
369 gboolean result = FALSE;
370
371 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
372 csrc = GST_AUDIO_SRC_GET_CLASS (src);
373
374 if (csrc->prepare)
375 result = csrc->prepare (src, spec);
376
377 if (!result)
378 goto could_not_open;
379
380 buf->size = spec->segtotal * spec->segsize;
381 buf->memory = g_malloc (buf->size);
382 if (buf->spec.type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_RAW) {
383 gst_audio_format_fill_silence (buf->spec.info.finfo, buf->memory,
384 buf->size);
385 } else {
386 /* FIXME, non-raw formats get 0 as the empty sample */
387 memset (buf->memory, 0, buf->size);
388 }
389
390 abuf = GST_AUDIO_SRC_RING_BUFFER (buf);
391 abuf->running = TRUE;
392
393 /* FIXME: handle thread creation failure */
394 src->thread = g_thread_try_new ("audiosrc-ringbuffer",
395 (GThreadFunc) audioringbuffer_thread_func, buf, NULL);
396
397 GST_AUDIO_SRC_RING_BUFFER_WAIT (buf);
398
399 return result;
400
401 could_not_open:
402 {
403 return FALSE;
404 }
405 }
406
407 /* function is called with LOCK */
408 static gboolean
gst_audio_src_ring_buffer_release(GstAudioRingBuffer * buf)409 gst_audio_src_ring_buffer_release (GstAudioRingBuffer * buf)
410 {
411 GstAudioSrc *src;
412 GstAudioSrcClass *csrc;
413 GstAudioSrcRingBuffer *abuf;
414 gboolean result = FALSE;
415
416 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
417 csrc = GST_AUDIO_SRC_GET_CLASS (src);
418 abuf = GST_AUDIO_SRC_RING_BUFFER (buf);
419
420 abuf->running = FALSE;
421 GST_AUDIO_SRC_RING_BUFFER_SIGNAL (buf);
422 GST_OBJECT_UNLOCK (buf);
423
424 /* join the thread */
425 g_thread_join (src->thread);
426
427 GST_OBJECT_LOCK (buf);
428
429 /* free the buffer */
430 g_free (buf->memory);
431 buf->memory = NULL;
432
433 if (csrc->unprepare)
434 result = csrc->unprepare (src);
435
436 return result;
437 }
438
439 static gboolean
gst_audio_src_ring_buffer_start(GstAudioRingBuffer * buf)440 gst_audio_src_ring_buffer_start (GstAudioRingBuffer * buf)
441 {
442 GST_DEBUG ("start, sending signal");
443 GST_AUDIO_SRC_RING_BUFFER_SIGNAL (buf);
444
445 return TRUE;
446 }
447
448 static gboolean
gst_audio_src_ring_buffer_stop(GstAudioRingBuffer * buf)449 gst_audio_src_ring_buffer_stop (GstAudioRingBuffer * buf)
450 {
451 GstAudioSrc *src;
452 GstAudioSrcClass *csrc;
453
454 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
455 csrc = GST_AUDIO_SRC_GET_CLASS (src);
456
457 /* unblock any pending writes to the audio device */
458 if (csrc->reset) {
459 GST_DEBUG ("reset...");
460 csrc->reset (src);
461 GST_DEBUG ("reset done");
462 }
463 #if 0
464 GST_DEBUG ("stop, waiting...");
465 GST_AUDIO_SRC_RING_BUFFER_WAIT (buf);
466 GST_DEBUG ("stoped");
467 #endif
468
469 return TRUE;
470 }
471
472 static guint
gst_audio_src_ring_buffer_delay(GstAudioRingBuffer * buf)473 gst_audio_src_ring_buffer_delay (GstAudioRingBuffer * buf)
474 {
475 GstAudioSrc *src;
476 GstAudioSrcClass *csrc;
477 guint res = 0;
478
479 src = GST_AUDIO_SRC (GST_OBJECT_PARENT (buf));
480 csrc = GST_AUDIO_SRC_GET_CLASS (src);
481
482 if (csrc->delay)
483 res = csrc->delay (src);
484
485 return res;
486 }
487
488 /* AudioSrc signals and args */
489 enum
490 {
491 /* FILL ME */
492 LAST_SIGNAL
493 };
494
495 enum
496 {
497 ARG_0,
498 };
499
500 #define _do_init \
501 GST_DEBUG_CATEGORY_INIT (gst_audio_src_debug, "audiosrc", 0, "audiosrc element");
502 #define gst_audio_src_parent_class parent_class
503 G_DEFINE_TYPE_WITH_CODE (GstAudioSrc, gst_audio_src,
504 GST_TYPE_AUDIO_BASE_SRC, _do_init);
505
506 static GstAudioRingBuffer *gst_audio_src_create_ringbuffer (GstAudioBaseSrc *
507 src);
508
509 static void
gst_audio_src_class_init(GstAudioSrcClass * klass)510 gst_audio_src_class_init (GstAudioSrcClass * klass)
511 {
512 GstAudioBaseSrcClass *gstaudiobasesrc_class;
513
514 gstaudiobasesrc_class = (GstAudioBaseSrcClass *) klass;
515
516 gstaudiobasesrc_class->create_ringbuffer =
517 GST_DEBUG_FUNCPTR (gst_audio_src_create_ringbuffer);
518
519 g_type_class_ref (GST_TYPE_AUDIO_SRC_RING_BUFFER);
520 }
521
522 static void
gst_audio_src_init(GstAudioSrc * audiosrc)523 gst_audio_src_init (GstAudioSrc * audiosrc)
524 {
525 }
526
527 static GstAudioRingBuffer *
gst_audio_src_create_ringbuffer(GstAudioBaseSrc * src)528 gst_audio_src_create_ringbuffer (GstAudioBaseSrc * src)
529 {
530 GstAudioRingBuffer *buffer;
531
532 GST_DEBUG ("creating ringbuffer");
533 buffer = g_object_new (GST_TYPE_AUDIO_SRC_RING_BUFFER, NULL);
534 GST_DEBUG ("created ringbuffer @%p", buffer);
535
536 return buffer;
537 }
538