gst_plugins_good/gst/autodetect/gstautoaudiosrc.c
author hgs
Wed, 24 Mar 2010 18:04:17 -0500
changeset 16 8e837d1bf446
permissions -rw-r--r--
201009
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
16
hgs
parents:
diff changeset
     1
/* GStreamer
hgs
parents:
diff changeset
     2
 * (c) 2005 Ronald S. Bultje <rbultje@ronald.bitfreak.net>
hgs
parents:
diff changeset
     3
 * (c) 2006 Jan Schmidt <thaytan@noraisin.net>
hgs
parents:
diff changeset
     4
 * (c) 2008 Stefan Kost <ensonic@users.sf.net>
hgs
parents:
diff changeset
     5
 *
hgs
parents:
diff changeset
     6
 * This library is free software; you can redistribute it and/or
hgs
parents:
diff changeset
     7
 * modify it under the terms of the GNU Library General Public
hgs
parents:
diff changeset
     8
 * License as published by the Free Software Foundation; either
hgs
parents:
diff changeset
     9
 * version 2 of the License, or (at your option) any later version.
hgs
parents:
diff changeset
    10
 *
hgs
parents:
diff changeset
    11
 * This library is distributed in the hope that it will be useful,
hgs
parents:
diff changeset
    12
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
hgs
parents:
diff changeset
    13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
hgs
parents:
diff changeset
    14
 * Library General Public License for more details.
hgs
parents:
diff changeset
    15
 *
hgs
parents:
diff changeset
    16
 * You should have received a copy of the GNU Library General Public
hgs
parents:
diff changeset
    17
 * License along with this library; if not, write to the
hgs
parents:
diff changeset
    18
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
hgs
parents:
diff changeset
    19
 * Boston, MA 02111-1307, USA.
hgs
parents:
diff changeset
    20
 */
hgs
parents:
diff changeset
    21
hgs
parents:
diff changeset
    22
/**
hgs
parents:
diff changeset
    23
 * SECTION:element-autoaudiosrc
hgs
parents:
diff changeset
    24
 * @see_also: autovideosrc, alsasrc, osssrc
hgs
parents:
diff changeset
    25
 *
hgs
parents:
diff changeset
    26
 * autoaudiosrc is an audio source that automatically detects an appropriate
hgs
parents:
diff changeset
    27
 * audio source to use.  It does so by scanning the registry for all elements
hgs
parents:
diff changeset
    28
 * that have <quote>Source</quote> and <quote>Audio</quote> in the class field
hgs
parents:
diff changeset
    29
 * of their element information, and also have a non-zero autoplugging rank.
hgs
parents:
diff changeset
    30
 *
hgs
parents:
diff changeset
    31
 * <refsect2>
hgs
parents:
diff changeset
    32
 * <title>Example launch line</title>
hgs
parents:
diff changeset
    33
 * |[
hgs
parents:
diff changeset
    34
 * gst-launch -v -m autoaudiosrc ! audioconvert ! audioresample ! autoaudiosink
hgs
parents:
diff changeset
    35
 * ]|
hgs
parents:
diff changeset
    36
 * </refsect2>
hgs
parents:
diff changeset
    37
 */
hgs
parents:
diff changeset
    38
hgs
parents:
diff changeset
    39
#ifdef HAVE_CONFIG_H
hgs
parents:
diff changeset
    40
#include "config.h"
hgs
parents:
diff changeset
    41
#endif
hgs
parents:
diff changeset
    42
hgs
parents:
diff changeset
    43
#include <string.h>
hgs
parents:
diff changeset
    44
hgs
parents:
diff changeset
    45
#include "gstautoaudiosrc.h"
hgs
parents:
diff changeset
    46
#include "gstautodetect.h"
hgs
parents:
diff changeset
    47
hgs
parents:
diff changeset
    48
/* Properties */
hgs
parents:
diff changeset
    49
enum
hgs
parents:
diff changeset
    50
{
hgs
parents:
diff changeset
    51
  PROP_0,
hgs
parents:
diff changeset
    52
  PROP_CAPS,
hgs
parents:
diff changeset
    53
};
hgs
parents:
diff changeset
    54
hgs
parents:
diff changeset
    55
static GstStateChangeReturn
hgs
parents:
diff changeset
    56
gst_auto_audio_src_change_state (GstElement * element,
hgs
parents:
diff changeset
    57
    GstStateChange transition);
