gst_plugins_good/gst/autodetect/gstautovideosrc.c
author hgs
Fri, 09 Jul 2010 16:26:45 -0500
changeset 24 bc39b352897e
parent 16 8e837d1bf446
permissions -rw-r--r--
201027
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-autovideosrc
hgs
parents:
diff changeset
    24
 * @see_also: autoaudiosrc, v4l2src, v4lsrc
hgs
parents:
diff changeset
    25
 *
hgs
parents:
diff changeset
    26
 * autovideosrc is a video src that automatically detects an appropriate
hgs
parents:
diff changeset
    27
 * video 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>Video</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 autovideosrc ! xvimagesink
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 "gstautovideosrc.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_video_src_change_state (GstElement * element,
hgs
parents:
diff changeset
    57
    GstStateChange transition);
hgs
parents:
diff changeset
    58
static void gst_auto_video_src_dispose (GstAutoVideoSrc * src);
hgs
parents:
diff changeset
    59
static void gst_auto_video_src_clear_kid (GstAutoVideoSrc * src);
hgs
parents:
diff changeset
    60
hgs
parents:
diff changeset
    61
static void gst_auto_video_src_set_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
    62
    const GValue * value, GParamSpec * pspec);
hgs
parents:
diff changeset
    63
static void gst_auto_video_src_get_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
    64
    GValue * value, GParamSpec * pspec);
hgs
parents:
diff changeset
    65
hgs
parents:
diff changeset
    66
GST_BOILERPLATE (GstAutoVideoSrc, gst_auto_video_src, GstBin, GST_TYPE_BIN);
hgs
parents:
diff changeset
    67
hgs
parents:
diff changeset
    68
static const GstElementDetails gst_auto_video_src_details =
hgs
parents:
diff changeset
    69
GST_ELEMENT_DETAILS ("Auto video source",
hgs
parents:
diff changeset
    70
    "Source/Video",
hgs
parents:
diff changeset
    71
    "Wrapper video source for automatically detected video source",
hgs
parents:
diff changeset
    72
    "Ronald Bultje <rbultje@ronald.bitfreak.net>\n"
hgs
parents:
diff changeset
    73
    "Jan Schmidt <thaytan@noraisin.net>\n"
hgs
parents:
diff changeset
    74
    "Stefan Kost <ensonic@users.sf.net>");
hgs
parents:
diff changeset
    75
hgs
parents:
diff changeset
    76
static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
hgs
parents:
diff changeset
    77
    GST_PAD_SRC,
hgs
parents:
diff changeset
    78
    GST_PAD_ALWAYS,
hgs
parents:
diff changeset
    79
    GST_STATIC_CAPS_ANY);
hgs
parents:
diff changeset
    80
hgs
parents:
diff changeset
    81
static void
hgs
parents:
diff changeset
    82
