2 * Copyright (C) 2008 Wim Taymans <wim.taymans at gmail.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.
20 #include "rtsp-media-factory.h"
22 #define DEFAULT_LAUNCH NULL
23 #define DEFAULT_SHARED FALSE
24 #define DEFAULT_EOS_SHUTDOWN FALSE
35 GST_DEBUG_CATEGORY (rtsp_media_debug);
36 #define GST_CAT_DEFAULT rtsp_media_debug
38 static void gst_rtsp_media_factory_get_property (GObject * object, guint propid,
39 GValue * value, GParamSpec * pspec);
40 static void gst_rtsp_media_factory_set_property (GObject * object, guint propid,
41 const GValue * value, GParamSpec * pspec);
42 static void gst_rtsp_media_factory_finalize (GObject * obj);
44 static gchar *default_gen_key (GstRTSPMediaFactory * factory,
45 const GstRTSPUrl * url);
46 static GstElement *default_get_element (GstRTSPMediaFactory * factory,
47 const GstRTSPUrl * url);
48 static GstRTSPMedia *default_construct (GstRTSPMediaFactory * factory,
49 const GstRTSPUrl * url);
50 static void default_configure (GstRTSPMediaFactory * factory,
51 GstRTSPMedia * media);
52 static GstElement *default_create_pipeline (GstRTSPMediaFactory * factory,
53 GstRTSPMedia * media);
55 G_DEFINE_TYPE (GstRTSPMediaFactory, gst_rtsp_media_factory, G_TYPE_OBJECT);
58 gst_rtsp_media_factory_class_init (GstRTSPMediaFactoryClass * klass)
60 GObjectClass *gobject_class;
62 gobject_class = G_OBJECT_CLASS (klass);
64 gobject_class->get_property = gst_rtsp_media_factory_get_property;
65 gobject_class->set_property = gst_rtsp_media_factory_set_property;
66 gobject_class->finalize = gst_rtsp_media_factory_finalize;
69 * GstRTSPMediaFactory::launch
71 * The gst_parse_launch() line to use for constructing the pipeline in the
72 * default prepare vmethod.
74 * The pipeline description should return a GstBin as the toplevel element
75 * which can be accomplished by enclosing the dscription with brackets '('
78 * The description should return a pipeline with payloaders named pay0, pay1,
79 * etc.. Each of the payloaders will result in a stream.
81 * Support for dynamic payloaders can be accomplished by adding payloaders
82 * named dynpay0, dynpay1, etc..
84 g_object_class_install_property (gobject_class, PROP_LAUNCH,
85 g_param_spec_string ("launch", "Launch",
86 "A launch description of the pipeline", DEFAULT_LAUNCH,
87 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
89 g_object_class_install_property (gobject_class, PROP_SHARED,
90 g_param_spec_boolean ("shared", "Shared",
91 "If media from this factory is shared", DEFAULT_SHARED,
92 G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
94 g_object_class_install_property (gobject_class, PROP_EOS_SHUTDOWN,
95 g_param_spec_boolean ("eos-shutdown", "EOS Shutdown",
96 "Send EOS down the pipeline before shutting down",
97 DEFAULT_EOS_SHUTDOWN, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
99 klass->gen_key = default_gen_key;
100 klass->get_element = default_get_element;
101 klass->construct = default_construct;
102 klass->configure = default_configure;
103 klass->create_pipeline = default_create_pipeline;
105 GST_DEBUG_CATEGORY_INIT (rtsp_media_debug, "rtspmedia", 0, "GstRTSPMedia");
109 gst_rtsp_media_factory_init (GstRTSPMediaFactory * factory)
111 factory->launch = g_strdup (DEFAULT_LAUNCH);
112 factory->shared = DEFAULT_SHARED;
113 factory->eos_shutdown = DEFAULT_EOS_SHUTDOWN;
115 factory->lock = g_mutex_new ();
116 factory->medias_lock = g_mutex_new ();
117 factory->medias = g_hash_table_new_full (g_str_hash, g_str_equal,
118 g_free, g_object_unref);
122 gst_rtsp_media_factory_finalize (GObject * obj)
124 GstRTSPMediaFactory *factory = GST_RTSP_MEDIA_FACTORY (obj);
126 g_hash_table_unref (factory->medias);
127 g_mutex_free (factory->medias_lock);
128 g_free (factory->launch);
129 g_mutex_free (factory->lock);
131 G_OBJECT_CLASS (gst_rtsp_media_factory_parent_class)->finalize (obj);
135 gst_rtsp_media_factory_get_property (GObject * object, guint propid,
136 GValue * value, GParamSpec * pspec)
138 GstRTSPMediaFactory *factory = GST_RTSP_MEDIA_FACTORY (object);
142 g_value_take_string (value, gst_rtsp_media_factory_get_launch (factory));
145 g_value_set_boolean (value, gst_rtsp_media_factory_is_shared (factory));
147 case PROP_EOS_SHUTDOWN:
148 g_value_set_boolean (value,
149 gst_rtsp_media_factory_is_eos_shutdown (factory));
152 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, propid, pspec);
157 gst_rtsp_media_factory_set_property (GObject * object, guint propid,
158 const GValue * value, GParamSpec * pspec)
160 GstRTSPMediaFactory *factory = GST_RTSP_MEDIA_FACTORY (object);
164 gst_rtsp_media_factory_set_launch (factory, g_value_get_string (value));
167 gst_rtsp_media_factory_set_shared (factory, g_value_get_boolean (value));
169 case PROP_EOS_SHUTDOWN:
170 gst_rtsp_media_factory_set_eos_shutdown (factory,
171 g_value_get_boolean (value));
174 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, propid, pspec);
179 * gst_rtsp_media_factory_new:
181 * Create a new #GstRTSPMediaFactory instance.
183 * Returns: a new #GstRTSPMediaFactory object.
185 GstRTSPMediaFactory *
186 gst_rtsp_media_factory_new (void)
188 GstRTSPMediaFactory *result;
190 result = g_object_new (GST_TYPE_RTSP_MEDIA_FACTORY, NULL);
196 * gst_rtsp_media_factory_set_launch:
197 * @factory: a #GstRTSPMediaFactory
198 * @launch: the launch description
201 * The gst_parse_launch() line to use for constructing the pipeline in the
202 * default prepare vmethod.
204 * The pipeline description should return a GstBin as the toplevel element
205 * which can be accomplished by enclosing the dscription with brackets '('
208 * The description should return a pipeline with payloaders named pay0, pay1,
209 * etc.. Each of the payloaders will result in a stream.
212 gst_rtsp_media_factory_set_launch (GstRTSPMediaFactory * factory,
213 const gchar * launch)
215 g_return_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory));
216 g_return_if_fail (launch != NULL);
218 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
219 g_free (factory->launch);
220 factory->launch = g_strdup (launch);
221 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
225 * gst_rtsp_media_factory_get_launch:
226 * @factory: a #GstRTSPMediaFactory
228 * Get the gst_parse_launch() pipeline description that will be used in the
229 * default prepare vmethod.
231 * Returns: the configured launch description. g_free() after usage.
234 gst_rtsp_media_factory_get_launch (GstRTSPMediaFactory * factory)
238 g_return_val_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory), NULL);
240 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
241 result = g_strdup (factory->launch);
242 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
248 * gst_rtsp_media_factory_set_shared:
249 * @factory: a #GstRTSPMediaFactory
250 * @shared: the new value
252 * Configure if media created from this factory can be shared between clients.
255 gst_rtsp_media_factory_set_shared (GstRTSPMediaFactory * factory,
258 g_return_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory));
260 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
261 factory->shared = shared;
262 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
266 * gst_rtsp_media_factory_is_shared:
267 * @factory: a #GstRTSPMediaFactory
269 * Get if media created from this factory can be shared between clients.
271 * Returns: %TRUE if the media will be shared between clients.
274 gst_rtsp_media_factory_is_shared (GstRTSPMediaFactory * factory)
278 g_return_val_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory), FALSE);
280 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
281 result = factory->shared;
282 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
288 * gst_rtsp_media_factory_set_eos_shutdown:
289 * @factory: a #GstRTSPMediaFactory
290 * @eos_shutdown: the new value
292 * Configure if media created from this factory will have an EOS sent to the
293 * pipeline before shutdown.
296 gst_rtsp_media_factory_set_eos_shutdown (GstRTSPMediaFactory * factory,
297 gboolean eos_shutdown)
299 g_return_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory));
301 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
302 factory->eos_shutdown = eos_shutdown;
303 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
307 * gst_rtsp_media_factory_is_eos_shutdown:
308 * @factory: a #GstRTSPMediaFactory
310 * Get if media created from this factory will have an EOS event sent to the
311 * pipeline before shutdown.
313 * Returns: %TRUE if the media will receive EOS before shutdown.
316 gst_rtsp_media_factory_is_eos_shutdown (GstRTSPMediaFactory * factory)
320 g_return_val_if_fail (GST_IS_RTSP_MEDIA_FACTORY (factory), FALSE);
322 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
323 result = factory->eos_shutdown;
324 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
330 compare_media (gpointer key, GstRTSPMedia * media1, GstRTSPMedia * media2)
332 return (media1 == media2);
336 media_unprepared (GstRTSPMedia * media, GstRTSPMediaFactory * factory)
338 g_mutex_lock (factory->medias_lock);
339 g_hash_table_foreach_remove (factory->medias, (GHRFunc) compare_media, media);
340 g_mutex_unlock (factory->medias_lock);
344 * gst_rtsp_media_factory_construct:
345 * @factory: a #GstRTSPMediaFactory
348 * Prepare the media object and create its streams. Implementations
349 * should create the needed gstreamer elements and add them to the result
350 * object. No state changes should be performed on them yet.
352 * One or more GstRTSPMediaStream objects should be added to the result with
353 * the srcpad member set to a source pad that produces buffer of type
356 * Returns: a new #GstRTSPMedia if the media could be prepared.
359 gst_rtsp_media_factory_construct (GstRTSPMediaFactory * factory,
360 const GstRTSPUrl * url)
364 GstRTSPMediaFactoryClass *klass;
366 klass = GST_RTSP_MEDIA_FACTORY_GET_CLASS (factory);
368 /* convert the url to a key for the hashtable. NULL return or a NULL function
369 * will not cache anything for this factory. */
371 key = klass->gen_key (factory, url);
375 g_mutex_lock (factory->medias_lock);
377 /* we have a key, see if we find a cached media */
378 media = g_hash_table_lookup (factory->medias, key);
380 g_object_ref (media);
385 /* nothing cached found, try to create one */
386 if (klass->construct)
387 media = klass->construct (factory, url);
392 /* configure the media */
393 if (klass->configure)
394 klass->configure (factory, media);
396 /* check if we can cache this media */
397 if (gst_rtsp_media_is_shared (media)) {
398 /* insert in the hashtable, takes ownership of the key */
399 g_object_ref (media);
400 g_hash_table_insert (factory->medias, key, media);
403 if (!gst_rtsp_media_is_reusable (media)) {
404 /* when not reusable, connect to the unprepare signal to remove the item
405 * from our cache when it gets unprepared */
406 g_signal_connect (media, "unprepared", (GCallback) media_unprepared,
411 g_mutex_unlock (factory->medias_lock);
416 GST_INFO ("constructed media %p for url %s", media, url->abspath);
422 default_gen_key (GstRTSPMediaFactory * factory, const GstRTSPUrl * url)
425 const gchar *pre_query;
428 pre_query = url->query ? "?" : "";
429 query = url->query ? url->query : "";
432 g_strdup_printf ("%u%s%s%s", url->port, url->abspath, pre_query, query);
438 default_get_element (GstRTSPMediaFactory * factory, const GstRTSPUrl * url)
441 GError *error = NULL;
443 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
444 /* we need a parse syntax */
445 if (factory->launch == NULL)
448 /* parse the user provided launch line */
449 element = gst_parse_launch (factory->launch, &error);
453 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
456 /* a recoverable error was encountered */
457 GST_WARNING ("recoverable parsing error: %s", error->message);
458 g_error_free (error);
465 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
466 g_critical ("no launch line specified");
471 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
472 g_critical ("could not parse launch syntax (%s): %s", factory->launch,
473 (error ? error->message : "unknown reason"));
475 g_error_free (error);
480 /* try to find all the payloader elements, they should be named 'pay%d'. for
481 * each of the payloaders we will create a stream and collect the source pad. */
483 gst_rtsp_media_factory_collect_streams (GstRTSPMediaFactory * factory,
484 const GstRTSPUrl * url, GstRTSPMedia * media)
486 GstElement *element, *elem;
489 GstRTSPMediaStream *stream;
492 element = media->element;
495 for (i = 0; have_elem; i++) {
500 name = g_strdup_printf ("pay%d", i);
501 if ((elem = gst_bin_get_by_name (GST_BIN (element), name))) {
502 /* create the stream */
503 stream = g_new0 (GstRTSPMediaStream, 1);
504 stream->payloader = elem;
506 GST_INFO ("found stream %d with payloader %p", i, elem);
508 pad = gst_element_get_static_pad (elem, "src");
510 /* ghost the pad of the payloader to the element */
511 stream->srcpad = gst_ghost_pad_new (name, pad);
512 gst_pad_set_active (stream->srcpad, TRUE);
513 gst_element_add_pad (media->element, stream->srcpad);
514 gst_object_unref (elem);
517 g_array_append_val (media->streams, stream);
522 name = g_strdup_printf ("dynpay%d", i);
523 if ((elem = gst_bin_get_by_name (GST_BIN (element), name))) {
524 /* a stream that will dynamically create pads to provide RTP packets */
526 GST_INFO ("found dynamic element %d, %p", i, elem);
528 media->dynamic = g_list_prepend (media->dynamic, elem);
536 static GstRTSPMedia *
537 default_construct (GstRTSPMediaFactory * factory, const GstRTSPUrl * url)
541 GstRTSPMediaFactoryClass *klass;
543 klass = GST_RTSP_MEDIA_FACTORY_GET_CLASS (factory);
545 if (!klass->create_pipeline)
548 if (klass->get_element)
549 element = klass->get_element (factory, url);
555 /* create a new empty media */
556 media = gst_rtsp_media_new ();
557 media->element = element;
559 media->pipeline = klass->create_pipeline (factory, media);
560 if (media->pipeline == NULL)
563 gst_rtsp_media_factory_collect_streams (factory, url, media);
570 g_critical ("no create_pipeline function");
575 g_critical ("could not create element");
580 g_critical ("can't create pipeline");
581 g_object_unref (media);
587 default_create_pipeline (GstRTSPMediaFactory * factory, GstRTSPMedia * media)
589 GstElement *pipeline;
591 if (media->element == NULL)
594 pipeline = gst_pipeline_new ("media-pipeline");
595 gst_bin_add (GST_BIN_CAST (pipeline), media->element);
602 g_critical ("no element");
608 default_configure (GstRTSPMediaFactory * factory, GstRTSPMedia * media)
610 gboolean shared, eos_shutdown;
612 /* configure the sharedness */
613 GST_RTSP_MEDIA_FACTORY_LOCK (factory);
614 shared = factory->shared;
615 eos_shutdown = factory->eos_shutdown;
616 GST_RTSP_MEDIA_FACTORY_UNLOCK (factory);
618 gst_rtsp_media_set_shared (media, shared);
619 gst_rtsp_media_set_eos_shutdown (media, eos_shutdown);