hgs
parents:
diff changeset
    58
static void gst_auto_audio_src_dispose (GstAutoAudioSrc * src);
hgs
parents:
diff changeset
    59
static void gst_auto_audio_src_clear_kid (GstAutoAudioSrc * src);
hgs
parents:
diff changeset
    60
static void gst_auto_audio_src_set_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
    61
    const GValue * value, GParamSpec * pspec);
hgs
parents:
diff changeset
    62
static void gst_auto_audio_src_get_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
    63
    GValue * value, GParamSpec * pspec);
hgs
parents:
diff changeset
    64
hgs
parents:
diff changeset
    65
GST_BOILERPLATE (GstAutoAudioSrc, gst_auto_audio_src, GstBin, GST_TYPE_BIN);
hgs
parents:
diff changeset
    66
hgs
parents:
diff changeset
    67
static const GstElementDetails gst_auto_audio_src_details =
hgs
parents:
diff changeset
    68
GST_ELEMENT_DETAILS ("Auto audio source",
hgs
parents:
diff changeset
    69
    "Source/Audio",
hgs
parents:
diff changeset
    70
    "Wrapper audio source for automatically detected audio source",
hgs
parents:
diff changeset
    71
    "Ronald Bultje <rbultje@ronald.bitfreak.net>\n"
hgs
parents:
diff changeset
    72
    "Jan Schmidt <thaytan@noraisin.net>\n"
hgs
parents:
diff changeset
    73
    "Stefan Kost <ensonic@users.sf.net>");
hgs
parents:
diff changeset
    74
hgs
parents:
diff changeset
    75
static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
hgs
parents:
diff changeset
    76
    GST_PAD_SRC,
hgs
parents:
diff changeset
    77
    GST_PAD_ALWAYS,
hgs
parents:
diff changeset
    78
    GST_STATIC_CAPS_ANY);
hgs
parents:
diff changeset
    79
hgs
parents:
diff changeset
    80
static void
hgs
parents:
diff changeset
    81
gst_auto_audio_src_base_init (gpointer klass)
hgs
parents:
diff changeset
    82
{
hgs
parents:
diff changeset
    83
  GstElementClass *eklass = GST_ELEMENT_CLASS (klass);
hgs
parents:
diff changeset
    84
hgs
parents:
diff changeset
    85
  gst_element_class_add_pad_template (eklass,
hgs
parents:
diff changeset
    86
      gst_static_pad_template_get (&src_template));
hgs
parents:
diff changeset
    87
hgs
parents:
diff changeset
    88
  gst_element_class_set_details (eklass, &gst_auto_audio_src_details);
hgs
parents:
diff changeset
    89
}
hgs
parents:
diff changeset
    90
hgs
parents:
diff changeset
    91
static void
hgs
parents:
diff changeset
    92
gst_auto_audio_src_class_init (GstAutoAudioSrcClass * klass)
hgs
parents:
diff changeset
    93
{
hgs
parents:
diff changeset
    94
  GObjectClass *gobject_class;
hgs
parents:
diff changeset
    95
  GstElementClass *eklass;
hgs
parents:
diff changeset
    96
hgs
parents:
diff changeset
    97
  gobject_class = G_OBJECT_CLASS (klass);
hgs
parents:
diff changeset
    98
  eklass = GST_ELEMENT_CLASS (klass);
hgs
parents:
diff changeset
    99
hgs
parents:
diff changeset
   100
  gobject_class->dispose =
hgs
parents:
diff changeset
   101
      (GObjectFinalizeFunc) GST_DEBUG_FUNCPTR (gst_auto_audio_src_dispose);
hgs
parents:
diff changeset
   102
  eklass->change_state = GST_DEBUG_FUNCPTR (gst_auto_audio_src_change_state);
hgs
parents:
diff changeset
   103
  gobject_class->set_property =
hgs
parents:
diff changeset
   104
      GST_DEBUG_FUNCPTR (gst_auto_audio_src_set_property);
hgs
parents:
diff changeset
   105
  gobject_class->get_property =
hgs
parents:
diff changeset
   106
      GST_DEBUG_FUNCPTR (gst_auto_audio_src_get_property);
hgs
parents:
diff changeset
   107
hgs
parents:
diff changeset
   108
  /**
hgs
parents:
diff changeset
   109
   * GstAutoAudioSrc:filter-caps
hgs
parents:
diff changeset
   110
   *
hgs
parents:
diff changeset
   111
   * This property will filter out candidate sinks that can handle the specified
hgs
parents:
diff changeset
   112
   * caps. By default only audio sinks that support raw floating point and
hgs
parents:
diff changeset
   113
   * integer audio are selected.
hgs
parents:
diff changeset
   114
   *
hgs
parents:
diff changeset
   115
   * This property can only be set before the element goes to the READY state.
hgs
parents:
diff changeset
   116
   *
hgs
parents:
diff changeset
   117
   * Since: 0.10.14
hgs
parents:
diff changeset
   118
   **/
hgs
parents:
diff changeset
   119
  g_object_class_install_property (gobject_class, PROP_CAPS,
hgs
parents:
diff changeset
   120
      g_param_spec_boxed ("filter-caps", "Filter caps",
hgs
parents:
diff changeset
   121
          "Filter sink candidates using these caps.", GST_TYPE_CAPS,
hgs
parents:
diff changeset
   122
          G_PARAM_READWRITE));
hgs
parents:
diff changeset
   123
}
hgs
parents:
diff changeset
   124