gst_auto_video_src_base_init (gpointer klass)
hgs
parents:
diff changeset
    83
{
hgs
parents:
diff changeset
    84
  GstElementClass *eklass = GST_ELEMENT_CLASS (klass);
hgs
parents:
diff changeset
    85
hgs
parents:
diff changeset
    86
  gst_element_class_add_pad_template (eklass,
hgs
parents:
diff changeset
    87
      gst_static_pad_template_get (&src_template));
hgs
parents:
diff changeset
    88
  gst_element_class_set_details (eklass, &gst_auto_video_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_video_src_class_init (GstAutoVideoSrcClass * klass)
hgs
parents:
diff changeset
    93
{
hgs
parents:
diff changeset
    94
  GObjectClass *gobject_class;
hgs
parents:
diff changeset
    95
  GstElementClass *eklass = GST_ELEMENT_CLASS (klass);
hgs
parents:
diff changeset
    96
hgs
parents:
diff changeset
    97
  gobject_class = G_OBJECT_CLASS (klass);
hgs
parents:
diff changeset
    98
  gobject_class->dispose =
hgs
parents:
diff changeset
    99
      (GObjectFinalizeFunc) GST_DEBUG_FUNCPTR (gst_auto_video_src_dispose);
hgs
parents:
diff changeset
   100
  eklass->change_state = GST_DEBUG_FUNCPTR (gst_auto_video_src_change_state);
hgs
parents:
diff changeset
   101
  gobject_class->set_property =
hgs
parents:
diff changeset
   102
      GST_DEBUG_FUNCPTR (gst_auto_video_src_set_property);
hgs
parents:
diff changeset
   103
  gobject_class->get_property =
hgs
parents:
diff changeset
   104
      GST_DEBUG_FUNCPTR (gst_auto_video_src_get_property);
hgs
parents:
diff changeset
   105
hgs
parents:
diff changeset
   106
  /**
hgs
parents:
diff changeset
   107
   * GstAutoVideoSrc:filter-caps
hgs
parents:
diff changeset
   108
   *
hgs
parents:
diff changeset
   109
   * This property will filter out candidate sources that can handle the specified
hgs
parents:
diff changeset
   110
   * caps. By default only video sources that support raw rgb and yuv video
hgs
parents:
diff changeset
   111
   * are selected.
hgs
parents:
diff changeset
   112
   *
hgs
parents:
diff changeset
   113
   * This property can only be set before the element goes to the READY state.
hgs
parents:
diff changeset
   114
   *
hgs
parents:
diff changeset
   115
   * Since: 0.10.14
hgs
parents:
diff changeset
   116
   **/
hgs
parents:
diff changeset
   117
  g_object_class_install_property (gobject_class, PROP_CAPS,
hgs
parents:
diff changeset
   118
      g_param_spec_boxed ("filter-caps", "Filter caps",
hgs
parents:
diff changeset
   119
          "Filter src candidates using these caps.", GST_TYPE_CAPS,
hgs
parents:
diff changeset
   120
          G_PARAM_READWRITE));
hgs
parents:
diff changeset
   121
}
hgs
parents:
diff changeset
   122
hgs
parents:
diff changeset
   123
static void
hgs
parents:
diff changeset
   124
gst_auto_video_src_dispose (GstAutoVideoSrc * src)
hgs
parents:
diff changeset
   125
{
hgs
parents:
diff changeset
   126
  gst_auto_video_src_clear_kid (src);
hgs
parents:
diff changeset
   127
hgs
parents:
diff changeset
   128
  if (src->filter_caps)
hgs
parents:
diff changeset
   129
    gst_caps_unref (src->filter_caps);
hgs
parents:
diff changeset
   130
  src->filter_caps = NULL;
hgs
parents:
diff changeset
   131
hgs
parents:
diff changeset
   132
  G_OBJECT_CLASS (parent_class)->dispose ((GObject *) src);
hgs
parents:
diff changeset
   133
}
hgs
parents:
diff changeset
   134
hgs
parents:
diff changeset
   135
static void
hgs
parents:
diff changeset
   136
gst_auto_video_src_clear_kid (GstAutoVideoSrc * src)
hgs
parents:
diff changeset
   137
{
hgs
parents:
diff changeset
   138
  if (src->kid) {
hgs
parents:
diff changeset
   139
    gst_element_set_state (src->kid, GST_STATE_NULL);
hgs
parents:
diff changeset
   140
    gst_bin_remove (GST_BIN (src), src->kid);
hgs
parents:
diff changeset
   141
    src->kid = NULL;
hgs
parents:
diff changeset
   142
  }
hgs
parents:
diff changeset
   143
}
hgs
parents:
diff changeset
   144
hgs
parents:
diff changeset
   145
/*
hgs
parents:
diff changeset
   146
 * Hack to make initial linking work; ideally, this'd work even when
hgs
parents:
diff changeset
   147
 * no target has been assigned to the ghostpad yet.
hgs
parents:
diff changeset
   148
 */
hgs
parents:
diff changeset
   149
hgs
parents:
diff changeset
   150
static void
hgs
parents:
diff changeset
   151
gst_auto_video_src_reset (GstAutoVideoSrc * src)
hgs
parents:
diff changeset
   152
{
hgs
parents:
diff changeset
   153
  GstPad *targetpad;
hgs
parents:
diff changeset
   154
hgs
parents:
diff changeset
   155
  /* Remove any existing element */
hgs
parents:
diff changeset
   156
  gst_auto_video_src_clear_kid (src);
hgs
parents:
diff changeset
   157
hgs
parents:
diff changeset
   158
  /* fakesrc 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 ("video/x-raw-yuv; video/x-raw-rgb");
hgs
parents:
diff changeset
   170
hgs
parents:
diff changeset
   171
static void
hgs
parents:
diff changeset
   172
gst_auto_video_src_init (GstAutoVideoSrc * src, GstAutoVideoSrcClass * 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_video_src_reset (src);
hgs
parents:
diff changeset
   178
hgs
parents:
diff changeset
   179
  /* set the default raw video 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_video_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
  /* video sources */
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, "Video")))
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_video_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_video_src_create_element_with_pretty_name (GstAutoVideoSrc * 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_video_src_find_best (GstAutoVideoSrc * 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_video_src_factory_filter, FALSE, src);
hgs
parents:
diff changeset
   256
  list = g_list_sort (list, (GCompareFunc) gst_auto_video_src_compare_ranks);
hgs
parents:
diff changeset
   257
hgs
parents:
diff changeset
   258
  GST_LOG_OBJECT (src, "Trying to find usable video devices ...");
hgs
parents:
diff changeset
   259
hgs
parents:
diff changeset
   260
  for (item = list; item != NULL; item = item->next) {
hgs
parents:
diff changeset
   261
    GstElementFactory *f = GST_ELEMENT_FACTORY (item->data);
hgs
parents:
diff changeset
   262
    GstElement *el;
hgs
parents:
diff changeset
   263
hgs
parents:
diff changeset
   264
    if ((el = gst_auto_video_src_create_element_with_pretty_name (src, f))) {
hgs
parents:
diff changeset
   265
      GstStateChangeReturn ret;
hgs
parents:
diff changeset
   266
hgs
parents:
diff changeset
   267
      GST_DEBUG_OBJECT (src, "Testing %s", GST_PLUGIN_FEATURE (f)->name);
hgs
parents:
diff changeset
   268
hgs
parents:
diff changeset
   269
      /* If AutoVideoSrc has been provided with filter caps,
hgs
parents:
diff changeset
   270
       * accept only sources that match with the filter caps */
hgs
parents:
diff changeset
   271
      if (src->filter_caps) {
hgs
parents:
diff changeset
   272
        el_pad = gst_element_get_static_pad (GST_ELEMENT (el), "src");
hgs
parents:
diff changeset
   273
        el_caps = gst_pad_get_caps (el_pad);
hgs
parents:
diff changeset
   274
        gst_object_unref (el_pad);
hgs
parents:
diff changeset
   275
        GST_DEBUG_OBJECT (src,
hgs
parents:
diff changeset
   276
            "Checking caps: %" GST_PTR_FORMAT " vs. %" GST_PTR_FORMAT,
hgs
parents:
diff changeset
   277
            src->filter_caps, el_caps);
hgs
parents:
diff changeset
   278
        intersect = gst_caps_intersect (src->filter_caps, el_caps);
hgs
parents:
diff changeset
   279
        no_match = gst_caps_is_empty (intersect);
hgs
parents:
diff changeset
   280
        gst_caps_unref (el_caps);
hgs
parents:
diff changeset
   281
        gst_caps_unref (intersect);
hgs
parents:
diff changeset
   282
hgs
parents:
diff changeset
   283
        if (no_match) {
hgs
parents:
diff changeset
   284
          GST_DEBUG_OBJECT (src, "Incompatible caps");
hgs
parents:
diff changeset
   285
          gst_object_unref (el);
hgs
parents:
diff changeset
   286
          continue;
hgs
parents:
diff changeset
   287
        } else {
hgs
parents:
diff changeset
   288
          GST_DEBUG_OBJECT (src, "Found compatible caps");
hgs
parents:
diff changeset
   289
        }
hgs
parents:
diff changeset
   290
      }
hgs
parents:
diff changeset
   291
hgs
parents:
diff changeset
   292
      gst_element_set_bus (el, bus);
hgs
parents:
diff changeset
   293
      ret = gst_element_set_state (el, GST_STATE_READY);
hgs
parents:
diff changeset
   294
      if (ret == GST_STATE_CHANGE_SUCCESS) {
hgs
parents:
diff changeset
   295
        GST_DEBUG_OBJECT (src, "This worked!");
hgs
parents:
diff changeset
   296
        choice = el;
hgs
parents:
diff changeset
   297
        break;
hgs
parents:
diff changeset
   298
      }
hgs
parents:
diff changeset
   299
hgs
parents:
diff changeset
   300
      /* collect all error messages */
hgs
parents:
diff changeset
   301
      while ((message = gst_bus_pop_filtered (bus, GST_MESSAGE_ERROR))) {
hgs
parents:
diff changeset
   302
        GST_DEBUG_OBJECT (src, "error message %" GST_PTR_FORMAT, message);
hgs
parents:
diff changeset
   303
        errors = g_slist_append (errors, message);
hgs
parents:
diff changeset
   304
      }
hgs
parents:
diff changeset
   305
hgs
parents:
diff changeset
   306
      gst_element_set_state (el, GST_STATE_NULL);
hgs
parents:
diff changeset
   307
      gst_object_unref (el);
hgs
parents:
diff changeset
   308
    }
hgs
parents:
diff changeset
   309
  }
hgs
parents:
diff changeset
   310
hgs
parents:
diff changeset
   311
  GST_DEBUG_OBJECT (src, "done trying");
hgs
parents:
diff changeset
   312
  if (!choice) {
hgs
parents:
diff changeset
   313
    if (errors) {
hgs
parents:
diff changeset
   314
      /* FIXME: we forward the first error for now; but later on it might make
hgs
parents:
diff changeset
   315
       * sense to actually analyse them */
hgs
parents:
diff changeset
   316
      gst_message_ref (GST_MESSAGE (errors->data));
hgs
parents:
diff changeset
   317
      GST_DEBUG_OBJECT (src, "reposting message %p", errors->data);
hgs
parents:
diff changeset
   318
      gst_element_post_message (GST_ELEMENT (src), GST_MESSAGE (errors->data));
hgs
parents:
diff changeset
   319
    } else {
hgs
parents:
diff changeset
   320
      /* send warning message to application and use a fakesrc */
hgs
parents:
diff changeset
   321
      GST_ELEMENT_WARNING (src, RESOURCE, NOT_FOUND, (NULL),
hgs
parents:
diff changeset
   322
          ("Failed to find a usable video source"));
hgs
parents:
diff changeset
   323
      choice = gst_element_factory_make ("fakesrc", "fake-video-src");
hgs
parents:
diff changeset
   324
      if (g_object_class_find_property (G_OBJECT_GET_CLASS (choice), "sync"))
hgs
parents:
diff changeset
   325
        g_object_set (choice, "sync", TRUE, NULL);
hgs
parents:
diff changeset
   326
      gst_element_set_state (choice, GST_STATE_READY);
hgs
parents:
diff changeset
   327
    }
hgs
parents:
diff changeset
   328
  }
hgs
parents:
diff changeset
   329
  gst_object_unref (bus);
hgs
parents:
diff changeset
   330
  gst_plugin_feature_list_free (list);
hgs
parents:
diff changeset
   331
  g_slist_foreach (errors, (GFunc) gst_mini_object_unref, NULL);
hgs
parents:
diff changeset
   332
  g_slist_free (errors);
hgs
parents:
diff changeset
   333
hgs
parents:
diff changeset
   334
  return choice;
hgs
parents:
diff changeset
   335
}
hgs
parents:
diff changeset
   336
