/* GStreamer * Copyright (C) 2005 Wim Taymans * * gstalsasrc.c: * * 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. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include "gstalsasrc.h" /* elementfactory information */ static GstElementDetails gst_alsasrc_details = GST_ELEMENT_DETAILS ("Audio Src (ALSA)", "Src/Audio", "Output to a sound card via ALSA", "Wim Taymans "); enum { PROP_0, PROP_DEVICE, }; static void gst_alsasrc_base_init (gpointer g_class); static void gst_alsasrc_class_init (GstAlsaSrcClass * klass); static void gst_alsasrc_init (GstAlsaSrc * alsasrc); static void gst_alsasrc_dispose (GObject * object); static void gst_alsasrc_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec); static void gst_alsasrc_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec); static GstCaps *gst_alsasrc_getcaps (GstBaseSrc * bsrc); static gboolean gst_alsasrc_open (GstAudioSrc * asrc, GstRingBufferSpec * spec); static gboolean gst_alsasrc_close (GstAudioSrc * asrc); static guint gst_alsasrc_read (GstAudioSrc * asrc, gpointer data, guint length); static guint gst_alsasrc_delay (GstAudioSrc * asrc); static void gst_alsasrc_reset (GstAudioSrc * asrc); /* AlsaSrc signals and args */ enum { LAST_SIGNAL }; static GstStaticPadTemplate alsasrc_src_factory = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS ("audio/x-raw-int, " "endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, " "signed = (boolean) { TRUE, FALSE }, " "width = (int) 16, " "depth = (int) 16, " "rate = (int) [ 1, MAX ], " "channels = (int) [ 1, 2 ]; " "audio/x-raw-int, " "signed = (boolean) { TRUE, FALSE }, " "width = (int) 8, " "depth = (int) 8, " "rate = (int) [ 1, MAX ], " "channels = (int) [ 1, 2 ]") ); static GstElementClass *parent_class = NULL; /* static guint gst_alsasrc_signals[LAST_SIGNAL] = { 0 }; */ GType gst_alsasrc_get_type (void) { static GType alsasrc_type = 0; if (!alsasrc_type) { static const GTypeInfo alsasrc_info = { sizeof (GstAlsaSrcClass), gst_alsasrc_base_init, NULL, (GClassInitFunc) gst_alsasrc_class_init, NULL, NULL, sizeof (GstAlsaSrc), 0, (GInstanceInitFunc) gst_alsasrc_init, }; alsasrc_type = g_type_register_static (GST_TYPE_AUDIO_SRC, "GstAlsaSrc", &alsasrc_info, 0); } return alsasrc_type; } static void gst_alsasrc_dispose (GObject * object) { G_OBJECT_CLASS (parent_class)->dispose (object); } static void gst_alsasrc_base_init (gpointer g_class) { GstElementClass *element_class = GST_ELEMENT_CLASS (g_class); gst_element_class_set_details (element_class, &gst_alsasrc_details); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&alsasrc_src_factory)); } static void gst_alsasrc_class_init (GstAlsaSrcClass * klass) { GObjectClass *gobject_class; GstElementClass *gstelement_class; GstBaseSrcClass *gstbasesrc_class; GstBaseAudioSrcClass *gstbaseaudiosrc_class; GstAudioSrcClass *gstaudiosrc_class; gobject_class = (GObjectClass *) klass; gstelement_class = (GstElementClass *) klass; gstbasesrc_class = (GstBaseSrcClass *) klass; gstbaseaudiosrc_class = (GstBaseAudioSrcClass *) klass; gstaudiosrc_class = (GstAudioSrcClass *) klass; parent_class = g_type_class_ref (GST_TYPE_BASE_AUDIO_SRC); gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_alsasrc_dispose); gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_alsasrc_get_property); gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_alsasrc_set_property); gstbasesrc_class->get_caps = GST_DEBUG_FUNCPTR (gst_alsasrc_getcaps); gstaudiosrc_class->open = GST_DEBUG_FUNCPTR (gst_alsasrc_open); gstaudiosrc_class->close = GST_DEBUG_FUNCPTR (gst_alsasrc_close); gstaudiosrc_class->read = GST_DEBUG_FUNCPTR (gst_alsasrc_read); gstaudiosrc_class->delay = GST_DEBUG_FUNCPTR (gst_alsasrc_delay); gstaudiosrc_class->reset = GST_DEBUG_FUNCPTR (gst_alsasrc_reset); g_object_class_install_property (gobject_class, PROP_DEVICE, g_param_spec_string ("device", "Device", "ALSA device, as defined in an asound configuration file", "default", G_PARAM_READWRITE)); } static void gst_alsasrc_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec) { GstAlsaSrc *src; src = GST_ALSA_SRC (object); switch (prop_id) { case PROP_DEVICE: if (src->device) g_free (src->device); src->device = g_strdup (g_value_get_string (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gst_alsasrc_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec) { GstAlsaSrc *src; src = GST_ALSA_SRC (object); switch (prop_id) { case PROP_DEVICE: g_value_set_string (value, src->device); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gst_alsasrc_init (GstAlsaSrc * alsasrc) { GST_DEBUG ("initializing alsasrc"); alsasrc->device = g_strdup ("default"); } static GstCaps * gst_alsasrc_getcaps (GstBaseSrc * bsrc) { return NULL; } #define CHECK(call, error) \ G_STMT_START { \ if ((err = call) < 0) \ goto error; \ } G_STMT_END; static int set_hwparams (GstAlsaSrc * alsa) { guint rrate; gint err, dir; snd_pcm_hw_params_t *params; snd_pcm_hw_params_alloca (¶ms); /* choose all parameters */ CHECK (snd_pcm_hw_params_any (alsa->handle, params), no_config); /* set the interleaved read/write format */ CHECK (snd_pcm_hw_params_set_access (alsa->handle, params, alsa->access), wrong_access); /* set the sample format */ CHECK (snd_pcm_hw_params_set_format (alsa->handle, params, alsa->format), no_sample_format); /* set the count of channels */ CHECK (snd_pcm_hw_params_set_channels (alsa->handle, params, alsa->channels), no_channels); /* set the stream rate */ rrate = alsa->rate; CHECK (snd_pcm_hw_params_set_rate_near (alsa->handle, params, &rrate, 0), no_rate); if (rrate != alsa->rate) goto rate_match; if (alsa->buffer_time != -1) { /* set the buffer time */ CHECK (snd_pcm_hw_params_set_buffer_time_near (alsa->handle, params, &alsa->buffer_time, &dir), buffer_time); } if (alsa->period_time != -1) { /* set the period time */ CHECK (snd_pcm_hw_params_set_period_time_near (alsa->handle, params, &alsa->period_time, &dir), period_time); } /* write the parameters to device */ CHECK (snd_pcm_hw_params (alsa->handle, params), set_hw_params); CHECK (snd_pcm_hw_params_get_buffer_size (params, &alsa->buffer_size), buffer_size); CHECK (snd_pcm_hw_params_get_period_size (params, &alsa->period_size, &dir), period_size); return 0; /* ERRORS */ no_config: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Broken configuration for recording: no configurations available: %s", snd_strerror (err)), (NULL)); return err; } wrong_access: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Access type not available for recording: %s", snd_strerror (err)), (NULL)); return err; } no_sample_format: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Sample format not available for recording: %s", snd_strerror (err)), (NULL)); return err; } no_channels: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Channels count (%i) not available for recording: %s", alsa->channels, snd_strerror (err)), (NULL)); return err; } no_rate: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Rate %iHz not available for recording: %s", alsa->rate, snd_strerror (err)), (NULL)); return err; } rate_match: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Rate doesn't match (requested %iHz, get %iHz)", alsa->rate, err), (NULL)); return -EINVAL; } buffer_time: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set buffer time %i for recording: %s", alsa->buffer_time, snd_strerror (err)), (NULL)); return err; } buffer_size: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to get buffer size for recording: %s", snd_strerror (err)), (NULL)); return err; } period_time: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set period time %i for recording: %s", alsa->period_time, snd_strerror (err)), (NULL)); return err; } period_size: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to get period size for recording: %s", snd_strerror (err)), (NULL)); return err; } set_hw_params: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set hw params for recording: %s", snd_strerror (err)), (NULL)); return err; } } static int set_swparams (GstAlsaSrc * alsa) { int err; snd_pcm_sw_params_t *params; snd_pcm_sw_params_alloca (¶ms); /* get the current swparams */ CHECK (snd_pcm_sw_params_current (alsa->handle, params), no_config); /* start the transfer when the buffer is almost full: */ /* (buffer_size / avail_min) * avail_min */ #if 0 CHECK (snd_pcm_sw_params_set_start_threshold (alsa->handle, params, (alsa->buffer_size / alsa->period_size) * alsa->period_size), start_threshold); /* allow the transfer when at least period_size samples can be processed */ CHECK (snd_pcm_sw_params_set_avail_min (alsa->handle, params, alsa->period_size), set_avail); #endif /* align all transfers to 1 sample */ CHECK (snd_pcm_sw_params_set_xfer_align (alsa->handle, params, 1), set_align); /* write the parameters to the recording device */ CHECK (snd_pcm_sw_params (alsa->handle, params), set_sw_params); return 0; /* ERRORS */ no_config: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to determine current swparams for recording: %s", snd_strerror (err)), (NULL)); return err; } #if 0 start_threshold: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set start threshold mode for recording: %s", snd_strerror (err)), (NULL)); return err; } set_avail: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set avail min for recording: %s", snd_strerror (err)), (NULL)); return err; } #endif set_align: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set transfer align for recording: %s", snd_strerror (err)), (NULL)); return err; } set_sw_params: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Unable to set sw params for recording: %s", snd_strerror (err)), (NULL)); return err; } } static gboolean alsasrc_parse_spec (GstAlsaSrc * alsa, GstRingBufferSpec * spec) { switch (spec->type) { case GST_BUFTYPE_LINEAR: alsa->format = snd_pcm_build_linear_format (spec->depth, spec->width, spec->sign ? 0 : 1, spec->bigend ? 1 : 0); break; case GST_BUFTYPE_FLOAT: switch (spec->format) { case GST_FLOAT32_LE: alsa->format = SND_PCM_FORMAT_FLOAT_LE; break; case GST_FLOAT32_BE: alsa->format = SND_PCM_FORMAT_FLOAT_BE; break; case GST_FLOAT64_LE: alsa->format = SND_PCM_FORMAT_FLOAT64_LE; break; case GST_FLOAT64_BE: alsa->format = SND_PCM_FORMAT_FLOAT64_BE; break; default: goto error; } break; case GST_BUFTYPE_A_LAW: alsa->format = SND_PCM_FORMAT_A_LAW; break; case GST_BUFTYPE_MU_LAW: alsa->format = SND_PCM_FORMAT_MU_LAW; break; default: goto error; } alsa->rate = spec->rate; alsa->channels = spec->channels; alsa->buffer_time = spec->buffer_time; alsa->period_time = spec->latency_time; alsa->access = SND_PCM_ACCESS_RW_INTERLEAVED; return TRUE; /* ERRORS */ error: { return FALSE; } } static gboolean gst_alsasrc_open (GstAudioSrc * asrc, GstRingBufferSpec * spec) { GstAlsaSrc *alsa; gint err; alsa = GST_ALSA_SRC (asrc); if (!alsasrc_parse_spec (alsa, spec)) goto spec_parse; CHECK (snd_pcm_open (&alsa->handle, alsa->device, SND_PCM_STREAM_CAPTURE, SND_PCM_NONBLOCK), open_error); CHECK (snd_pcm_nonblock (alsa->handle, 0), non_block); CHECK (set_hwparams (alsa), hw_params_failed); CHECK (set_swparams (alsa), sw_params_failed); CHECK (snd_pcm_prepare (alsa->handle), prepare_failed); alsa->bytes_per_sample = spec->bytes_per_sample; spec->segsize = alsa->period_size * spec->bytes_per_sample; spec->segtotal = alsa->buffer_size / alsa->period_size; spec->silence_sample[0] = 0; spec->silence_sample[1] = 0; spec->silence_sample[2] = 0; spec->silence_sample[3] = 0; return TRUE; /* ERRORS */ spec_parse: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Error parsing spec"), (NULL)); return FALSE; } open_error: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Recording open error: %s", snd_strerror (err)), (NULL)); return FALSE; } non_block: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Could not set device to blocking: %s", snd_strerror (err)), (NULL)); return FALSE; } hw_params_failed: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Setting of hwparams failed: %s", snd_strerror (err)), (NULL)); return FALSE; } sw_params_failed: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Setting of swparams failed: %s", snd_strerror (err)), (NULL)); return FALSE; } prepare_failed: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("Prepare failed: %s", snd_strerror (err)), (NULL)); return FALSE; } } static gboolean gst_alsasrc_close (GstAudioSrc * asrc) { GstAlsaSrc *alsa; alsa = GST_ALSA_SRC (asrc); snd_pcm_close (alsa->handle); return TRUE; } /* * Underrun and suspend recovery */ static gint xrun_recovery (snd_pcm_t * handle, gint err) { GST_DEBUG ("xrun recovery %d", err); if (err == -EPIPE) { /* under-run */ err = snd_pcm_prepare (handle); if (err < 0) GST_WARNING ("Can't recovery from underrun, prepare failed: %s", snd_strerror (err)); return 0; } else if (err == -ESTRPIPE) { while ((err = snd_pcm_resume (handle)) == -EAGAIN) g_usleep (100); /* wait until the suspend flag is released */ if (err < 0) { err = snd_pcm_prepare (handle); if (err < 0) GST_WARNING ("Can't recovery from suspend, prepare failed: %s", snd_strerror (err)); } return 0; } return err; } static guint gst_alsasrc_read (GstAudioSrc * asrc, gpointer data, guint length) { GstAlsaSrc *alsa; gint err; gint cptr; gint16 *ptr; alsa = GST_ALSA_SRC (asrc); cptr = length / alsa->bytes_per_sample; ptr = data; while (cptr > 0) { if ((err = snd_pcm_readi (alsa->handle, ptr, cptr)) < 0) { if (err == -EAGAIN) { GST_DEBUG ("Read error: %s", snd_strerror (err)); continue; } else if (xrun_recovery (alsa->handle, err) < 0) { goto read_error; } continue; } ptr += err * alsa->channels; cptr -= err; } return length - cptr; read_error: { return length; /* skip one period */ } } static guint gst_alsasrc_delay (GstAudioSrc * asrc) { GstAlsaSrc *alsa; snd_pcm_sframes_t delay; alsa = GST_ALSA_SRC (asrc); snd_pcm_delay (alsa->handle, &delay); return delay; } static void gst_alsasrc_reset (GstAudioSrc * asrc) { #if 0 GstAlsaSrc *alsa; gint err; alsa = GST_ALSA_SRC (asrc); CHECK (snd_pcm_drop (alsa->handle), drop_error); CHECK (snd_pcm_prepare (alsa->handle), prepare_error); return; /* ERRORS */ drop_error: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("alsa-reset: pcm drop error: %s", snd_strerror (err)), (NULL)); return; } prepare_error: { GST_ELEMENT_ERROR (alsa, RESOURCE, OPEN_READ, ("alsa-reset: pcm prepare error: %s", snd_strerror (err)), (NULL)); return; } #endif }