hgs
parents:
diff changeset
   125
static void
hgs
parents:
diff changeset
   126
gst_auto_audio_src_dispose (GstAutoAudioSrc * sink)
hgs
parents:
diff changeset
   127
{
hgs
parents:
diff changeset
   128
  gst_auto_audio_src_clear_kid (sink);
hgs
parents:
diff changeset
   129
hgs
parents:
diff changeset
   130
  if (sink->filter_caps)
hgs
parents:
diff changeset
   131
    gst_caps_unref (sink->filter_caps);
hgs
parents:
diff changeset
   132
  sink->filter_caps = NULL;
hgs
parents:
diff changeset
   133
hgs
parents:
diff changeset
   134
  G_OBJECT_CLASS (parent_class)->dispose ((GObject *) sink);
hgs
parents:
diff changeset
   135
}
hgs
parents:
diff changeset
   136
hgs
parents:
diff changeset
   137
static void
hgs
parents:
diff changeset
   138
gst_auto_audio_src_clear_kid (GstAutoAudioSrc * sink)
hgs
parents:
diff changeset
   139
{
hgs
parents:
diff changeset
   140
  if (sink->kid) {
hgs
parents:
diff changeset
   141
    gst_element_set_state (sink->kid, GST_STATE_NULL);
hgs
parents:
diff changeset
   142
    gst_bin_remove (GST_BIN (sink), sink->kid);
hgs
parents:
diff changeset
   143
    sink->kid = NULL;
hgs
parents:
diff changeset
   144
  }
hgs
parents:
diff changeset
   145
}
hgs
parents:
diff changeset
   146
hgs
parents:
diff changeset
   147
/*
hgs
parents:
diff changeset
   148
 * Hack to make initial linking work; ideally, this'd work even when
hgs
parents:
diff changeset
   149
 * no target has been assigned to the ghostpad yet.
hgs
parents:
diff changeset
   150
 */
hgs
parents:
diff changeset
   151
static void
hgs
parents:
diff changeset
   152
gst_auto_audio_src_reset (GstAutoAudioSrc * src)
hgs
parents:
diff changeset
   153
{
hgs
parents:
diff changeset
   154
  GstPad *targetpad;
hgs
parents:
diff changeset
   155
hgs
parents:
diff changeset
   156
  gst_auto_audio_src_clear_kid (src);
hgs
parents:
diff changeset
   157
hgs
parents:
diff changeset
   158
  /* fakesink placeholder */
hgs
parents:
diff changeset
   159
  src->kid = gst_element_factory_make ("fakesrc", "tempsrc");
hgs
parents:
diff changeset
   160
  gst_bin_add (GST_BIN (src), src->kid);
hgs
parents:
diff changeset
   161
hgs
parents:
diff changeset
   162
  /* pad */
hgs
parents:
diff changeset
   163
  targetpad = gst_element_get_static_pad (src->kid, "src");
hgs
parents:
diff changeset
   164
  gst_ghost_pad_set_target (GST_GHOST_PAD (src->pad), targetpad);
hgs
parents:
diff changeset
   165
  gst_object_unref (targetpad);
hgs
parents:
diff changeset
   166
}
hgs
parents:
diff changeset
   167
hgs
parents:
diff changeset
   168