hgs
parents:
diff changeset
   337
static gboolean
hgs
parents:
diff changeset
   338
gst_auto_video_src_detect (GstAutoVideoSrc * src)
hgs
parents:
diff changeset
   339
{
hgs
parents:
diff changeset
   340
  GstElement *esrc;
hgs
parents:
diff changeset
   341
  GstPad *targetpad;
hgs
parents:
diff changeset
   342
hgs
parents:
diff changeset
   343
  gst_auto_video_src_clear_kid (src);
hgs
parents:
diff changeset
   344
hgs
parents:
diff changeset
   345
  /* find element */
hgs
parents:
diff changeset
   346
  GST_DEBUG_OBJECT (src, "Creating new kid");
hgs
parents:
diff changeset
   347
  if (!(esrc = gst_auto_video_src_find_best (src)))
hgs
parents:
diff changeset
   348
    goto no_src;
hgs
parents:
diff changeset
   349
hgs
parents:
diff changeset
   350
  src->kid = esrc;
hgs
parents:
diff changeset
   351
  gst_bin_add (GST_BIN (src), esrc);
hgs
parents:
diff changeset
   352
hgs
parents:
diff changeset
   353
  /* attach ghost pad */
hgs
parents:
diff changeset
   354
  GST_DEBUG_OBJECT (src, "Re-assigning ghostpad");
hgs
parents:
diff changeset
   355
  targetpad = gst_element_get_static_pad (src->kid, "src");
hgs
parents:
diff changeset
   356
  if (!gst_ghost_pad_set_target (GST_GHOST_PAD (src->pad), targetpad))
hgs
parents:
diff changeset
   357
    goto target_failed;
hgs
parents:
diff changeset
   358
hgs
parents:
diff changeset
   359
  gst_object_unref (targetpad);
hgs
parents:
diff changeset
   360
  GST_DEBUG_OBJECT (src, "done changing auto video source");
hgs
parents:
diff changeset
   361
hgs
parents:
diff changeset
   362
  return TRUE;
hgs
parents:
diff changeset
   363
hgs
parents:
diff changeset
   364
  /* ERRORS */
hgs
parents:
diff changeset
   365
no_src:
hgs
parents:
diff changeset
   366
  {
hgs
parents:
diff changeset
   367
    GST_ELEMENT_ERROR (src, LIBRARY, INIT, (NULL),
hgs
parents:
diff changeset
   368
        ("Failed to find a supported video source"));
hgs
parents:
diff changeset
   369
    return FALSE;
hgs
parents:
diff changeset
   370
  }
hgs
parents:
diff changeset
   371
target_failed:
hgs
parents:
diff changeset
   372
  {
hgs
parents:
diff changeset
   373
    GST_ELEMENT_ERROR (src, LIBRARY, INIT, (NULL),
hgs
parents:
diff changeset
   374
        ("Failed to set target pad"));
hgs
parents:
diff changeset
   375
    gst_object_unref (targetpad);
hgs
parents:
diff changeset
   376
    return FALSE;
hgs
parents:
diff changeset
   377
  }
hgs
parents:
diff changeset
   378
}
hgs
parents:
diff changeset
   379
