2 * Copyright (C) 2008 Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com>
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.
21 * SECTION:element-wasapisrc
23 * Provides audio capture from the Windows Audio Session API available with
27 * <title>Example pipelines</title>
29 * gst-launch-0.10 -v wasapisrc ! fakesink
30 * ]| Capture from the default audio device and render to fakesink.
34 #include "gstwasapisrc.h"
35 #include <gst/audio/gstaudioclock.h>
37 GST_DEBUG_CATEGORY_STATIC (gst_wasapi_src_debug);
38 #define GST_CAT_DEFAULT gst_wasapi_src_debug
40 static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
43 GST_STATIC_CAPS ("audio/x-raw-int, "
47 "channels = (int) 1, "
48 "signed = (boolean) TRUE, "
49 "endianness = (int) " G_STRINGIFY (G_BYTE_ORDER)));
51 static void gst_wasapi_src_dispose (GObject * object);
52 static void gst_wasapi_src_finalize (GObject * object);
54 static GstClock *gst_wasapi_src_provide_clock (GstElement * element);
56 static gboolean gst_wasapi_src_start (GstBaseSrc * src);
57 static gboolean gst_wasapi_src_stop (GstBaseSrc * src);
58 static gboolean gst_wasapi_src_query (GstBaseSrc * src, GstQuery * query);
60 static GstFlowReturn gst_wasapi_src_create (GstPushSrc * src, GstBuffer ** buf);
62 static GstClockTime gst_wasapi_src_get_time (GstClock * clock,
65 GST_BOILERPLATE (GstWasapiSrc, gst_wasapi_src, GstPushSrc, GST_TYPE_PUSH_SRC);
68 gst_wasapi_src_base_init (gpointer gclass)
70 GstElementClass *element_class = GST_ELEMENT_CLASS (gclass);
72 gst_element_class_add_static_pad_template (element_class, &src_template);
73 gst_element_class_set_details_simple (element_class, "WasapiSrc",
75 "Stream audio from an audio capture device through WASAPI",
76 "Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com>");
80 gst_wasapi_src_class_init (GstWasapiSrcClass * klass)
82 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
83 GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass);
84 GstBaseSrcClass *gstbasesrc_class = GST_BASE_SRC_CLASS (klass);
85 GstPushSrcClass *gstpushsrc_class = GST_PUSH_SRC_CLASS (klass);
87 gobject_class->dispose = gst_wasapi_src_dispose;
88 gobject_class->finalize = gst_wasapi_src_finalize;
90 gstelement_class->provide_clock = gst_wasapi_src_provide_clock;
92 gstbasesrc_class->start = gst_wasapi_src_start;
93 gstbasesrc_class->stop = gst_wasapi_src_stop;
94 gstbasesrc_class->query = gst_wasapi_src_query;
96 gstpushsrc_class->create = gst_wasapi_src_create;
98 GST_DEBUG_CATEGORY_INIT (gst_wasapi_src_debug, "wasapisrc",
99 0, "Windows audio session API source");
103 gst_wasapi_src_init (GstWasapiSrc * self, GstWasapiSrcClass * gclass)
105 GstBaseSrc *basesrc = GST_BASE_SRC (self);
107 gst_base_src_set_format (basesrc, GST_FORMAT_TIME);
108 gst_base_src_set_live (basesrc, TRUE);
111 self->buffer_time = 20 * GST_MSECOND;
112 self->period_time = 20 * GST_MSECOND;
113 self->latency = GST_CLOCK_TIME_NONE;
114 self->samples_per_buffer = self->rate / (GST_SECOND / self->period_time);
116 self->start_time = GST_CLOCK_TIME_NONE;
117 self->next_time = GST_CLOCK_TIME_NONE;
119 #if GST_CHECK_VERSION(0, 10, 31) || (GST_CHECK_VERSION(0, 10, 30) && GST_VERSION_NANO > 0)
120 self->clock = gst_audio_clock_new_full ("GstWasapiSrcClock",
121 gst_wasapi_src_get_time, gst_object_ref (self),
122 (GDestroyNotify) gst_object_unref);
124 self->clock = gst_audio_clock_new ("GstWasapiSrcClock",
125 gst_wasapi_src_get_time, self);
132 gst_wasapi_src_dispose (GObject * object)
134 GstWasapiSrc *self = GST_WASAPI_SRC (object);
136 if (self->clock != NULL) {
137 gst_object_unref (self->clock);
141 G_OBJECT_CLASS (parent_class)->dispose (object);
145 gst_wasapi_src_finalize (GObject * object)
147 GstWasapiSrc *self = GST_WASAPI_SRC (object);
151 G_OBJECT_CLASS (parent_class)->finalize (object);
155 gst_wasapi_src_provide_clock (GstElement * element)
157 GstWasapiSrc *self = GST_WASAPI_SRC (element);
160 GST_OBJECT_LOCK (self);
162 if (self->client_clock == NULL)
165 clock = GST_CLOCK (gst_object_ref (self->clock));
167 GST_OBJECT_UNLOCK (self);
173 GST_OBJECT_UNLOCK (self);
174 GST_DEBUG_OBJECT (self, "IAudioClock not acquired");
180 gst_wasapi_src_start (GstBaseSrc * src)
182 GstWasapiSrc *self = GST_WASAPI_SRC (src);
183 gboolean res = FALSE;
184 IAudioClient *client = NULL;
185 IAudioClock *client_clock = NULL;
186 guint64 client_clock_freq = 0;
187 IAudioCaptureClient *capture_client = NULL;
190 if (!gst_wasapi_util_get_default_device_client (GST_ELEMENT (self),
191 TRUE, self->rate, self->buffer_time, self->period_time, 0, &client,
195 hr = IAudioClient_GetService (client, &IID_IAudioClock, &client_clock);
197 GST_ERROR_OBJECT (self, "IAudioClient::GetService (IID_IAudioClock) "
202 hr = IAudioClock_GetFrequency (client_clock, &client_clock_freq);
204 GST_ERROR_OBJECT (self, "IAudioClock::GetFrequency () failed");
208 hr = IAudioClient_GetService (client, &IID_IAudioCaptureClient,
211 GST_ERROR_OBJECT (self, "IAudioClient::GetService "
212 "(IID_IAudioCaptureClient) failed");
216 hr = IAudioClient_Start (client);
218 GST_ERROR_OBJECT (self, "IAudioClient::Start failed");
222 self->client = client;
223 self->client_clock = client_clock;
224 self->client_clock_freq = client_clock_freq;
225 self->capture_client = capture_client;
231 if (capture_client != NULL)
232 IUnknown_Release (capture_client);
234 if (client_clock != NULL)
235 IUnknown_Release (client_clock);
238 IUnknown_Release (client);
245 gst_wasapi_src_stop (GstBaseSrc * src)
247 GstWasapiSrc *self = GST_WASAPI_SRC (src);
249 if (self->client != NULL) {
250 IAudioClient_Stop (self->client);
253 if (self->capture_client != NULL) {
254 IUnknown_Release (self->capture_client);
255 self->capture_client = NULL;
258 if (self->client_clock != NULL) {
259 IUnknown_Release (self->client_clock);
260 self->client_clock = NULL;
263 if (self->client != NULL) {
264 IUnknown_Release (self->client);
272 gst_wasapi_src_query (GstBaseSrc * src, GstQuery * query)
274 GstWasapiSrc *self = GST_WASAPI_SRC (src);
275 gboolean ret = FALSE;
277 GST_DEBUG_OBJECT (self, "query for %s",
278 gst_query_type_get_name (GST_QUERY_TYPE (query)));
280 switch (GST_QUERY_TYPE (query)) {
281 case GST_QUERY_LATENCY:{
282 GstClockTime min_latency, max_latency;
284 min_latency = self->latency + self->period_time;
285 max_latency = min_latency;
287 GST_DEBUG_OBJECT (self, "reporting latency of min %" GST_TIME_FORMAT
288 " max %" GST_TIME_FORMAT,
289 GST_TIME_ARGS (min_latency), GST_TIME_ARGS (max_latency));
291 gst_query_set_latency (query, TRUE, min_latency, max_latency);
297 ret = GST_BASE_SRC_CLASS (parent_class)->query (src, query);
305 gst_wasapi_src_create (GstPushSrc * src, GstBuffer ** buf)
307 GstWasapiSrc *self = GST_WASAPI_SRC (src);
308 GstFlowReturn ret = GST_FLOW_OK;
310 GstClockTime timestamp, duration = self->period_time;
312 gint16 *samples = NULL;
313 guint32 nsamples_read = 0, nsamples;
317 GST_OBJECT_LOCK (self);
318 clock = GST_ELEMENT_CLOCK (self);
320 gst_object_ref (clock);
321 GST_OBJECT_UNLOCK (self);
323 if (clock != NULL && GST_CLOCK_TIME_IS_VALID (self->next_time)) {
326 id = gst_clock_new_single_shot_id (clock, self->next_time);
327 gst_clock_id_wait (id, NULL);
328 gst_clock_id_unref (id);
332 hr = IAudioCaptureClient_GetBuffer (self->capture_client,
333 (BYTE **) & samples, &nsamples_read, &flags, &devpos, NULL);
335 while (hr == AUDCLNT_S_BUFFER_EMPTY);
338 GST_ERROR_OBJECT (self, "IAudioCaptureClient::GetBuffer () failed: %s",
339 gst_wasapi_util_hresult_to_string (hr));
340 ret = GST_FLOW_ERROR;
345 GST_WARNING_OBJECT (self, "devpos %" G_GUINT64_FORMAT ": flags=0x%08x",
349 /* FIXME: Why do we get 1024 sometimes and not a multiple of
350 * samples_per_buffer? Shouldn't WASAPI provide a DISCONT
351 * flag if we read too slow?
353 nsamples = nsamples_read;
354 g_assert (nsamples >= self->samples_per_buffer);
355 if (nsamples > self->samples_per_buffer) {
356 GST_WARNING_OBJECT (self,
357 "devpos %" G_GUINT64_FORMAT ": got %d samples, expected %d, clipping!",
358 devpos, nsamples, self->samples_per_buffer);
360 nsamples = self->samples_per_buffer;
363 if (clock == NULL || clock == self->clock) {
365 gst_util_uint64_scale (devpos, GST_SECOND, self->client_clock_freq);
367 GstClockTime base_time;
369 timestamp = gst_clock_get_time (clock);
371 base_time = GST_ELEMENT_CAST (self)->base_time;
372 if (timestamp > base_time)
373 timestamp -= base_time;
377 if (timestamp > duration)
378 timestamp -= duration;
383 ret = gst_pad_alloc_buffer_and_set_caps (GST_BASE_SRC_PAD (self),
385 nsamples * sizeof (gint16), GST_PAD_CAPS (GST_BASE_SRC_PAD (self)), buf);
387 if (ret == GST_FLOW_OK) {
391 GST_BUFFER_OFFSET_END (*buf) = devpos + self->samples_per_buffer;
392 GST_BUFFER_TIMESTAMP (*buf) = timestamp;
393 GST_BUFFER_DURATION (*buf) = duration;
395 dst = (gint16 *) GST_BUFFER_DATA (*buf);
396 for (i = 0; i < nsamples; i++) {
404 hr = IAudioCaptureClient_ReleaseBuffer (self->capture_client, nsamples_read);
406 GST_ERROR_OBJECT (self, "IAudioCaptureClient::ReleaseBuffer () failed: %s",
407 gst_wasapi_util_hresult_to_string (hr));
408 ret = GST_FLOW_ERROR;
414 gst_object_unref (clock);
420 gst_wasapi_src_get_time (GstClock * clock, gpointer user_data)
422 GstWasapiSrc *self = GST_WASAPI_SRC (user_data);
427 if (G_UNLIKELY (self->client_clock == NULL))
428 return GST_CLOCK_TIME_NONE;
430 hr = IAudioClock_GetPosition (self->client_clock, &devpos, NULL);
431 if (G_UNLIKELY (hr != S_OK))
432 return GST_CLOCK_TIME_NONE;
434 result = gst_util_uint64_scale_int (devpos, GST_SECOND,
435 self->client_clock_freq);
438 GST_DEBUG_OBJECT (self, "devpos = %" G_GUINT64_FORMAT
439 " frequency = %" G_GUINT64_FORMAT
440 " result = %" G_GUINT64_FORMAT " ms",
441 devpos, self->client_clock_freq, GST_TIME_AS_MSECONDS (result));