static GstStaticCaps raw_caps =
hgs
parents:
diff changeset
   169
    GST_STATIC_CAPS ("audio/x-raw-int; audio/x-raw-float");
hgs
parents:
diff changeset
   170
hgs
parents:
diff changeset
   171
static void
hgs
parents:
diff changeset
   172
gst_auto_audio_src_init (GstAutoAudioSrc * src, GstAutoAudioSrcClass * g_class)
hgs
parents:
diff changeset
   173
{
hgs
parents:
diff changeset
   174
  src->pad = gst_ghost_pad_new_no_target ("src", GST_PAD_SRC);
hgs
parents:
diff changeset
   175
  gst_element_add_pad (GST_ELEMENT (src), src->pad);
hgs
parents:
diff changeset
   176
hgs
parents:
diff changeset
   177
  gst_auto_audio_src_reset (src);
hgs
parents:
diff changeset
   178
hgs
parents:
diff changeset
   179
  /* set the default raw audio caps */
hgs
parents:
diff changeset
   180
  src->filter_caps = gst_static_caps_get (&raw_caps);
hgs
parents:
diff changeset
   181
hgs
parents:
diff changeset
   182
  /* mark as source */
hgs
parents:
diff changeset
   183
  GST_OBJECT_FLAG_UNSET (src, GST_ELEMENT_IS_SINK);
hgs
parents:
diff changeset
   184
}
hgs
parents:
diff changeset
   185
hgs
parents:
diff changeset
   186
static gboolean
hgs
parents:
diff changeset
   187
gst_auto_audio_src_factory_filter (GstPluginFeature * feature, gpointer data)
hgs
parents:
diff changeset
   188
{
hgs
parents:
diff changeset
   189
  guint rank;
hgs
parents:
diff changeset
   190
  const gchar *klass;
hgs
parents:
diff changeset
   191
hgs
parents:
diff changeset
   192
  /* we only care about element factories */
hgs
parents:
diff changeset
   193
  if (!GST_IS_ELEMENT_FACTORY (feature))
hgs
parents:
diff changeset
   194
    return FALSE;
hgs
parents:
diff changeset
   195
hgs
parents:
diff changeset
   196
  /* audio sinks */
hgs
parents:
diff changeset
   197
  klass = gst_element_factory_get_klass (GST_ELEMENT_FACTORY (feature));
hgs
parents:
diff changeset
   198
  if (!(strstr (klass, "Source") && strstr (klass, "Audio")))
hgs
parents:
diff changeset
   199
    return FALSE;
hgs
parents:
diff changeset
   200
hgs
parents:
diff changeset
   201
  /* only select elements with autoplugging rank */
hgs
parents:
diff changeset
   202
  rank = gst_plugin_feature_get_rank (feature);
hgs
parents:
diff changeset
   203
  if (rank < GST_RANK_MARGINAL)
hgs
parents:
diff changeset
   204
    return FALSE;
hgs
parents:
diff changeset
   205
hgs
parents:
diff changeset
   206
  return TRUE;
hgs
parents:
diff changeset
   207
}
hgs
parents:
diff changeset
   208
hgs
parents:
diff changeset
   209
static gint
hgs
parents:
diff changeset
   210
gst_auto_audio_src_compare_ranks (GstPluginFeature * f1, GstPluginFeature * f2)
hgs
parents:
diff changeset
   211
{
hgs
parents:
diff changeset
   212
  gint diff;
hgs
parents:
diff changeset
   213
hgs
parents:
diff changeset
   214
  diff = gst_plugin_feature_get_rank (f2) - gst_plugin_feature_get_rank (f1);
hgs
parents:
diff changeset
   215
  if (diff != 0)
hgs
parents:
diff changeset
   216
    return diff;
hgs
parents:
diff changeset
   217
  return strcmp (gst_plugin_feature_get_name (f2),
hgs
parents:
diff changeset
   218
      gst_plugin_feature_get_name (f1));
hgs
parents:
diff changeset
   219
}
hgs
parents:
diff changeset
   220
hgs
parents:
diff changeset
   221
static GstElement *
hgs
parents:
diff changeset
   222
gst_auto_audio_src_create_element_with_pretty_name (GstAutoAudioSrc * src,
hgs
parents:
diff changeset
   223
    GstElementFactory * factory)
