2012-10-25 19:29:58 +00:00
|
|
|
/* GStreamer
|
|
|
|
* Copyright (C) 2008 Wim Taymans <wim.taymans at gmail.com>
|
|
|
|
*
|
|
|
|
* 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
|
2012-11-04 00:14:25 +00:00
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
2012-10-25 19:29:58 +00:00
|
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "rtsp-session.h"
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
#define GST_RTSP_SESSION_MEDIA_GET_PRIVATE(obj) \
|
|
|
|
(G_TYPE_INSTANCE_GET_PRIVATE ((obj), GST_TYPE_RTSP_SESSION_MEDIA, GstRTSPSessionMediaPrivate))
|
|
|
|
|
|
|
|
struct _GstRTSPSessionMediaPrivate
|
|
|
|
{
|
|
|
|
GMutex lock;
|
2013-07-03 10:37:48 +00:00
|
|
|
gchar *path; /* unmutable */
|
|
|
|
gint path_len; /* unmutable */
|
2013-02-18 19:59:58 +00:00
|
|
|
GstRTSPMedia *media; /* unmutable */
|
|
|
|
GstRTSPState state; /* protected by lock */
|
|
|
|
guint counter; /* protected by lock */
|
2012-11-29 10:11:05 +00:00
|
|
|
|
2013-02-18 19:59:58 +00:00
|
|
|
GPtrArray *transports; /* protected by lock */
|
2012-11-29 10:11:05 +00:00
|
|
|
};
|
|
|
|
|
2012-10-25 19:29:58 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (rtsp_session_media_debug);
|
|
|
|
#define GST_CAT_DEFAULT rtsp_session_media_debug
|
|
|
|
|
|
|
|
static void gst_rtsp_session_media_finalize (GObject * obj);
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (GstRTSPSessionMedia, gst_rtsp_session_media, G_TYPE_OBJECT);
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_rtsp_session_media_class_init (GstRTSPSessionMediaClass * klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class;
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
g_type_class_add_private (klass, sizeof (GstRTSPSessionMediaPrivate));
|
|
|
|
|
2012-10-25 19:29:58 +00:00
|
|
|
gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
gobject_class->finalize = gst_rtsp_session_media_finalize;
|
|
|
|
|
|
|
|
GST_DEBUG_CATEGORY_INIT (rtsp_session_media_debug, "rtspsessionmedia", 0,
|
|
|
|
"GstRTSPSessionMedia");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_rtsp_session_media_init (GstRTSPSessionMedia * media)
|
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv = GST_RTSP_SESSION_MEDIA_GET_PRIVATE (media);
|
|
|
|
|
|
|
|
media->priv = priv;
|
|
|
|
|
|
|
|
g_mutex_init (&priv->lock);
|
|
|
|
priv->state = GST_RTSP_STATE_INIT;
|
2012-10-25 19:29:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gst_rtsp_session_media_finalize (GObject * obj)
|
|
|
|
{
|
|
|
|
GstRTSPSessionMedia *media;
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
media = GST_RTSP_SESSION_MEDIA (obj);
|
2012-11-29 10:11:05 +00:00
|
|
|
priv = media->priv;
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
GST_INFO ("free session media %p", media);
|
|
|
|
|
|
|
|
gst_rtsp_session_media_set_state (media, GST_STATE_NULL);
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
g_ptr_array_unref (priv->transports);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
2013-07-03 10:37:48 +00:00
|
|
|
g_free (priv->path);
|
2012-11-29 10:11:05 +00:00
|
|
|
g_object_unref (priv->media);
|
|
|
|
g_mutex_clear (&priv->lock);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (gst_rtsp_session_media_parent_class)->finalize (obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_session_media (gpointer data)
|
|
|
|
{
|
|
|
|
if (data)
|
|
|
|
g_object_unref (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_new:
|
2013-07-03 10:37:48 +00:00
|
|
|
* @path: the path
|
2012-10-25 19:29:58 +00:00
|
|
|
* @media: the #GstRTSPMedia
|
|
|
|
*
|
|
|
|
* Create a new #GstRTPSessionMedia that manages the streams
|
2013-07-03 10:37:48 +00:00
|
|
|
* in @media for @path. @media should be prepared.
|
2012-10-25 19:29:58 +00:00
|
|
|
*
|
|
|
|
* Ownership is taken of @media.
|
|
|
|
*
|
|
|
|
* Returns: a new #GstRTSPSessionMedia.
|
|
|
|
*/
|
|
|
|
GstRTSPSessionMedia *
|
2013-07-03 10:37:48 +00:00
|
|
|
gst_rtsp_session_media_new (const gchar * path, GstRTSPMedia * media)
|
2012-10-25 19:29:58 +00:00
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
2012-10-25 19:29:58 +00:00
|
|
|
GstRTSPSessionMedia *result;
|
|
|
|
guint n_streams;
|
|
|
|
|
2013-07-03 10:37:48 +00:00
|
|
|
g_return_val_if_fail (path != NULL, NULL);
|
2012-10-25 19:29:58 +00:00
|
|
|
g_return_val_if_fail (GST_IS_RTSP_MEDIA (media), NULL);
|
2012-11-29 10:11:05 +00:00
|
|
|
g_return_val_if_fail (gst_rtsp_media_get_status (media) ==
|
|
|
|
GST_RTSP_MEDIA_STATUS_PREPARED, NULL);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
result = g_object_new (GST_TYPE_RTSP_SESSION_MEDIA, NULL);
|
2012-11-29 10:11:05 +00:00
|
|
|
priv = result->priv;
|
|
|
|
|
2013-07-03 10:37:48 +00:00
|
|
|
priv->path = g_strdup (path);
|
|
|
|
priv->path_len = strlen (path);
|
2012-11-29 10:11:05 +00:00
|
|
|
priv->media = media;
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
/* prealloc the streams now, filled with NULL */
|
|
|
|
n_streams = gst_rtsp_media_n_streams (media);
|
2012-11-29 10:11:05 +00:00
|
|
|
priv->transports = g_ptr_array_new_full (n_streams, free_session_media);
|
|
|
|
g_ptr_array_set_size (priv->transports, n_streams);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
/**
|
2013-07-03 10:37:48 +00:00
|
|
|
* gst_rtsp_session_media_matches:
|
2012-11-29 10:11:05 +00:00
|
|
|
* @media: a #GstRTSPSessionMedia
|
2013-07-03 10:37:48 +00:00
|
|
|
* @path: a path
|
|
|
|
* @matched: the amount of matched characters of @path
|
2012-11-29 10:11:05 +00:00
|
|
|
*
|
2013-07-03 10:37:48 +00:00
|
|
|
* Check if the path of @media matches @path. It @path matches, the amount of
|
|
|
|
* matched characters is returned in @matched.
|
2012-11-29 10:11:05 +00:00
|
|
|
*
|
2013-07-03 10:37:48 +00:00
|
|
|
* Returns: %TRUE when @path matches the path of @media.
|
2012-11-29 10:11:05 +00:00
|
|
|
*/
|
|
|
|
gboolean
|
2013-07-03 10:37:48 +00:00
|
|
|
gst_rtsp_session_media_matches (GstRTSPSessionMedia * media,
|
|
|
|
const gchar * path, gint * matched)
|
2012-11-29 10:11:05 +00:00
|
|
|
{
|
2013-07-03 10:37:48 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
|
|
|
gint len;
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
|
2013-07-03 10:37:48 +00:00
|
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
|
|
g_return_val_if_fail (matched != NULL, FALSE);
|
|
|
|
|
|
|
|
priv = media->priv;
|
|
|
|
len = strlen (path);
|
|
|
|
|
|
|
|
/* path needs to be smaller than the media path */
|
|
|
|
if (len < priv->path_len)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* if media path is larger, it there should be a / following the path */
|
|
|
|
if (len > priv->path_len && path[priv->path_len] != '/')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*matched = priv->path_len;
|
2012-11-29 10:11:05 +00:00
|
|
|
|
2013-07-03 10:37:48 +00:00
|
|
|
return strncmp (path, priv->path, priv->path_len) == 0;
|
2012-11-29 10:11:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_get_media:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
*
|
|
|
|
* Get the #GstRTSPMedia that was used when constructing @media
|
|
|
|
*
|
|
|
|
* Returns: (transfer none): the #GstRTSPMedia of @media. Remains valid as long
|
|
|
|
* as @media is valid.
|
|
|
|
*/
|
|
|
|
GstRTSPMedia *
|
|
|
|
gst_rtsp_session_media_get_media (GstRTSPSessionMedia * media)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
|
|
|
|
|
|
|
|
return media->priv->media;
|
|
|
|
}
|
|
|
|
|
2013-04-12 10:34:38 +00:00
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_get_base_time:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
*
|
|
|
|
* Get the base_time of the #GstRTSPMedia in @media
|
|
|
|
*
|
|
|
|
* Returns: the base_time of the media.
|
|
|
|
*/
|
|
|
|
GstClockTime
|
|
|
|
gst_rtsp_session_media_get_base_time (GstRTSPSessionMedia * media)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), GST_CLOCK_TIME_NONE);
|
|
|
|
|
|
|
|
return gst_rtsp_media_get_base_time (media->priv->media);
|
|
|
|
}
|
|
|
|
|
2012-10-27 22:23:57 +00:00
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_set_transport:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
* @stream: a #GstRTSPStream
|
|
|
|
* @tr: a #GstRTSPTransport
|
|
|
|
*
|
|
|
|
* Configure the transport for @stream to @tr in @media.
|
|
|
|
*
|
2012-10-28 12:48:44 +00:00
|
|
|
* Returns: (transfer none): the new or updated #GstRTSPStreamTransport for @stream.
|
2012-10-27 22:23:57 +00:00
|
|
|
*/
|
|
|
|
GstRTSPStreamTransport *
|
|
|
|
gst_rtsp_session_media_set_transport (GstRTSPSessionMedia * media,
|
2012-11-15 15:18:29 +00:00
|
|
|
GstRTSPStream * stream, GstRTSPTransport * tr)
|
2012-10-27 22:23:57 +00:00
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
2012-10-27 22:23:57 +00:00
|
|
|
GstRTSPStreamTransport *result;
|
2012-11-29 10:11:05 +00:00
|
|
|
guint idx;
|
2012-10-27 22:23:57 +00:00
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
|
2012-11-29 10:11:05 +00:00
|
|
|
g_return_val_if_fail (tr != NULL, NULL);
|
|
|
|
priv = media->priv;
|
|
|
|
idx = gst_rtsp_stream_get_index (stream);
|
|
|
|
g_return_val_if_fail (idx < priv->transports->len, NULL);
|
2012-10-27 22:23:57 +00:00
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
result = g_ptr_array_index (priv->transports, idx);
|
2012-10-27 22:23:57 +00:00
|
|
|
if (result == NULL) {
|
2012-11-15 15:18:29 +00:00
|
|
|
result = gst_rtsp_stream_transport_new (stream, tr);
|
2012-11-29 10:11:05 +00:00
|
|
|
g_ptr_array_index (priv->transports, idx) = result;
|
|
|
|
g_mutex_unlock (&priv->lock);
|
2012-10-27 22:23:57 +00:00
|
|
|
} else {
|
2012-11-15 15:18:29 +00:00
|
|
|
gst_rtsp_stream_transport_set_transport (result, tr);
|
2012-11-29 10:11:05 +00:00
|
|
|
g_mutex_unlock (&priv->lock);
|
2012-10-27 22:23:57 +00:00
|
|
|
}
|
2012-11-12 15:51:03 +00:00
|
|
|
|
2012-10-27 22:23:57 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-10-25 19:29:58 +00:00
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_get_transport:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
* @idx: the stream index
|
|
|
|
*
|
2012-10-27 22:23:57 +00:00
|
|
|
* Get a previously created #GstRTSPStreamTransport for the stream at @idx.
|
2012-10-25 19:29:58 +00:00
|
|
|
*
|
2012-10-28 12:48:44 +00:00
|
|
|
* Returns: (transfer none): a #GstRTSPStreamTransport that is valid until the
|
|
|
|
* session of @media is unreffed.
|
2012-10-25 19:29:58 +00:00
|
|
|
*/
|
|
|
|
GstRTSPStreamTransport *
|
|
|
|
gst_rtsp_session_media_get_transport (GstRTSPSessionMedia * media, guint idx)
|
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
2012-10-25 19:29:58 +00:00
|
|
|
GstRTSPStreamTransport *result;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), NULL);
|
2012-11-29 10:11:05 +00:00
|
|
|
priv = media->priv;
|
|
|
|
g_return_val_if_fail (idx < priv->transports->len, NULL);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
result = g_ptr_array_index (priv->transports, idx);
|
|
|
|
g_mutex_unlock (&priv->lock);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_alloc_channels:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
* @range: a #GstRTSPRange
|
|
|
|
*
|
|
|
|
* Fill @range with the next available min and max channels for
|
|
|
|
* interleaved transport.
|
|
|
|
*
|
|
|
|
* Returns: %TRUE on success.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gst_rtsp_session_media_alloc_channels (GstRTSPSessionMedia * media,
|
|
|
|
GstRTSPRange * range)
|
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
|
|
|
|
2012-10-25 19:29:58 +00:00
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
priv = media->priv;
|
|
|
|
|
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
range->min = priv->counter++;
|
|
|
|
range->max = priv->counter++;
|
|
|
|
g_mutex_unlock (&priv->lock);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_set_state:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
* @state: the new state
|
|
|
|
*
|
|
|
|
* Tell the media object @media to change to @state.
|
|
|
|
*
|
|
|
|
* Returns: %TRUE on success.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gst_rtsp_session_media_set_state (GstRTSPSessionMedia * media, GstState state)
|
|
|
|
{
|
2012-11-29 10:11:05 +00:00
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
2012-10-25 19:29:58 +00:00
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media), FALSE);
|
|
|
|
|
2012-11-29 10:11:05 +00:00
|
|
|
priv = media->priv;
|
|
|
|
|
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
ret = gst_rtsp_media_set_state (priv->media, state, priv->transports);
|
|
|
|
g_mutex_unlock (&priv->lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_set_rtsp_state:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
* @state: a #GstRTSPState
|
|
|
|
*
|
|
|
|
* Set the RTSP state of @media to @state.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gst_rtsp_session_media_set_rtsp_state (GstRTSPSessionMedia * media,
|
|
|
|
GstRTSPState state)
|
|
|
|
{
|
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
|
|
|
|
|
|
|
g_return_if_fail (GST_IS_RTSP_SESSION_MEDIA (media));
|
|
|
|
|
|
|
|
priv = media->priv;
|
|
|
|
|
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
priv->state = state;
|
|
|
|
g_mutex_unlock (&priv->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gst_rtsp_session_media_set_rtsp_state:
|
|
|
|
* @media: a #GstRTSPSessionMedia
|
|
|
|
*
|
|
|
|
* Get the current RTSP state of @media.
|
|
|
|
*
|
|
|
|
* Returns: the current RTSP state of @media.
|
|
|
|
*/
|
|
|
|
GstRTSPState
|
|
|
|
gst_rtsp_session_media_get_rtsp_state (GstRTSPSessionMedia * media)
|
|
|
|
{
|
|
|
|
GstRTSPSessionMediaPrivate *priv;
|
|
|
|
GstRTSPState ret;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_RTSP_SESSION_MEDIA (media),
|
|
|
|
GST_RTSP_STATE_INVALID);
|
|
|
|
|
|
|
|
priv = media->priv;
|
|
|
|
|
|
|
|
g_mutex_lock (&priv->lock);
|
|
|
|
ret = priv->state;
|
|
|
|
g_mutex_unlock (&priv->lock);
|
2012-10-25 19:29:58 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|