• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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, &timestamp);
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