hgs
parents:
diff changeset
   224
{
hgs
parents:
diff changeset
   225
  GstElement *element;
hgs
parents:
diff changeset
   226
  gchar *name, *marker;
hgs
parents:
diff changeset
   227
hgs
parents:
diff changeset
   228
  marker = g_strdup (GST_PLUGIN_FEATURE (factory)->name);
hgs
parents:
diff changeset
   229
  if (g_str_has_suffix (marker, "src"))
hgs
parents:
diff changeset
   230
    marker[strlen (marker) - 4] = '\0';
hgs
parents:
diff changeset
   231
  if (g_str_has_prefix (marker, "gst"))
hgs
parents:
diff changeset
   232
    g_memmove (marker, marker + 3, strlen (marker + 3) + 1);
hgs
parents:
diff changeset
   233
  name = g_strdup_printf ("%s-actual-src-%s", GST_OBJECT_NAME (src), marker);
hgs
parents:
diff changeset
   234
  g_free (marker);
hgs
parents:
diff changeset
   235
hgs
parents:
diff changeset
   236
  element = gst_element_factory_create (factory, name);
hgs
parents:
diff changeset
   237
  g_free (name);
hgs
parents:
diff changeset
   238
hgs
parents:
diff changeset
   239
  return element;
hgs
parents:
diff changeset
   240
}
hgs
parents:
diff changeset
   241
hgs
parents:
diff changeset
   242
static GstElement *
hgs
parents:
diff changeset
   243