hgs
parents:
diff changeset
   380
static GstStateChangeReturn
hgs
parents:
diff changeset
   381
gst_auto_video_src_change_state (GstElement * element,
hgs
parents:
diff changeset
   382
    GstStateChange transition)
hgs
parents:
diff changeset
   383
{
hgs
parents:
diff changeset
   384
  GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
hgs
parents:
diff changeset
   385
  GstAutoVideoSrc *src = GST_AUTO_VIDEO_SRC (element);
hgs
parents:
diff changeset
   386
hgs
parents:
diff changeset
   387
  switch (transition) {
hgs
parents:
diff changeset
   388
    case GST_STATE_CHANGE_NULL_TO_READY:
hgs
parents:
diff changeset
   389
      if (!gst_auto_video_src_detect (src))
hgs
parents:
diff changeset
   390
        return GST_STATE_CHANGE_FAILURE;
hgs
parents:
diff changeset
   391
      break;
hgs
parents:
diff changeset
   392
    default:
hgs
parents:
diff changeset
   393
      break;
hgs
parents:
diff changeset
   394
  }
hgs
parents:
diff changeset
   395
hgs
parents:
diff changeset
   396
  ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
hgs
parents:
diff changeset
   397
hgs
parents:
diff changeset
   398
  switch (transition) {
hgs
parents:
diff changeset
   399
    case GST_STATE_CHANGE_READY_TO_NULL:
hgs
parents:
diff changeset
   400
      gst_auto_video_src_reset (src);
hgs
parents:
diff changeset
   401
      break;
hgs
parents:
diff changeset
   402
    default:
hgs
parents:
diff changeset
   403
      break;
hgs
parents:
diff changeset
   404
  }
hgs
parents:
diff changeset
   405
hgs
parents:
diff changeset
   406
  return ret;
hgs
parents:
diff changeset
   407
}
hgs
parents:
diff changeset
   408
