796da5ac6f4401532a4b5426bb411b26b2e70a95
[platform/upstream/gst-rtsp-server.git] / gst / rtsp-server / rtsp-session-media.c
1 /* GStreamer
2  * Copyright (C) 2008 Wim Taymans <wim.taymans at gmail.com>
3  *
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.
8  *
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.
13  *
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., 51 Franklin St, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  */
19 /**
20  * SECTION:rtsp-session-media
21  * @short_description: Media managed in a session
22  * @see_also: #GstRTSPMedia, #GstRTSPSession
23  *
24  * The #GstRTSPSessionMedia object manages a #GstRTSPMedia with a given path.
25  *
26  * With gst_rtsp_session_media_get_transport() and
27  * gst_rtsp_session_media_set_transport() the transports of a #GstRTSPStream of
28  * the managed #GstRTSPMedia can be retrieved and configured.
29  *
30  * Use gst_rtsp_session_media_set_state() to control the media state and
31  * transports.
32  *
33  * Last reviewed on 2013-07-16 (1.0.0)
34  */
35
36 #include <string.h>
37
38 #include "rtsp-session.h"
39
40 #define GST_RTSP_SESSION_MEDIA_GET_PRIVATE(obj)  \
41     (G_TYPE_INSTANCE_GET_PRIVATE ((obj), GST_TYPE_RTSP_SESSION_MEDIA, GstRTSPSessionMediaPrivate))
42
43 struct _GstRTSPSessionMediaPrivate
44 {
45   GMutex lock;
46   gchar *path;                  /* unmutable */
47   gint path_len;                /* unmutable */
48   GstRTSPMedia *media;          /* unmutable */
49   GstRTSPState state;           /* protected by lock */
50   guint counter;                /* protected by lock */
51
52   GPtrArray *transports;        /* protected by lock */
53 };
54
55 enum
56 {
57   PROP_0,
58   PROP_LAST
59 };
60
61 GST_DEBUG_CATEGORY_STATIC (rtsp_session_media_debug);
62 #define GST_CAT_DEFAULT rtsp_session_media_debug
63
64 static void gst_rtsp_session_media_finalize (GObject * obj);
65
66 G_DEFINE_TYPE (GstRTSPSessionMedia, gst_rtsp_session_media, G_TYPE_OBJECT);
67
68 static void
69 gst_rtsp_session_media_class_init (GstRTSPSessionMediaClass * klass)
70 {
71   GObjectClass *gobject_class;
72
73   g_type_class_add_private (klass, sizeof (GstRTSPSessionMediaPrivate));
74
75   gobject_class = G_OBJECT_CLASS (klass);
76
77   gobject_class->finalize = gst_rtsp_session_media_finalize;
78
79   GST_DEBUG_CATEGORY_INIT (rtsp_session_media_debug, "rtspsessionmedia", 0,
80       "GstRTSPSessionMedia");
81 }
82
83 static void
84 gst_rtsp_session_media_init (GstRTSPSessionMedia * media)
85 {
86   GstRTSPSessionMediaPrivate *priv = GST_RTSP_SESSION_MEDIA_GET_PRIVATE (media);
87
88   media->priv = priv;
89
90   g_mutex_init (&priv->lock);
91   priv->state = GST_RTSP_STATE_INIT;
92 }
93
94 static void
95 gst_rtsp_session_media_finalize (GObject * obj)
96 {
97   GstRTSPSessionMedia *media;
98   GstRTSPSessionMediaPrivate *priv;
99
100   media = GST_RTSP_SESSION_MEDIA (obj);
101   priv = media->priv;
102
103   GST_INFO ("free session media %p", media);
104
105   gst_rtsp_session_media_set_state (media, GST_STATE_NULL);
106
107   g_ptr_array_unref (priv->transports);
108
109   g_free (priv->path);
110   g_object_unref (priv->media);
111   g_mutex_clear (&priv->lock);
112
113   G_OBJECT_CLASS (gst_rtsp_session_media_parent_class)->finalize (obj);
114 }
115
116 static void
117 free_session_media (gpointer data)
118 {
119   if (data)
120     g_object_unref (data);
121 }
122
123 /**
124  * gst_rtsp_session_media_new:
125  * @path: the path
126  * @media: the #GstRTSPMedia
127  *
128  * Create a new #GstRTSPSessionMedia that manages the streams
129  * in @media for @path. @media should be prepared.
130  *
131  * Ownership is taken of @media.
132  *
133  * Returns: a new #GstRTSPSessionMedia.
134  */
135 GstRTSPSessionMedia *
136 gst_rtsp_session_media_new (const gchar * path, GstRTSPMedia * media)
137 {
138   GstRTSPSessionMediaPrivate *priv;
139   GstRTSPSessionMedia *result;
140   guint n_streams;
141
142   g_return_val_if_fail (path != NULL, NULL);
143   g_return_val_if_fail (GST_IS_RTSP_MEDIA (media), NULL);
144   g_return_val_if_fail (gst_rtsp_media_get_status (media) ==
145       GST_RTSP_MEDIA_STATUS_PREPARED, NULL);
146
147   result = g_object_new (GST_TYPE_RTSP_SESSION_MEDIA, NULL);
148   priv = result->priv;
149
150   priv->path = g_strdup (path);
151   priv->path_len = strlen (path);
152   priv->media = media;
153
154   /* prealloc the streams now, filled with NULL */
155   n_streams = gst_rtsp_media_n_streams (media);
156   priv->transports = g_ptr_array_new_full (n_streams, free_session_media);
157   g_ptr_array_set_size (priv->transports, n_streams);
158
159   return result;
160 }
161
162 /**
163  * gst_rtsp_session_media_matches:
164  * @media: a #GstRTSPSessionMedia
165  * @path: a path
166  * @matched: (out): the amount of matched characters of @path
167  *
168  * Check if the path of @media matches @path. It @path matches, the amount of
169  * matched characters is returned in @matched.
170  *
171  * Returns: %TRUE when @path matches the path of @media.
172  */
173 gboolean
174 gst_rtsp_session_media_matches (GstRTSPSessionMedia * media,
175     const gchar * path, gint * matched)
176 {
177   GstRTSPSessionMediaPrivate *priv;
178   gint len;
179
180   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
181   g_return_val_if_fail (path != NULL, FALSE);
182   g_return_val_if_fail (matched != NULL, FALSE);
183
184   priv = media->priv;
185   len = strlen (path);
186
187   /* path needs to be smaller than the media path */
188   if (len < priv->path_len)
189     return FALSE;
190
191   /* if media path is larger, it there should be a / following the path */
192   if (len > priv->path_len && path[priv->path_len] != '/')
193     return FALSE;
194
195   *matched = priv->path_len;
196
197   return strncmp (path, priv->path, priv->path_len) == 0;
198 }
199
200 /**
201  * gst_rtsp_session_media_get_media:
202  * @media: a #GstRTSPSessionMedia
203  *
204  * Get the #GstRTSPMedia that was used when constructing @media
205  *
206  * Returns: (transfer none): the #GstRTSPMedia of @media. Remains valid as long
207  *         as @media is valid.
208  */
209 GstRTSPMedia *
210 gst_rtsp_session_media_get_media (GstRTSPSessionMedia * media)
211 {
212   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
213
214   return media->priv->media;
215 }
216
217 /**
218  * gst_rtsp_session_media_get_base_time:
219  * @media: a #GstRTSPSessionMedia
220  *
221  * Get the base_time of the #GstRTSPMedia in @media
222  *
223  * Returns: the base_time of the media.
224  */
225 GstClockTime
226 gst_rtsp_session_media_get_base_time (GstRTSPSessionMedia * media)
227 {
228   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), GST_CLOCK_TIME_NONE);
229
230   return gst_rtsp_media_get_base_time (media->priv->media);
231 }
232
233 /**
234  * gst_rtsp_session_media_set_transport:
235  * @media: a #GstRTSPSessionMedia
236  * @stream: a #GstRTSPStream
237  * @tr: a #GstRTSPTransport
238  *
239  * Configure the transport for @stream to @tr in @media.
240  *
241  * Returns: (transfer none): the new or updated #GstRTSPStreamTransport for @stream.
242  */
243 GstRTSPStreamTransport *
244 gst_rtsp_session_media_set_transport (GstRTSPSessionMedia * media,
245     GstRTSPStream * stream, GstRTSPTransport * tr)
246 {
247   GstRTSPSessionMediaPrivate *priv;
248   GstRTSPStreamTransport *result;
249   guint idx;
250
251   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
252   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
253   g_return_val_if_fail (tr != NULL, NULL);
254   priv = media->priv;
255   idx = gst_rtsp_stream_get_index (stream);
256   g_return_val_if_fail (idx < priv->transports->len, NULL);
257
258   g_mutex_lock (&priv->lock);
259   result = g_ptr_array_index (priv->transports, idx);
260   if (result == NULL) {
261     result = gst_rtsp_stream_transport_new (stream, tr);
262     g_ptr_array_index (priv->transports, idx) = result;
263     g_mutex_unlock (&priv->lock);
264   } else {
265     gst_rtsp_stream_transport_set_transport (result, tr);
266     g_mutex_unlock (&priv->lock);
267   }
268
269   return result;
270 }
271
272 /**
273  * gst_rtsp_session_media_get_transport:
274  * @media: a #GstRTSPSessionMedia
275  * @idx: the stream index
276  *
277  * Get a previously created #GstRTSPStreamTransport for the stream at @idx.
278  *
279  * Returns: (transfer none): a #GstRTSPStreamTransport that is valid until the
280  * session of @media is unreffed.
281  */
282 GstRTSPStreamTransport *
283 gst_rtsp_session_media_get_transport (GstRTSPSessionMedia * media, guint idx)
284 {
285   GstRTSPSessionMediaPrivate *priv;
286   GstRTSPStreamTransport *result;
287
288   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
289   priv = media->priv;
290   g_return_val_if_fail (idx < priv->transports->len, NULL);
291
292   g_mutex_lock (&priv->lock);
293   result = g_ptr_array_index (priv->transports, idx);
294   g_mutex_unlock (&priv->lock);
295
296   return result;
297 }
298
299 /**
300  * gst_rtsp_session_media_alloc_channels:
301  * @media: a #GstRTSPSessionMedia
302  * @range: a #GstRTSPRange
303  *
304  * Fill @range with the next available min and max channels for
305  * interleaved transport.
306  *
307  * Returns: %TRUE on success.
308  */
309 gboolean
310 gst_rtsp_session_media_alloc_channels (GstRTSPSessionMedia * media,
311     GstRTSPRange * range)
312 {
313   GstRTSPSessionMediaPrivate *priv;
314
315   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
316
317   priv = media->priv;
318
319   g_mutex_lock (&priv->lock);
320   range->min = priv->counter++;
321   range->max = priv->counter++;
322   g_mutex_unlock (&priv->lock);
323
324   return TRUE;
325 }
326
327 /**
328  * gst_rtsp_session_media_set_state:
329  * @media: a #GstRTSPSessionMedia
330  * @state: the new state
331  *
332  * Tell the media object @media to change to @state.
333  *
334  * Returns: %TRUE on success.
335  */
336 gboolean
337 gst_rtsp_session_media_set_state (GstRTSPSessionMedia * media, GstState state)
338 {
339   GstRTSPSessionMediaPrivate *priv;
340   gboolean ret;
341
342   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
343
344   priv = media->priv;
345
346   g_mutex_lock (&priv->lock);
347   ret = gst_rtsp_media_set_state (priv->media, state, priv->transports);
348   g_mutex_unlock (&priv->lock);
349
350   return ret;
351 }
352
353 /**
354  * gst_rtsp_session_media_set_rtsp_state:
355  * @media: a #GstRTSPSessionMedia
356  * @state: a #GstRTSPState
357  *
358  * Set the RTSP state of @media to @state.
359  */
360 void
361 gst_rtsp_session_media_set_rtsp_state (GstRTSPSessionMedia * media,
362     GstRTSPState state)
363 {
364   GstRTSPSessionMediaPrivate *priv;
365
366   g_return_if_fail (GST_IS_RTSP_SESSION_MEDIA (media));
367
368   priv = media->priv;
369
370   g_mutex_lock (&priv->lock);
371   priv->state = state;
372   g_mutex_unlock (&priv->lock);
373 }
374
375 /**
376  * gst_rtsp_session_media_get_rtsp_state:
377  * @media: a #GstRTSPSessionMedia
378  *
379  * Get the current RTSP state of @media.
380  *
381  * Returns: the current RTSP state of @media.
382  */
383 GstRTSPState
384 gst_rtsp_session_media_get_rtsp_state (GstRTSPSessionMedia * media)
385 {
386   GstRTSPSessionMediaPrivate *priv;
387   GstRTSPState ret;
388
389   g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media),
390       GST_RTSP_STATE_INVALID);
391
392   priv = media->priv;
393
394   g_mutex_lock (&priv->lock);
395   ret = priv->state;
396   g_mutex_unlock (&priv->lock);
397
398   return ret;
399 }