gst_auto_audio_src_find_best (GstAutoAudioSrc * src)
hgs
parents:
diff changeset
   244
{
hgs
parents:
diff changeset
   245
  GList *list, *item;
hgs
parents:
diff changeset
   246
  GstElement *choice = NULL;
hgs
parents:
diff changeset
   247
  GstMessage *message = NULL;
hgs
parents:
diff changeset
   248
  GSList *errors = NULL;
hgs
parents:
diff changeset
   249
  GstBus *bus = gst_bus_new ();
hgs
parents:
diff changeset
   250
  GstPad *el_pad = NULL;
hgs
parents:
diff changeset
   251
  GstCaps *el_caps = NULL, *intersect = NULL;
hgs
parents:
diff changeset
   252
  gboolean no_match = TRUE;
hgs
parents:
diff changeset
   253
hgs
parents:
diff changeset
   254
  list = gst_registry_feature_filter (gst_registry_get_default (),
hgs
parents:
diff changeset
   255
      (GstPluginFeatureFilter) gst_auto_audio_src_factory_filter, FALSE, src);
hgs
parents:
diff changeset
   256
  list = g_list_sort (list, (GCompareFunc) gst_auto_audio_src_compare_ranks);
hgs
parents:
diff changeset
   257
hgs
parents:
diff changeset
   258
  /* We don't treat sound server sources special. Our policy is that sound
hgs
parents:
diff changeset
   259
   * server sources that have a rank must not auto-spawn a daemon under any
hgs
parents:
diff changeset
   260
   * circumstances, so there's nothing for us to worry about here */
hgs
parents:
diff changeset
   261
  GST_LOG_OBJECT (src, "Trying to find usable audio devices ...");
hgs
parents:
diff changeset
   262
hgs
parents:
diff changeset
   263
  for (item = list; item != NULL; item = item->next) {
hgs
parents:
diff changeset
   264
    GstElementFactory *f = GST_ELEMENT_FACTORY (item->data);
hgs
parents:
diff changeset
   265
    GstElement *el;
hgs
parents:
diff changeset
   266
hgs
parents:
diff changeset
   267
    if ((el = gst_auto_audio_src_create_element_with_pretty_name (src, f))) {
hgs
parents:
diff changeset
   268
      GstStateChangeReturn ret;
hgs
parents:
diff changeset
   269
hgs
parents:
diff changeset
   270
      GST_DEBUG_OBJECT (src, "Testing %s", GST_PLUGIN_FEATURE (f)->name);
hgs
parents:
diff changeset
   271
hgs
parents:
diff changeset
   272
      /* If autoAudioSrc has been provided with filter caps,
hgs
parents:
diff changeset
   273
       * accept only sources that match with the filter caps */
hgs
parents:
diff changeset
   274
      if (src->filter_caps) {
hgs
parents:
diff changeset
   275
        el_pad = gst_element_get_static_pad (GST_ELEMENT (el), "src");
hgs
parents:
diff changeset
   276
        el_caps = gst_pad_get_caps (el_pad);
hgs
parents:
diff changeset
   277
        gst_object_unref (el_pad);
hgs
parents:
diff changeset
   278
        GST_DEBUG_OBJECT (src,
hgs
parents:
diff changeset
   279
            "Checking caps: %" GST_PTR_FORMAT " vs. %" GST_PTR_FORMAT,
hgs
parents:
diff changeset
   280
            src->filter_caps, el_caps);
hgs
parents:
diff changeset
   281
        intersect = gst_caps_intersect (src->filter_caps, el_caps);
hgs
parents:
diff changeset
   282
        no_match = gst_caps_is_empty (intersect);
hgs
parents:
diff changeset
   283
        gst_caps_unref (el_caps);
hgs
parents:
diff changeset
   284
        gst_caps_unref (intersect);
hgs
parents:
diff changeset
   285
hgs
parents:
diff changeset
   286
        if (no_match) {
hgs
parents:
diff changeset
   287
          GST_DEBUG_OBJECT (src, "Incompatible caps");
hgs
parents:
diff changeset
   288
          gst_object_unref (el);
hgs
parents:
diff changeset
   289
          continue;
hgs
parents:
diff changeset
   290
        } else {
hgs
parents:
diff changeset
   291
          GST_DEBUG_OBJECT (src, "Found compatible caps");
hgs
parents:
diff changeset
   292
        }
hgs
parents:
diff changeset
   293
      }
hgs
parents:
diff changeset
   294
hgs
parents:
diff changeset
   295
      gst_element_set_bus (el, bus);
hgs
parents:
diff changeset
   296
      ret = gst_element_set_state (el, GST_STATE_READY);
hgs
parents:
diff changeset
   297
      if (ret == GST_STATE_CHANGE_SUCCESS) {
hgs
parents:
diff changeset
   298
        GST_DEBUG_OBJECT (src, "This worked!");
hgs
parents:
diff changeset
   299
        choice = el;
hgs
parents:
diff changeset
   300
        break;
hgs
parents:
diff changeset
   301
      }
hgs
parents:
diff changeset
   302
hgs
parents:
diff changeset
   303
      /* collect all error messages */
hgs
parents:
diff changeset
   304
      while ((message = gst_bus_pop_filtered (bus, GST_MESSAGE_ERROR))) {
hgs
parents:
diff changeset
   305
        GST_DEBUG_OBJECT (src, "error message %" GST_PTR_FORMAT, message);
hgs
parents:
diff changeset
   306
        errors = g_slist_append (errors, message);
hgs
parents:
diff changeset
   307
      }
hgs
parents:
diff changeset
   308
hgs
parents:
diff changeset
   309
      gst_element_set_state (el, GST_STATE_NULL);
hgs
parents:
diff changeset
   310
      gst_object_unref (el);
hgs
parents:
diff changeset
   311
    }
hgs
parents:
diff changeset
   312
  }
hgs
parents:
diff changeset
   313
hgs
parents:
diff changeset
   314
  GST_DEBUG_OBJECT (src, "done trying");
hgs
parents:
diff changeset
   315
  if (!choice) {
hgs
parents:
diff changeset
   316
    if (errors) {
hgs
parents:
diff changeset
   317
      /* FIXME: we forward the first error for now; but later on it might make
hgs
parents:
diff changeset
   318
       * sense to actually analyse them */
hgs
parents:
diff changeset
   319
      gst_message_ref (GST_MESSAGE (errors->data));
hgs
parents:
diff changeset
   320
      GST_DEBUG_OBJECT (src, "reposting message %p", errors->data);
hgs
parents:
diff changeset
   321
      gst_element_post_message (GST_ELEMENT (src), GST_MESSAGE (errors->data));
hgs
parents:
diff changeset
   322
    } else {
hgs
parents:
diff changeset
   323
      /* send warning message to application and use a fakesrc */
hgs
parents:
diff changeset
   324
      GST_ELEMENT_WARNING (src, RESOURCE, NOT_FOUND, (NULL),
hgs
parents:
diff changeset
   325
          ("Failed to find a usable audio source"));
hgs
parents:
diff changeset
   326
      choice = gst_element_factory_make ("fakesrc", "fake-audio-src");
hgs
parents:
diff changeset
   327
      if (g_object_class_find_property (G_OBJECT_GET_CLASS (choice), "sync"))
hgs
parents:
diff changeset
   328
        g_object_set (choice, "sync", TRUE, NULL);
hgs
parents:
diff changeset
   329
      gst_element_set_state (choice, GST_STATE_READY);
hgs
parents:
diff changeset
   330
    }
hgs
parents:
diff changeset
   331
  }
hgs
parents:
diff changeset
   332
  gst_object_unref (bus);
hgs
parents:
diff changeset
   333
  gst_plugin_feature_list_free (list);
hgs
parents:
diff changeset
   334
  g_slist_foreach (errors, (GFunc) gst_mini_object_unref, NULL);
hgs
parents:
diff changeset
   335
  g_slist_free (errors);
hgs
parents:
diff changeset
   336
hgs
parents:
diff changeset
   337
  return choice;
hgs
parents:
diff changeset
   338
}
hgs
parents:
diff changeset
   339