hgs
parents:
diff changeset
   409
static void
hgs
parents:
diff changeset
   410
gst_auto_video_src_set_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
   411
    const GValue * value, GParamSpec * pspec)
hgs
parents:
diff changeset
   412
{
hgs
parents:
diff changeset
   413
  GstAutoVideoSrc *src = GST_AUTO_VIDEO_SRC (object);
hgs
parents:
diff changeset
   414
hgs
parents:
diff changeset
   415
  switch (prop_id) {
hgs
parents:
diff changeset
   416
    case PROP_CAPS:
hgs
parents:
diff changeset
   417
      if (src->filter_caps)
hgs
parents:
diff changeset
   418
        gst_caps_unref (src->filter_caps);
hgs
parents:
diff changeset
   419
      src->filter_caps = gst_caps_copy (gst_value_get_caps (value));
hgs
parents:
diff changeset
   420
      break;
hgs
parents:
diff changeset
   421
    default:
hgs
parents:
diff changeset
   422
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
hgs
parents:
diff changeset
   423
      break;
hgs
parents:
diff changeset
   424
  }
hgs
parents:
diff changeset
   425
}
hgs
parents:
diff changeset
   426
hgs
parents:
diff changeset
   427
static void
hgs
parents:
diff changeset
   428
gst_auto_video_src_get_property (GObject * object, guint prop_id,
hgs
parents:
diff changeset
   429
    GValue * value, GParamSpec * pspec)
hgs
parents:
diff changeset
   430
{
hgs
parents:
diff changeset
   431
  GstAutoVideoSrc *src = GST_AUTO_VIDEO_SRC (object);
hgs
parents:
diff changeset
   432
hgs
parents:
diff changeset
   433
  switch (prop_id) {
hgs
parents:
diff changeset
   434
    case PROP_CAPS:{
hgs
parents:
diff changeset
   435
      gst_value_set_caps (value, src->filter_caps);
hgs
parents:
diff changeset
   436
      break;
hgs
parents:
diff changeset
   437
    }
hgs
parents:
diff changeset
   438
    default:
hgs
parents:
diff changeset
   439
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
hgs
parents:
diff changeset
   440
      break;
hgs
parents:
diff changeset
   441
  }
hgs
parents:
diff changeset
   442
}