hgs
parents:
diff changeset
   340
static gboolean
hgs
parents:
diff changeset
   341
gst_auto_audio_src_detect (GstAutoAudioSrc * src)
hgs
parents:
diff changeset
   342
{
hgs
parents:
diff changeset
   343
  GstElement *esrc;
hgs
parents:
diff changeset
   344
  GstPad *targetpad;
hgs
parents:
diff changeset
   345
hgs
parents:
diff changeset
   346
  gst_auto_audio_src_clear_kid (src);
hgs
parents:
diff changeset
   347
hgs
parents:
diff changeset
   348
  /* find element */
hgs
parents:
diff changeset
   349
  GST_DEBUG_OBJECT (src, "Creating new kid");
hgs
parents:
diff changeset
   350
  if (!(esrc = gst_auto_audio_src_find_best (src)))
hgs
parents:
diff changeset
   351
    goto no_src;
hgs
parents:
diff changeset
   352
hgs
parents:
diff changeset
   353
  src->kid = esrc;
hgs
parents:
diff changeset
   354
  /* Ensure the child is brought up to the right state to match the parent
hgs
parents:
diff changeset
   355
   * although it's currently always in READY and 
hgs
parents:
diff changeset
   356
   * we're always doing NULL->READY. */
hgs
parents:
diff changeset
   357
  if (GST_STATE (src->kid) < GST_STATE (src))
hgs
parents:
diff changeset
   358
    gst_element_set_state (src->kid, GST_STATE (src));
hgs
parents:
diff changeset
   359
hgs
parents:
diff changeset
   360
  gst_bin_add (GST_BIN (src), esrc);
hgs
parents:
diff changeset
   361
hgs
parents:
diff changeset
   362
  /* attach ghost pad */
hgs
parents:
diff changeset
   363
  GST_DEBUG_OBJECT (src, "Re-assigning ghostpad");
hgs
parents:
diff changeset
   364
  targetpad = gst_element_get_static_pad (src->kid, "src");
hgs
parents:
diff changeset
   365
  if (!gst_ghost_pad_set_target (GST_GHOST_PAD (src->pad), targetpad))
hgs
parents:
diff changeset
   366
    goto target_failed;
hgs
parents:
diff changeset
   367
hgs
parents:
diff changeset
   368
  gst_object_unref (targetpad);
hgs
parents:
diff changeset
   369
  GST_DEBUG_OBJECT (src, "done changing auto audio source");
hgs
parents:
diff changeset
   370
hgs
parents:
diff changeset
   371
  return TRUE;
hgs
parents:
diff changeset
   372
hgs
parents:
diff changeset
   373
  /* ERRORS */
hgs
parents:
diff changeset
   374
no_src:
hgs
parents:
diff changeset
   375
  {
hgs
parents:
diff changeset
   376
    GST_ELEMENT_ERROR (src, LIBRARY, INIT, (NULL),
hgs
parents:
diff changeset
   377
        ("Failed to find a supported audio source"));
hgs
parents:
diff changeset
   378
    return FALSE;
hgs
parents:
diff changeset
   379
  }
hgs
parents:
diff changeset
   380
target_failed:
hgs
parents:
diff changeset
   381
  {
hgs
parents:
diff changeset
   382
    GST_ELEMENT_ERROR (src, LIBRARY, INIT, (NULL),
hgs
parents:
diff changeset
   383
        ("Failed to set target pad"));
hgs
parents:
diff changeset
   384
    gst_object_unref (targetpad);
hgs
parents:
diff changeset
   385
    return FALSE;
hgs
parents:
diff changeset
   386
  }
hgs
parents:
diff changeset
   387
}
hgs
parents:
diff changeset
   388
hgs
parents:
diff changeset
   389
static GstStateChangeReturn
hgs
parents:
diff changeset
   390
gst_auto_audio_src_change_state (GstElement * element,
hgs
parents:
diff changeset
   391
    GstStateChange transition)
hgs
parents:
diff changeset
   392
{
hgs
parents:
diff changeset
   393
  GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
hgs
parents:
diff changeset
   394
  GstAutoAudioSrc *src = GST_AUTO_AUDIO_SRC (element);
hgs
parents:
diff changeset
   395
hgs
parents:
diff changeset
   396
  switch (transition) {
hgs
parents:
diff changeset
   397
    case GST_STATE_CHANGE_NULL_TO_READY:
hgs
parents:
diff changeset
   398
      if (!gst_auto_audio_src_detect (src))
hgs
parents:
diff changeset
   399
        return GST_STATE_CHANGE_FAILURE;
hgs
parents:
diff changeset
   400
      break;
hgs
parents:
diff changeset
   401
    default:
hgs
parents:
diff changeset
   402
      break;
hgs
parents:
diff changeset
   403
  }
hgs
parents:
diff changeset
   404
hgs
parents:
diff changeset
   405
  ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
hgs
parents:
diff changeset
   406
  if (ret == GST_STATE_CHANGE_FAILURE)
hgs
parents:
diff changeset
   407
    return ret;
hgs
parents:
diff changeset
   408
hgs
parents:
diff changeset
   409
  switch (transition) {
hgs
parents:
diff changeset
   410
    case GST_STATE_CHANGE_READY_TO_NULL:
hgs
parents:
diff changeset
   411
      gst_auto_audio_src_reset (src);
hgs
parents:
diff changeset
   412
      break;
hgs
parents:
diff changeset
   413
    default:
hgs
parents:
diff changeset
   414
      break;
hgs
parents:
diff changeset
   415
  }
hgs
parents:
diff changeset
   416
hgs
parents:
diff changeset
   417
  return ret;
hgs
parents:
diff changeset
   418
}
hgs
parents:
diff changeset
   419
hgs
parents:
diff changeset
   420
static void
hgs
parents:
diff changeset
   421
gst_auto_audio_src_set_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
   422
    const GValue * value, GParamSpec * pspec)
hgs
parents:
diff changeset
   423
{
hgs
parents:
diff changeset
   424
  GstAutoAudioSrc *src = GST_AUTO_AUDIO_SRC (object);
hgs
parents:
diff changeset
   425
hgs
parents:
diff changeset
   426
  switch (prop_id) {
hgs
parents:
diff changeset
   427
    case PROP_CAPS:
hgs
parents:
diff changeset
   428
      if (src->filter_caps)
hgs
parents:
diff changeset
   429
        gst_caps_unref (src->filter_caps);
hgs
parents:
diff changeset
   430
      src->filter_caps = gst_caps_copy (gst_value_get_caps (value));
hgs
parents:
diff changeset
   431
      break;
hgs
parents:
diff changeset
   432
    default:
hgs
parents:
diff changeset
   433
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
hgs
parents:
diff changeset
   434
      break;
hgs
parents:
diff changeset
   435
  }
hgs
parents:
diff changeset
   436
}
hgs
parents:
diff changeset
   437
hgs
parents:
diff changeset
   438
static void
hgs
parents:
diff changeset
   439
gst_auto_audio_src_get_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
   440
    GValue * value, GParamSpec * pspec)
hgs
parents:
diff changeset
   441
{
hgs
parents:
diff changeset
   442
  GstAutoAudioSrc *src = GST_AUTO_AUDIO_SRC (object);
hgs
parents:
diff changeset
   443
hgs
parents:
diff changeset
   444
  switch (prop_id) {
hgs
parents:
diff changeset
   445
    case PROP_CAPS:{
hgs
parents:
diff changeset
   446
      gst_value_set_caps (value, src->filter_caps);
hgs
parents:
diff changeset
   447
      break;
hgs
parents:
diff changeset
   448
    }
hgs
parents:
diff changeset
   449
    default:
hgs
parents:
diff changeset
   450
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
hgs
parents:
diff changeset
   451
      break;
hgs
parents:
diff changeset
   452
  }
hgs
parents:
diff changeset
   453
}