rtspstream: handle both ipv4 and ipv6 clients
[platform/upstream/gstreamer.git] / gst / rtsp-server / rtsp-stream.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 #include <string.h>
21 #include <stdlib.h>
22
23 #include <gio/gio.h>
24
25 #include <gst/app/gstappsrc.h>
26 #include <gst/app/gstappsink.h>
27
28 #include "rtsp-stream.h"
29
30 #define GST_RTSP_STREAM_GET_PRIVATE(obj)  \
31      (G_TYPE_INSTANCE_GET_PRIVATE ((obj), GST_TYPE_RTSP_STREAM, GstRTSPStreamPrivate))
32
33 struct _GstRTSPStreamPrivate
34 {
35   GMutex lock;
36   guint idx;
37   GstPad *srcpad;
38   GstElement *payloader;
39   guint buffer_size;
40   gboolean is_joined;
41
42   /* pads on the rtpbin */
43   GstPad *send_rtp_sink;
44   GstPad *recv_sink[2];
45   GstPad *send_src[2];
46
47   /* the RTPSession object */
48   GObject *session;
49
50   /* sinks used for sending and receiving RTP and RTCP over ipv4, they share
51    * sockets */
52   GstElement *udpsrc_v4[2];
53
54   /* sinks used for sending and receiving RTP and RTCP over ipv6, they share
55    * sockets */
56   GstElement *udpsrc_v6[2];
57
58   GstElement *udpsink[2];
59
60   /* for TCP transport */
61   GstElement *appsrc[2];
62   GstElement *appqueue[2];
63   GstElement *appsink[2];
64
65   GstElement *tee[2];
66   GstElement *funnel[2];
67
68   /* server ports for sending/receiving over ipv4 */
69   GstRTSPRange server_port_v4;
70   GstRTSPAddress *server_addr_v4;
71
72   /* server ports for sending/receiving over ipv6 */
73   GstRTSPRange server_port_v6;
74   GstRTSPAddress *server_addr_v6;
75
76   /* multicast addresses */
77   GstRTSPAddressPool *pool;
78   GstRTSPAddress *addr;
79
80   /* the caps of the stream */
81   gulong caps_sig;
82   GstCaps *caps;
83
84   /* transports we stream to */
85   guint n_active;
86   GList *transports;
87 };
88
89
90 enum
91 {
92   PROP_0,
93   PROP_LAST
94 };
95
96 GST_DEBUG_CATEGORY_STATIC (rtsp_stream_debug);
97 #define GST_CAT_DEFAULT rtsp_stream_debug
98
99 static GQuark ssrc_stream_map_key;
100
101 static void gst_rtsp_stream_finalize (GObject * obj);
102
103 G_DEFINE_TYPE (GstRTSPStream, gst_rtsp_stream, G_TYPE_OBJECT);
104
105 static void
106 gst_rtsp_stream_class_init (GstRTSPStreamClass * klass)
107 {
108   GObjectClass *gobject_class;
109
110   g_type_class_add_private (klass, sizeof (GstRTSPStreamPrivate));
111
112   gobject_class = G_OBJECT_CLASS (klass);
113
114   gobject_class->finalize = gst_rtsp_stream_finalize;
115
116   GST_DEBUG_CATEGORY_INIT (rtsp_stream_debug, "rtspstream", 0, "GstRTSPStream");
117
118   ssrc_stream_map_key = g_quark_from_static_string ("GstRTSPServer.stream");
119 }
120
121 static void
122 gst_rtsp_stream_init (GstRTSPStream * stream)
123 {
124   GstRTSPStreamPrivate *priv = GST_RTSP_STREAM_GET_PRIVATE (stream);
125
126   GST_DEBUG ("new stream %p", stream);
127
128   stream->priv = priv;
129
130   g_mutex_init (&priv->lock);
131 }
132
133 static void
134 gst_rtsp_stream_finalize (GObject * obj)
135 {
136   GstRTSPStream *stream;
137   GstRTSPStreamPrivate *priv;
138
139   stream = GST_RTSP_STREAM (obj);
140   priv = stream->priv;
141
142   GST_DEBUG ("finalize stream %p", stream);
143
144   /* we really need to be unjoined now */
145   g_return_if_fail (!priv->is_joined);
146
147   if (priv->addr)
148     gst_rtsp_address_free (priv->addr);
149   if (priv->server_addr_v4)
150     gst_rtsp_address_free (priv->server_addr_v4);
151   if (priv->server_addr_v6)
152     gst_rtsp_address_free (priv->server_addr_v6);
153   if (priv->pool)
154     g_object_unref (priv->pool);
155   gst_object_unref (priv->payloader);
156   gst_object_unref (priv->srcpad);
157   g_mutex_clear (&priv->lock);
158
159   G_OBJECT_CLASS (gst_rtsp_stream_parent_class)->finalize (obj);
160 }
161
162 /**
163  * gst_rtsp_stream_new:
164  * @idx: an index
165  * @srcpad: a #GstPad
166  * @payloader: a #GstElement
167  *
168  * Create a new media stream with index @idx that handles RTP data on
169  * @srcpad and has a payloader element @payloader.
170  *
171  * Returns: a new #GstRTSPStream
172  */
173 GstRTSPStream *
174 gst_rtsp_stream_new (guint idx, GstElement * payloader, GstPad * srcpad)
175 {
176   GstRTSPStreamPrivate *priv;
177   GstRTSPStream *stream;
178
179   g_return_val_if_fail (GST_IS_ELEMENT (payloader), NULL);
180   g_return_val_if_fail (GST_IS_PAD (srcpad), NULL);
181   g_return_val_if_fail (GST_PAD_IS_SRC (srcpad), NULL);
182
183   stream = g_object_new (GST_TYPE_RTSP_STREAM, NULL);
184   priv = stream->priv;
185   priv->idx = idx;
186   priv->payloader = gst_object_ref (payloader);
187   priv->srcpad = gst_object_ref (srcpad);
188
189   return stream;
190 }
191
192 /**
193  * gst_rtsp_stream_get_index:
194  * @stream: a #GstRTSPStream
195  *
196  * Get the stream index.
197  *
198  * Return: the stream index.
199  */
200 guint
201 gst_rtsp_stream_get_index (GstRTSPStream * stream)
202 {
203   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), -1);
204
205   return stream->priv->idx;
206 }
207
208  /**
209  * gst_rtsp_stream_get_srcpad:
210  * @stream: a #GstRTSPStream
211  *
212  * Get the srcpad associated with @stream.
213  *
214  * Return: the srcpad. Unref after usage.
215  */
216 GstPad *
217 gst_rtsp_stream_get_srcpad (GstRTSPStream * stream)
218 {
219   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
220
221   return gst_object_ref (stream->priv->srcpad);
222 }
223
224 /**
225  * gst_rtsp_stream_set_mtu:
226  * @stream: a #GstRTSPStream
227  * @mtu: a new MTU
228  *
229  * Configure the mtu in the payloader of @stream to @mtu.
230  */
231 void
232 gst_rtsp_stream_set_mtu (GstRTSPStream * stream, guint mtu)
233 {
234   GstRTSPStreamPrivate *priv;
235
236   g_return_if_fail (GST_IS_RTSP_STREAM (stream));
237
238   priv = stream->priv;
239
240   GST_LOG_OBJECT (stream, "set MTU %u", mtu);
241
242   g_object_set (G_OBJECT (priv->payloader), "mtu", mtu, NULL);
243 }
244
245 /**
246  * gst_rtsp_stream_get_mtu:
247  * @stream: a #GstRTSPStream
248  *
249  * Get the configured MTU in the payloader of @stream.
250  *
251  * Returns: the MTU of the payloader.
252  */
253 guint
254 gst_rtsp_stream_get_mtu (GstRTSPStream * stream)
255 {
256   GstRTSPStreamPrivate *priv;
257   guint mtu;
258
259   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), 0);
260
261   priv = stream->priv;
262
263   g_object_get (G_OBJECT (priv->payloader), "mtu", &mtu, NULL);
264
265   return mtu;
266 }
267
268 /**
269  * gst_rtsp_stream_set_address_pool:
270  * @stream: a #GstRTSPStream
271  * @pool: a #GstRTSPAddressPool
272  *
273  * configure @pool to be used as the address pool of @stream.
274  */
275 void
276 gst_rtsp_stream_set_address_pool (GstRTSPStream * stream,
277     GstRTSPAddressPool * pool)
278 {
279   GstRTSPStreamPrivate *priv;
280   GstRTSPAddressPool *old;
281
282   g_return_if_fail (GST_IS_RTSP_STREAM (stream));
283
284   priv = stream->priv;
285
286   GST_LOG_OBJECT (stream, "set address pool %p", pool);
287
288   g_mutex_lock (&priv->lock);
289   if ((old = priv->pool) != pool)
290     priv->pool = pool ? g_object_ref (pool) : NULL;
291   else
292     old = NULL;
293   g_mutex_unlock (&priv->lock);
294
295   if (old)
296     g_object_unref (old);
297 }
298
299 /**
300  * gst_rtsp_stream_get_address_pool:
301  * @stream: a #GstRTSPStream
302  *
303  * Get the #GstRTSPAddressPool used as the address pool of @stream.
304  *
305  * Returns: (transfer full): the #GstRTSPAddressPool of @stream. g_object_unref() after
306  * usage.
307  */
308 GstRTSPAddressPool *
309 gst_rtsp_stream_get_address_pool (GstRTSPStream * stream)
310 {
311   GstRTSPStreamPrivate *priv;
312   GstRTSPAddressPool *result;
313
314   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
315
316   priv = stream->priv;
317
318   g_mutex_lock (&priv->lock);
319   if ((result = priv->pool))
320     g_object_ref (result);
321   g_mutex_unlock (&priv->lock);
322
323   return result;
324 }
325
326 /**
327  * gst_rtsp_stream_get_address:
328  * @stream: a #GstRTSPStream
329  *
330  * Get the multicast address of @stream.
331  *
332  * Returns: the #GstRTSPAddress of @stream or %NULL when no address could be
333  * allocated. gst_rtsp_address_free() after usage.
334  */
335 GstRTSPAddress *
336 gst_rtsp_stream_get_address (GstRTSPStream * stream)
337 {
338   GstRTSPStreamPrivate *priv;
339   GstRTSPAddress *result;
340
341   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
342
343   priv = stream->priv;
344
345   g_mutex_lock (&priv->lock);
346   if (priv->addr == NULL) {
347     if (priv->pool == NULL)
348       goto no_pool;
349
350     priv->addr = gst_rtsp_address_pool_acquire_address (priv->pool,
351         GST_RTSP_ADDRESS_FLAG_EVEN_PORT | GST_RTSP_ADDRESS_FLAG_MULTICAST, 2);
352     if (priv->addr == NULL)
353       goto no_address;
354   }
355   result = gst_rtsp_address_copy (priv->addr);
356   g_mutex_unlock (&priv->lock);
357
358   return result;
359
360   /* ERRORS */
361 no_pool:
362   {
363     GST_ERROR_OBJECT (stream, "no address pool specified");
364     g_mutex_unlock (&priv->lock);
365     return NULL;
366   }
367 no_address:
368   {
369     GST_ERROR_OBJECT (stream, "failed to acquire address from pool");
370     g_mutex_unlock (&priv->lock);
371     return NULL;
372   }
373 }
374
375 /**
376  * gst_rtsp_stream_reserve_address:
377  * @stream: a #GstRTSPStream
378  *
379  * Get a specific multicast address of @stream.
380  *
381  * Returns: the #GstRTSPAddress of @stream or %NULL when no address could be
382  * allocated. gst_rtsp_address_free() after usage.
383  */
384 GstRTSPAddress *
385 gst_rtsp_stream_reserve_address (GstRTSPStream * stream,
386     const gchar * address, guint port, guint n_ports, guint ttl)
387 {
388   GstRTSPStreamPrivate *priv;
389   GstRTSPAddress *result;
390
391   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
392   g_return_val_if_fail (address != NULL, NULL);
393   g_return_val_if_fail (port > 0, NULL);
394   g_return_val_if_fail (n_ports > 0, NULL);
395   g_return_val_if_fail (ttl > 0, NULL);
396
397   priv = stream->priv;
398
399   g_mutex_lock (&priv->lock);
400   if (priv->addr == NULL) {
401     if (priv->pool == NULL)
402       goto no_pool;
403
404     priv->addr = gst_rtsp_address_pool_reserve_address (priv->pool, address,
405         port, n_ports, ttl);
406     if (priv->addr == NULL)
407       goto no_address;
408   } else {
409     if (strcmp (priv->addr->address, address) ||
410         priv->addr->port != port || priv->addr->n_ports != n_ports ||
411         priv->addr->ttl != ttl)
412       goto different_address;
413   }
414   result = gst_rtsp_address_copy (priv->addr);
415   g_mutex_unlock (&priv->lock);
416
417   return result;
418
419   /* ERRORS */
420 no_pool:
421   {
422     GST_ERROR_OBJECT (stream, "no address pool specified");
423     g_mutex_unlock (&priv->lock);
424     return NULL;
425   }
426 no_address:
427   {
428     GST_ERROR_OBJECT (stream, "failed to acquire address %s from pool",
429         address);
430     g_mutex_unlock (&priv->lock);
431     return NULL;
432   }
433 different_address:
434   {
435     GST_ERROR_OBJECT (stream, "address %s is not the same that was already"
436         " reserved", address);
437     g_mutex_unlock (&priv->lock);
438     return NULL;
439   }
440 }
441
442 static gboolean
443 alloc_ports_one_family (GstRTSPAddressPool * pool, gint buffer_size,
444     GSocketFamily family, GstElement * udpsrc_out[2],
445     GstElement * udpsink_out[2], GstRTSPRange * server_port_out,
446     GstRTSPAddress ** server_addr_out)
447 {
448   GstStateChangeReturn ret;
449   GstElement *udpsrc0, *udpsrc1;
450   GstElement *udpsink0, *udpsink1;
451   GSocket *rtp_socket = NULL;
452   GSocket *rtcp_socket;
453   gint tmp_rtp, tmp_rtcp;
454   guint count;
455   gint rtpport, rtcpport;
456   GList *rejected_addresses = NULL;
457   GstRTSPAddress *addr = NULL;
458   GInetAddress *inetaddr = NULL;
459   GSocketAddress *rtp_sockaddr = NULL;
460   GSocketAddress *rtcp_sockaddr = NULL;
461   const gchar *multisink_socket = "socket";
462
463   if (family == G_SOCKET_FAMILY_IPV6) {
464     multisink_socket = "socket-v6";
465   }
466
467   udpsrc0 = NULL;
468   udpsrc1 = NULL;
469   udpsink0 = NULL;
470   udpsink1 = NULL;
471   count = 0;
472
473   /* Start with random port */
474   tmp_rtp = 0;
475
476   rtcp_socket = g_socket_new (family, G_SOCKET_TYPE_DATAGRAM,
477       G_SOCKET_PROTOCOL_UDP, NULL);
478   if (!rtcp_socket)
479     goto no_udp_protocol;
480
481   if (*server_addr_out)
482     gst_rtsp_address_free (*server_addr_out);
483
484   /* try to allocate 2 UDP ports, the RTP port should be an even
485    * number and the RTCP port should be the next (uneven) port */
486 again:
487
488   if (rtp_socket == NULL) {
489     rtp_socket = g_socket_new (family, G_SOCKET_TYPE_DATAGRAM,
490         G_SOCKET_PROTOCOL_UDP, NULL);
491     if (!rtp_socket)
492       goto no_udp_protocol;
493   }
494
495   if (pool && gst_rtsp_address_pool_has_unicast_addresses (pool)) {
496     GstRTSPAddressFlags flags;
497
498     if (addr)
499       rejected_addresses = g_list_prepend (rejected_addresses, addr);
500
501     flags = GST_RTSP_ADDRESS_FLAG_EVEN_PORT | GST_RTSP_ADDRESS_FLAG_UNICAST;
502     if (family == G_SOCKET_FAMILY_IPV6)
503       flags |= GST_RTSP_ADDRESS_FLAG_IPV6;
504     else
505       flags |= GST_RTSP_ADDRESS_FLAG_IPV4;
506
507     addr = gst_rtsp_address_pool_acquire_address (pool, flags, 2);
508
509     if (addr == NULL)
510       goto no_ports;
511
512     tmp_rtp = addr->port;
513
514     g_clear_object (&inetaddr);
515     inetaddr = g_inet_address_new_from_string (addr->address);
516   } else {
517     if (tmp_rtp != 0) {
518       tmp_rtp += 2;
519       if (++count > 20)
520         goto no_ports;
521     }
522
523     if (inetaddr == NULL)
524       inetaddr = g_inet_address_new_any (family);
525   }
526
527   rtp_sockaddr = g_inet_socket_address_new (inetaddr, tmp_rtp);
528   if (!g_socket_bind (rtp_socket, rtp_sockaddr, FALSE, NULL)) {
529     g_object_unref (rtp_sockaddr);
530     goto again;
531   }
532   g_object_unref (rtp_sockaddr);
533
534   rtp_sockaddr = g_socket_get_local_address (rtp_socket, NULL);
535   if (rtp_sockaddr == NULL || !G_IS_INET_SOCKET_ADDRESS (rtp_sockaddr)) {
536     g_clear_object (&rtp_sockaddr);
537     goto socket_error;
538   }
539
540   tmp_rtp =
541       g_inet_socket_address_get_port (G_INET_SOCKET_ADDRESS (rtp_sockaddr));
542   g_object_unref (rtp_sockaddr);
543
544   /* check if port is even */
545   if ((tmp_rtp & 1) != 0) {
546     /* port not even, close and allocate another */
547     tmp_rtp++;
548     g_clear_object (&rtp_socket);
549     goto again;
550   }
551
552   /* set port */
553   tmp_rtcp = tmp_rtp + 1;
554
555   rtcp_sockaddr = g_inet_socket_address_new (inetaddr, tmp_rtcp);
556   if (!g_socket_bind (rtcp_socket, rtcp_sockaddr, FALSE, NULL)) {
557     g_object_unref (rtcp_sockaddr);
558     g_clear_object (&rtp_socket);
559     goto again;
560   }
561   g_object_unref (rtcp_sockaddr);
562
563   g_clear_object (&inetaddr);
564
565   udpsrc0 = gst_element_factory_make ("udpsrc", NULL);
566   udpsrc1 = gst_element_factory_make ("udpsrc", NULL);
567
568   if (udpsrc0 == NULL || udpsrc1 == NULL)
569     goto no_udp_protocol;
570
571   g_object_set (G_OBJECT (udpsrc0), "socket", rtp_socket, NULL);
572   g_object_set (G_OBJECT (udpsrc1), "socket", rtcp_socket, NULL);
573
574   ret = gst_element_set_state (udpsrc0, GST_STATE_PAUSED);
575   if (ret == GST_STATE_CHANGE_FAILURE)
576     goto element_error;
577   ret = gst_element_set_state (udpsrc1, GST_STATE_PAUSED);
578   if (ret == GST_STATE_CHANGE_FAILURE)
579     goto element_error;
580
581   /* all fine, do port check */
582   g_object_get (G_OBJECT (udpsrc0), "port", &rtpport, NULL);
583   g_object_get (G_OBJECT (udpsrc1), "port", &rtcpport, NULL);
584
585   /* this should not happen... */
586   if (rtpport != tmp_rtp || rtcpport != tmp_rtcp)
587     goto port_error;
588
589   if (udpsink_out[0])
590     udpsink0 = udpsink_out[0];
591   else
592     udpsink0 = gst_element_factory_make ("multiudpsink", NULL);
593
594   if (!udpsink0)
595     goto no_udp_protocol;
596
597   g_object_set (G_OBJECT (udpsink0), "close-socket", FALSE, NULL);
598   g_object_set (G_OBJECT (udpsink0), multisink_socket, rtp_socket, NULL);
599
600   if (udpsink_out[1])
601     udpsink1 = udpsink_out[1];
602   else
603     udpsink1 = gst_element_factory_make ("multiudpsink", NULL);
604
605   if (!udpsink1)
606     goto no_udp_protocol;
607
608   g_object_set (G_OBJECT (udpsink0), "send-duplicates", FALSE, NULL);
609   g_object_set (G_OBJECT (udpsink1), "send-duplicates", FALSE, NULL);
610   g_object_set (G_OBJECT (udpsink0), "buffer-size", buffer_size, NULL);
611
612   g_object_set (G_OBJECT (udpsink1), "close-socket", FALSE, NULL);
613   g_object_set (G_OBJECT (udpsink1), multisink_socket, rtcp_socket, NULL);
614   g_object_set (G_OBJECT (udpsink1), "sync", FALSE, NULL);
615   g_object_set (G_OBJECT (udpsink1), "async", FALSE, NULL);
616   g_object_set (G_OBJECT (udpsink0), "auto-multicast", FALSE, NULL);
617   g_object_set (G_OBJECT (udpsink0), "loop", FALSE, NULL);
618   g_object_set (G_OBJECT (udpsink1), "auto-multicast", FALSE, NULL);
619   g_object_set (G_OBJECT (udpsink1), "loop", FALSE, NULL);
620
621   /* we keep these elements, we will further configure them when the
622    * client told us to really use the UDP ports. */
623   udpsrc_out[0] = udpsrc0;
624   udpsrc_out[1] = udpsrc1;
625   udpsink_out[0] = udpsink0;
626   udpsink_out[1] = udpsink1;
627   server_port_out->min = rtpport;
628   server_port_out->max = rtcpport;
629
630   *server_addr_out = addr;
631   g_list_free_full (rejected_addresses, (GDestroyNotify) gst_rtsp_address_free);
632
633   g_object_unref (rtp_socket);
634   g_object_unref (rtcp_socket);
635
636   return TRUE;
637
638   /* ERRORS */
639 no_udp_protocol:
640   {
641     goto cleanup;
642   }
643 no_ports:
644   {
645     goto cleanup;
646   }
647 port_error:
648   {
649     goto cleanup;
650   }
651 socket_error:
652   {
653     goto cleanup;
654   }
655 element_error:
656   {
657     goto cleanup;
658   }
659 cleanup:
660   {
661     if (udpsrc0) {
662       gst_element_set_state (udpsrc0, GST_STATE_NULL);
663       gst_object_unref (udpsrc0);
664     }
665     if (udpsrc1) {
666       gst_element_set_state (udpsrc1, GST_STATE_NULL);
667       gst_object_unref (udpsrc1);
668     }
669     if (udpsink0) {
670       gst_element_set_state (udpsink0, GST_STATE_NULL);
671       gst_object_unref (udpsink0);
672     }
673     if (udpsink1) {
674       gst_element_set_state (udpsink1, GST_STATE_NULL);
675       gst_object_unref (udpsink1);
676     }
677     if (inetaddr)
678       g_object_unref (inetaddr);
679     g_list_free_full (rejected_addresses,
680         (GDestroyNotify) gst_rtsp_address_free);
681     if (addr)
682       gst_rtsp_address_free (addr);
683     if (rtp_socket)
684       g_object_unref (rtp_socket);
685     if (rtcp_socket)
686       g_object_unref (rtcp_socket);
687     return FALSE;
688   }
689 }
690
691 /* must be called with lock */
692 static gboolean
693 alloc_ports (GstRTSPStream * stream)
694 {
695   GstRTSPStreamPrivate *priv = stream->priv;
696
697   return alloc_ports_one_family (priv->pool, priv->buffer_size,
698       G_SOCKET_FAMILY_IPV4, priv->udpsrc_v4, priv->udpsink,
699       &priv->server_port_v4, &priv->server_addr_v4) &&
700       alloc_ports_one_family (priv->pool, priv->buffer_size,
701       G_SOCKET_FAMILY_IPV6, priv->udpsrc_v6, priv->udpsink,
702       &priv->server_port_v6, &priv->server_addr_v6);
703 }
704
705 /**
706  * gst_rtsp_stream_get_server_port:
707  * @stream: a #GstRTSPStream
708  * @server_port: (out): result server port
709  *
710  * Fill @server_port with the port pair used by the server. This function can
711  * only be called when @stream has been joined.
712  */
713 void
714 gst_rtsp_stream_get_server_port (GstRTSPStream * stream,
715     GstRTSPRange * server_port, GSocketFamily family)
716 {
717   GstRTSPStreamPrivate *priv;
718
719   g_return_if_fail (GST_IS_RTSP_STREAM (stream));
720   priv = stream->priv;
721   g_return_if_fail (priv->is_joined);
722
723   g_mutex_lock (&priv->lock);
724   if (family == G_SOCKET_FAMILY_IPV4) {
725     if (server_port)
726       *server_port = priv->server_port_v4;
727   } else {
728     if (server_port)
729       *server_port = priv->server_port_v6;
730   }
731   g_mutex_unlock (&priv->lock);
732 }
733
734 /**
735  * gst_rtsp_stream_get_ssrc:
736  * @stream: a #GstRTSPStream
737  * @ssrc: (out): result ssrc
738  *
739  * Get the SSRC used by the RTP session of this stream. This function can only
740  * be called when @stream has been joined.
741  */
742 void
743 gst_rtsp_stream_get_ssrc (GstRTSPStream * stream, guint * ssrc)
744 {
745   GstRTSPStreamPrivate *priv;
746
747   g_return_if_fail (GST_IS_RTSP_STREAM (stream));
748   priv = stream->priv;
749   g_return_if_fail (priv->is_joined);
750
751   g_mutex_lock (&priv->lock);
752   if (ssrc && priv->session)
753     g_object_get (priv->session, "internal-ssrc", ssrc, NULL);
754   g_mutex_unlock (&priv->lock);
755 }
756
757 /* executed from streaming thread */
758 static void
759 caps_notify (GstPad * pad, GParamSpec * unused, GstRTSPStream * stream)
760 {
761   GstRTSPStreamPrivate *priv = stream->priv;
762   GstCaps *newcaps, *oldcaps;
763
764   newcaps = gst_pad_get_current_caps (pad);
765
766   GST_INFO ("stream %p received caps %p, %" GST_PTR_FORMAT, stream, newcaps,
767       newcaps);
768
769   g_mutex_lock (&priv->lock);
770   oldcaps = priv->caps;
771   priv->caps = newcaps;
772   g_mutex_unlock (&priv->lock);
773
774   if (oldcaps)
775     gst_caps_unref (oldcaps);
776 }
777
778 static void
779 dump_structure (const GstStructure * s)
780 {
781   gchar *sstr;
782
783   sstr = gst_structure_to_string (s);
784   GST_INFO ("structure: %s", sstr);
785   g_free (sstr);
786 }
787
788 static GstRTSPStreamTransport *
789 find_transport (GstRTSPStream * stream, const gchar * rtcp_from)
790 {
791   GstRTSPStreamPrivate *priv = stream->priv;
792   GList *walk;
793   GstRTSPStreamTransport *result = NULL;
794   const gchar *tmp;
795   gchar *dest;
796   guint port;
797
798   if (rtcp_from == NULL)
799     return NULL;
800
801   tmp = g_strrstr (rtcp_from, ":");
802   if (tmp == NULL)
803     return NULL;
804
805   port = atoi (tmp + 1);
806   dest = g_strndup (rtcp_from, tmp - rtcp_from);
807
808   g_mutex_lock (&priv->lock);
809   GST_INFO ("finding %s:%d in %d transports", dest, port,
810       g_list_length (priv->transports));
811
812   for (walk = priv->transports; walk; walk = g_list_next (walk)) {
813     GstRTSPStreamTransport *trans = walk->data;
814     const GstRTSPTransport *tr;
815     gint min, max;
816
817     tr = gst_rtsp_stream_transport_get_transport (trans);
818
819     min = tr->client_port.min;
820     max = tr->client_port.max;
821
822     if ((strcmp (tr->destination, dest) == 0) && (min == port || max == port)) {
823       result = trans;
824       break;
825     }
826   }
827   if (result)
828     g_object_ref (result);
829   g_mutex_unlock (&priv->lock);
830
831   g_free (dest);
832
833   return result;
834 }
835
836 static GstRTSPStreamTransport *
837 check_transport (GObject * source, GstRTSPStream * stream)
838 {
839   GstStructure *stats;
840   GstRTSPStreamTransport *trans;
841
842   /* see if we have a stream to match with the origin of the RTCP packet */
843   trans = g_object_get_qdata (source, ssrc_stream_map_key);
844   if (trans == NULL) {
845     g_object_get (source, "stats", &stats, NULL);
846     if (stats) {
847       const gchar *rtcp_from;
848
849       dump_structure (stats);
850
851       rtcp_from = gst_structure_get_string (stats, "rtcp-from");
852       if ((trans = find_transport (stream, rtcp_from))) {
853         GST_INFO ("%p: found transport %p for source  %p", stream, trans,
854             source);
855         g_object_set_qdata_full (source, ssrc_stream_map_key, trans,
856             g_object_unref);
857       }
858       gst_structure_free (stats);
859     }
860   }
861   return trans;
862 }
863
864
865 static void
866 on_new_ssrc (GObject * session, GObject * source, GstRTSPStream * stream)
867 {
868   GstRTSPStreamTransport *trans;
869
870   GST_INFO ("%p: new source %p", stream, source);
871
872   trans = check_transport (source, stream);
873
874   if (trans)
875     GST_INFO ("%p: source %p for transport %p", stream, source, trans);
876 }
877
878 static void
879 on_ssrc_sdes (GObject * session, GObject * source, GstRTSPStream * stream)
880 {
881   GST_INFO ("%p: new SDES %p", stream, source);
882 }
883
884 static void
885 on_ssrc_active (GObject * session, GObject * source, GstRTSPStream * stream)
886 {
887   GstRTSPStreamTransport *trans;
888
889   trans = check_transport (source, stream);
890
891   if (trans) {
892     GST_INFO ("%p: source %p in transport %p is active", stream, source, trans);
893     gst_rtsp_stream_transport_keep_alive (trans);
894   }
895 #ifdef DUMP_STATS
896   {
897     GstStructure *stats;
898     g_object_get (source, "stats", &stats, NULL);
899     if (stats) {
900       dump_structure (stats);
901       gst_structure_free (stats);
902     }
903   }
904 #endif
905 }
906
907 static void
908 on_bye_ssrc (GObject * session, GObject * source, GstRTSPStream * stream)
909 {
910   GST_INFO ("%p: source %p bye", stream, source);
911 }
912
913 static void
914 on_bye_timeout (GObject * session, GObject * source, GstRTSPStream * stream)
915 {
916   GstRTSPStreamTransport *trans;
917
918   GST_INFO ("%p: source %p bye timeout", stream, source);
919
920   if ((trans = g_object_get_qdata (source, ssrc_stream_map_key))) {
921     gst_rtsp_stream_transport_set_timed_out (trans, TRUE);
922     g_object_set_qdata (source, ssrc_stream_map_key, NULL);
923   }
924 }
925
926 static void
927 on_timeout (GObject * session, GObject * source, GstRTSPStream * stream)
928 {
929   GstRTSPStreamTransport *trans;
930
931   GST_INFO ("%p: source %p timeout", stream, source);
932
933   if ((trans = g_object_get_qdata (source, ssrc_stream_map_key))) {
934     gst_rtsp_stream_transport_set_timed_out (trans, TRUE);
935     g_object_set_qdata (source, ssrc_stream_map_key, NULL);
936   }
937 }
938
939 static GstFlowReturn
940 handle_new_sample (GstAppSink * sink, gpointer user_data)
941 {
942   GstRTSPStreamPrivate *priv;
943   GList *walk;
944   GstSample *sample;
945   GstBuffer *buffer;
946   GstRTSPStream *stream;
947
948   sample = gst_app_sink_pull_sample (sink);
949   if (!sample)
950     return GST_FLOW_OK;
951
952   stream = (GstRTSPStream *) user_data;
953   priv = stream->priv;
954   buffer = gst_sample_get_buffer (sample);
955
956   g_mutex_lock (&priv->lock);
957   for (walk = priv->transports; walk; walk = g_list_next (walk)) {
958     GstRTSPStreamTransport *tr = (GstRTSPStreamTransport *) walk->data;
959
960     if (GST_ELEMENT_CAST (sink) == priv->appsink[0]) {
961       gst_rtsp_stream_transport_send_rtp (tr, buffer);
962     } else {
963       gst_rtsp_stream_transport_send_rtcp (tr, buffer);
964     }
965   }
966   g_mutex_unlock (&priv->lock);
967
968   gst_sample_unref (sample);
969
970   return GST_FLOW_OK;
971 }
972
973 static GstAppSinkCallbacks sink_cb = {
974   NULL,                         /* not interested in EOS */
975   NULL,                         /* not interested in preroll samples */
976   handle_new_sample,
977 };
978
979 /**
980  * gst_rtsp_stream_join_bin:
981  * @stream: a #GstRTSPStream
982  * @bin: a #GstBin to join
983  * @rtpbin: a rtpbin element in @bin
984  * @state: the target state of the new elements
985  *
986  * Join the #Gstbin @bin that contains the element @rtpbin.
987  *
988  * @stream will link to @rtpbin, which must be inside @bin. The elements
989  * added to @bin will be set to the state given in @state.
990  *
991  * Returns: %TRUE on success.
992  */
993 gboolean
994 gst_rtsp_stream_join_bin (GstRTSPStream * stream, GstBin * bin,
995     GstElement * rtpbin, GstState state)
996 {
997   GstRTSPStreamPrivate *priv;
998   gint i, idx;
999   gchar *name;
1000   GstPad *pad, *teepad, *queuepad, *selpad;
1001   GstPadLinkReturn ret;
1002
1003   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), FALSE);
1004   g_return_val_if_fail (GST_IS_BIN (bin), FALSE);
1005   g_return_val_if_fail (GST_IS_ELEMENT (rtpbin), FALSE);
1006
1007   priv = stream->priv;
1008
1009   g_mutex_lock (&priv->lock);
1010   if (priv->is_joined)
1011     goto was_joined;
1012
1013   /* create a session with the same index as the stream */
1014   idx = priv->idx;
1015
1016   GST_INFO ("stream %p joining bin as session %d", stream, idx);
1017
1018   if (!alloc_ports (stream))
1019     goto no_ports;
1020
1021   /* get a pad for sending RTP */
1022   name = g_strdup_printf ("send_rtp_sink_%u", idx);
1023   priv->send_rtp_sink = gst_element_get_request_pad (rtpbin, name);
1024   g_free (name);
1025   /* link the RTP pad to the session manager, it should not really fail unless
1026    * this is not really an RTP pad */
1027   ret = gst_pad_link (priv->srcpad, priv->send_rtp_sink);
1028   if (ret != GST_PAD_LINK_OK)
1029     goto link_failed;
1030
1031   /* get pads from the RTP session element for sending and receiving
1032    * RTP/RTCP*/
1033   name = g_strdup_printf ("send_rtp_src_%u", idx);
1034   priv->send_src[0] = gst_element_get_static_pad (rtpbin, name);
1035   g_free (name);
1036   name = g_strdup_printf ("send_rtcp_src_%u", idx);
1037   priv->send_src[1] = gst_element_get_request_pad (rtpbin, name);
1038   g_free (name);
1039   name = g_strdup_printf ("recv_rtp_sink_%u", idx);
1040   priv->recv_sink[0] = gst_element_get_request_pad (rtpbin, name);
1041   g_free (name);
1042   name = g_strdup_printf ("recv_rtcp_sink_%u", idx);
1043   priv->recv_sink[1] = gst_element_get_request_pad (rtpbin, name);
1044   g_free (name);
1045
1046   /* get the session */
1047   g_signal_emit_by_name (rtpbin, "get-internal-session", idx, &priv->session);
1048
1049   g_signal_connect (priv->session, "on-new-ssrc", (GCallback) on_new_ssrc,
1050       stream);
1051   g_signal_connect (priv->session, "on-ssrc-sdes", (GCallback) on_ssrc_sdes,
1052       stream);
1053   g_signal_connect (priv->session, "on-ssrc-active",
1054       (GCallback) on_ssrc_active, stream);
1055   g_signal_connect (priv->session, "on-bye-ssrc", (GCallback) on_bye_ssrc,
1056       stream);
1057   g_signal_connect (priv->session, "on-bye-timeout",
1058       (GCallback) on_bye_timeout, stream);
1059   g_signal_connect (priv->session, "on-timeout", (GCallback) on_timeout,
1060       stream);
1061
1062   for (i = 0; i < 2; i++) {
1063     /* For the sender we create this bit of pipeline for both
1064      * RTP and RTCP. Sync and preroll are enabled on udpsink so
1065      * we need to add a queue before appsink to make the pipeline
1066      * not block. For the TCP case, we want to pump data to the
1067      * client as fast as possible anyway.
1068      *
1069      * .--------.      .-----.    .---------.
1070      * | rtpbin |      | tee |    | udpsink |
1071      * |       send->sink   src->sink       |
1072      * '--------'      |     |    '---------'
1073      *                 |     |    .---------.    .---------.
1074      *                 |     |    |  queue  |    | appsink |
1075      *                 |    src->sink      src->sink       |
1076      *                 '-----'    '---------'    '---------'
1077      */
1078     /* make tee for RTP/RTCP */
1079     priv->tee[i] = gst_element_factory_make ("tee", NULL);
1080     gst_bin_add (bin, priv->tee[i]);
1081
1082     /* and link to rtpbin send pad */
1083     pad = gst_element_get_static_pad (priv->tee[i], "sink");
1084     gst_pad_link (priv->send_src[i], pad);
1085     gst_object_unref (pad);
1086
1087     /* add udpsink */
1088     gst_bin_add (bin, priv->udpsink[i]);
1089
1090     /* link tee to udpsink */
1091     teepad = gst_element_get_request_pad (priv->tee[i], "src_%u");
1092     pad = gst_element_get_static_pad (priv->udpsink[i], "sink");
1093     gst_pad_link (teepad, pad);
1094     gst_object_unref (pad);
1095     gst_object_unref (teepad);
1096
1097     /* make queue */
1098     priv->appqueue[i] = gst_element_factory_make ("queue", NULL);
1099     gst_bin_add (bin, priv->appqueue[i]);
1100     /* and link to tee */
1101     teepad = gst_element_get_request_pad (priv->tee[i], "src_%u");
1102     pad = gst_element_get_static_pad (priv->appqueue[i], "sink");
1103     gst_pad_link (teepad, pad);
1104     gst_object_unref (pad);
1105     gst_object_unref (teepad);
1106
1107     /* make appsink */
1108     priv->appsink[i] = gst_element_factory_make ("appsink", NULL);
1109     g_object_set (priv->appsink[i], "async", FALSE, "sync", FALSE, NULL);
1110     g_object_set (priv->appsink[i], "emit-signals", FALSE, NULL);
1111     gst_bin_add (bin, priv->appsink[i]);
1112     gst_app_sink_set_callbacks (GST_APP_SINK_CAST (priv->appsink[i]),
1113         &sink_cb, stream, NULL);
1114     /* and link to queue */
1115     queuepad = gst_element_get_static_pad (priv->appqueue[i], "src");
1116     pad = gst_element_get_static_pad (priv->appsink[i], "sink");
1117     gst_pad_link (queuepad, pad);
1118     gst_object_unref (pad);
1119     gst_object_unref (queuepad);
1120
1121     /* For the receiver we create this bit of pipeline for both
1122      * RTP and RTCP. We receive RTP/RTCP on appsrc and udpsrc
1123      * and it is all funneled into the rtpbin receive pad.
1124      *
1125      * .--------.     .--------.    .--------.
1126      * | udpsrc |     | funnel |    | rtpbin |
1127      * |       src->sink      src->sink      |
1128      * '--------'     |        |    '--------'
1129      * .--------.     |        |
1130      * | appsrc |     |        |
1131      * |       src->sink       |
1132      * '--------'     '--------'
1133      */
1134     /* make funnel for the RTP/RTCP receivers */
1135     priv->funnel[i] = gst_element_factory_make ("funnel", NULL);
1136     gst_bin_add (bin, priv->funnel[i]);
1137
1138     pad = gst_element_get_static_pad (priv->funnel[i], "src");
1139     gst_pad_link (pad, priv->recv_sink[i]);
1140     gst_object_unref (pad);
1141
1142     /* we set and keep these to playing so that they don't cause NO_PREROLL return
1143      * values */
1144     gst_element_set_state (priv->udpsrc_v4[i], GST_STATE_PLAYING);
1145     gst_element_set_state (priv->udpsrc_v6[i], GST_STATE_PLAYING);
1146     gst_element_set_locked_state (priv->udpsrc_v4[i], TRUE);
1147     gst_element_set_locked_state (priv->udpsrc_v6[i], TRUE);
1148     /* add udpsrc */
1149     gst_bin_add (bin, priv->udpsrc_v4[i]);
1150     gst_bin_add (bin, priv->udpsrc_v6[i]);
1151     /* and link to the funnel v4 */
1152     selpad = gst_element_get_request_pad (priv->funnel[i], "sink_%u");
1153     pad = gst_element_get_static_pad (priv->udpsrc_v4[i], "src");
1154     gst_pad_link (pad, selpad);
1155     gst_object_unref (pad);
1156     gst_object_unref (selpad);
1157
1158     /* and link to the funnel v6 */
1159     selpad = gst_element_get_request_pad (priv->funnel[i], "sink_%u");
1160     pad = gst_element_get_static_pad (priv->udpsrc_v6[i], "src");
1161     gst_pad_link (pad, selpad);
1162     gst_object_unref (pad);
1163     gst_object_unref (selpad);
1164
1165     /* make and add appsrc */
1166     priv->appsrc[i] = gst_element_factory_make ("appsrc", NULL);
1167     gst_bin_add (bin, priv->appsrc[i]);
1168     /* and link to the funnel */
1169     selpad = gst_element_get_request_pad (priv->funnel[i], "sink_%u");
1170     pad = gst_element_get_static_pad (priv->appsrc[i], "src");
1171     gst_pad_link (pad, selpad);
1172     gst_object_unref (pad);
1173     gst_object_unref (selpad);
1174
1175     /* check if we need to set to a special state */
1176     if (state != GST_STATE_NULL) {
1177       gst_element_set_state (priv->udpsink[i], state);
1178       gst_element_set_state (priv->appsink[i], state);
1179       gst_element_set_state (priv->appqueue[i], state);
1180       gst_element_set_state (priv->tee[i], state);
1181       gst_element_set_state (priv->funnel[i], state);
1182       gst_element_set_state (priv->appsrc[i], state);
1183     }
1184   }
1185
1186   /* be notified of caps changes */
1187   priv->caps_sig = g_signal_connect (priv->send_rtp_sink, "notify::caps",
1188       (GCallback) caps_notify, stream);
1189
1190   priv->is_joined = TRUE;
1191   g_mutex_unlock (&priv->lock);
1192
1193   return TRUE;
1194
1195   /* ERRORS */
1196 was_joined:
1197   {
1198     g_mutex_unlock (&priv->lock);
1199     return TRUE;
1200   }
1201 no_ports:
1202   {
1203     g_mutex_unlock (&priv->lock);
1204     GST_WARNING ("failed to allocate ports %d", idx);
1205     return FALSE;
1206   }
1207 link_failed:
1208   {
1209     GST_WARNING ("failed to link stream %d", idx);
1210     gst_object_unref (priv->send_rtp_sink);
1211     priv->send_rtp_sink = NULL;
1212     g_mutex_unlock (&priv->lock);
1213     return FALSE;
1214   }
1215 }
1216
1217 /**
1218  * gst_rtsp_stream_leave_bin:
1219  * @stream: a #GstRTSPStream
1220  * @bin: a #GstBin
1221  * @rtpbin: a rtpbin #GstElement
1222  *
1223  * Remove the elements of @stream from @bin.
1224  *
1225  * Return: %TRUE on success.
1226  */
1227 gboolean
1228 gst_rtsp_stream_leave_bin (GstRTSPStream * stream, GstBin * bin,
1229     GstElement * rtpbin)
1230 {
1231   GstRTSPStreamPrivate *priv;
1232   gint i;
1233
1234   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), FALSE);
1235   g_return_val_if_fail (GST_IS_BIN (bin), FALSE);
1236   g_return_val_if_fail (GST_IS_ELEMENT (rtpbin), FALSE);
1237
1238   priv = stream->priv;
1239
1240   g_mutex_lock (&priv->lock);
1241   if (!priv->is_joined)
1242     goto was_not_joined;
1243
1244   /* all transports must be removed by now */
1245   g_return_val_if_fail (priv->transports == NULL, FALSE);
1246
1247   GST_INFO ("stream %p leaving bin", stream);
1248
1249   gst_pad_unlink (priv->srcpad, priv->send_rtp_sink);
1250   g_signal_handler_disconnect (priv->send_rtp_sink, priv->caps_sig);
1251   gst_element_release_request_pad (rtpbin, priv->send_rtp_sink);
1252   gst_object_unref (priv->send_rtp_sink);
1253   priv->send_rtp_sink = NULL;
1254
1255   for (i = 0; i < 2; i++) {
1256     gst_element_set_state (priv->udpsink[i], GST_STATE_NULL);
1257     gst_element_set_state (priv->appsink[i], GST_STATE_NULL);
1258     gst_element_set_state (priv->appqueue[i], GST_STATE_NULL);
1259     gst_element_set_state (priv->tee[i], GST_STATE_NULL);
1260     gst_element_set_state (priv->funnel[i], GST_STATE_NULL);
1261     gst_element_set_state (priv->appsrc[i], GST_STATE_NULL);
1262     /* and set udpsrc to NULL now before removing */
1263     gst_element_set_locked_state (priv->udpsrc_v4[i], FALSE);
1264     gst_element_set_state (priv->udpsrc_v4[i], GST_STATE_NULL);
1265     gst_element_set_locked_state (priv->udpsrc_v6[i], FALSE);
1266     gst_element_set_state (priv->udpsrc_v6[i], GST_STATE_NULL);
1267
1268     /* removing them should also nicely release the request
1269      * pads when they finalize */
1270     gst_bin_remove (bin, priv->udpsrc_v4[i]);
1271     gst_bin_remove (bin, priv->udpsrc_v6[i]);
1272     gst_bin_remove (bin, priv->udpsink[i]);
1273     gst_bin_remove (bin, priv->appsrc[i]);
1274     gst_bin_remove (bin, priv->appsink[i]);
1275     gst_bin_remove (bin, priv->appqueue[i]);
1276     gst_bin_remove (bin, priv->tee[i]);
1277     gst_bin_remove (bin, priv->funnel[i]);
1278
1279     gst_element_release_request_pad (rtpbin, priv->recv_sink[i]);
1280     gst_object_unref (priv->recv_sink[i]);
1281     priv->recv_sink[i] = NULL;
1282
1283     priv->udpsrc_v4[i] = NULL;
1284     priv->udpsrc_v6[i] = NULL;
1285     priv->udpsink[i] = NULL;
1286     priv->appsrc[i] = NULL;
1287     priv->appsink[i] = NULL;
1288     priv->appqueue[i] = NULL;
1289     priv->tee[i] = NULL;
1290     priv->funnel[i] = NULL;
1291   }
1292   gst_object_unref (priv->send_src[0]);
1293   priv->send_src[0] = NULL;
1294
1295   gst_element_release_request_pad (rtpbin, priv->send_src[1]);
1296   gst_object_unref (priv->send_src[1]);
1297   priv->send_src[1] = NULL;
1298
1299   g_object_unref (priv->session);
1300   priv->session = NULL;
1301   if (priv->caps)
1302     gst_caps_unref (priv->caps);
1303   priv->caps = NULL;
1304
1305   priv->is_joined = FALSE;
1306   g_mutex_unlock (&priv->lock);
1307
1308   return TRUE;
1309
1310 was_not_joined:
1311   {
1312     return TRUE;
1313   }
1314 }
1315
1316 /**
1317  * gst_rtsp_stream_get_rtpinfo:
1318  * @stream: a #GstRTSPStream
1319  * @rtptime: result RTP timestamp
1320  * @seq: result RTP seqnum
1321  *
1322  * Retrieve the current rtptime and seq. This is used to
1323  * construct a RTPInfo reply header.
1324  *
1325  * Returns: %TRUE when rtptime and seq could be determined.
1326  */
1327 gboolean
1328 gst_rtsp_stream_get_rtpinfo (GstRTSPStream * stream,
1329     guint * rtptime, guint * seq)
1330 {
1331   GstRTSPStreamPrivate *priv;
1332   GObjectClass *payobjclass;
1333
1334   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), FALSE);
1335   g_return_val_if_fail (rtptime != NULL, FALSE);
1336   g_return_val_if_fail (seq != NULL, FALSE);
1337
1338   priv = stream->priv;
1339
1340   payobjclass = G_OBJECT_GET_CLASS (priv->payloader);
1341
1342   if (!g_object_class_find_property (payobjclass, "seqnum") ||
1343       !g_object_class_find_property (payobjclass, "timestamp"))
1344     return FALSE;
1345
1346   g_object_get (priv->payloader, "seqnum", seq, "timestamp", rtptime, NULL);
1347
1348   return TRUE;
1349 }
1350
1351 /**
1352  * gst_rtsp_stream_get_caps:
1353  * @stream: a #GstRTSPStream
1354  *
1355  * Retrieve the current caps of @stream.
1356  *
1357  * Returns: (transfer full): the #GstCaps of @stream. use gst_caps_unref()
1358  *    after usage.
1359  */
1360 GstCaps *
1361 gst_rtsp_stream_get_caps (GstRTSPStream * stream)
1362 {
1363   GstRTSPStreamPrivate *priv;
1364   GstCaps *result;
1365
1366   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), NULL);
1367
1368   priv = stream->priv;
1369
1370   g_mutex_lock (&priv->lock);
1371   if ((result = priv->caps))
1372     gst_caps_ref (result);
1373   g_mutex_unlock (&priv->lock);
1374
1375   return result;
1376 }
1377
1378 /**
1379  * gst_rtsp_stream_recv_rtp:
1380  * @stream: a #GstRTSPStream
1381  * @buffer: (transfer full): a #GstBuffer
1382  *
1383  * Handle an RTP buffer for the stream. This method is usually called when a
1384  * message has been received from a client using the TCP transport.
1385  *
1386  * This function takes ownership of @buffer.
1387  *
1388  * Returns: a GstFlowReturn.
1389  */
1390 GstFlowReturn
1391 gst_rtsp_stream_recv_rtp (GstRTSPStream * stream, GstBuffer * buffer)
1392 {
1393   GstRTSPStreamPrivate *priv;
1394   GstFlowReturn ret;
1395   GstElement *element;
1396
1397   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), GST_FLOW_ERROR);
1398   priv = stream->priv;
1399   g_return_val_if_fail (GST_IS_BUFFER (buffer), GST_FLOW_ERROR);
1400   g_return_val_if_fail (priv->is_joined, FALSE);
1401
1402   g_mutex_lock (&priv->lock);
1403   element = gst_object_ref (priv->appsrc[0]);
1404   g_mutex_unlock (&priv->lock);
1405
1406   ret = gst_app_src_push_buffer (GST_APP_SRC_CAST (element), buffer);
1407
1408   gst_object_unref (element);
1409
1410   return ret;
1411 }
1412
1413 /**
1414  * gst_rtsp_stream_recv_rtcp:
1415  * @stream: a #GstRTSPStream
1416  * @buffer: (transfer full): a #GstBuffer
1417  *
1418  * Handle an RTCP buffer for the stream. This method is usually called when a
1419  * message has been received from a client using the TCP transport.
1420  *
1421  * This function takes ownership of @buffer.
1422  *
1423  * Returns: a GstFlowReturn.
1424  */
1425 GstFlowReturn
1426 gst_rtsp_stream_recv_rtcp (GstRTSPStream * stream, GstBuffer * buffer)
1427 {
1428   GstRTSPStreamPrivate *priv;
1429   GstFlowReturn ret;
1430   GstElement *element;
1431
1432   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), GST_FLOW_ERROR);
1433   priv = stream->priv;
1434   g_return_val_if_fail (GST_IS_BUFFER (buffer), GST_FLOW_ERROR);
1435   g_return_val_if_fail (priv->is_joined, FALSE);
1436
1437   g_mutex_lock (&priv->lock);
1438   element = gst_object_ref (priv->appsrc[1]);
1439   g_mutex_unlock (&priv->lock);
1440
1441   ret = gst_app_src_push_buffer (GST_APP_SRC_CAST (element), buffer);
1442
1443   gst_object_unref (element);
1444
1445   return ret;
1446 }
1447
1448 /* must be called with lock */
1449 static gboolean
1450 update_transport (GstRTSPStream * stream, GstRTSPStreamTransport * trans,
1451     gboolean add)
1452 {
1453   GstRTSPStreamPrivate *priv = stream->priv;
1454   const GstRTSPTransport *tr;
1455
1456   tr = gst_rtsp_stream_transport_get_transport (trans);
1457
1458   switch (tr->lower_transport) {
1459     case GST_RTSP_LOWER_TRANS_UDP:
1460     case GST_RTSP_LOWER_TRANS_UDP_MCAST:
1461     {
1462       gchar *dest;
1463       gint min, max;
1464       guint ttl = 0;
1465
1466       dest = tr->destination;
1467       if (tr->lower_transport == GST_RTSP_LOWER_TRANS_UDP_MCAST) {
1468         min = tr->port.min;
1469         max = tr->port.max;
1470         ttl = tr->ttl;
1471       } else {
1472         min = tr->client_port.min;
1473         max = tr->client_port.max;
1474       }
1475
1476       if (add) {
1477         GST_INFO ("adding %s:%d-%d", dest, min, max);
1478         g_signal_emit_by_name (priv->udpsink[0], "add", dest, min, NULL);
1479         g_signal_emit_by_name (priv->udpsink[1], "add", dest, max, NULL);
1480         if (ttl > 0) {
1481           GST_INFO ("setting ttl-mc %d", ttl);
1482           g_object_set (G_OBJECT (priv->udpsink[0]), "ttl-mc", ttl, NULL);
1483           g_object_set (G_OBJECT (priv->udpsink[1]), "ttl-mc", ttl, NULL);
1484         }
1485         priv->transports = g_list_prepend (priv->transports, trans);
1486       } else {
1487         GST_INFO ("removing %s:%d-%d", dest, min, max);
1488         g_signal_emit_by_name (priv->udpsink[0], "remove", dest, min, NULL);
1489         g_signal_emit_by_name (priv->udpsink[1], "remove", dest, max, NULL);
1490         priv->transports = g_list_remove (priv->transports, trans);
1491       }
1492       break;
1493     }
1494     case GST_RTSP_LOWER_TRANS_TCP:
1495       if (add) {
1496         GST_INFO ("adding TCP %s", tr->destination);
1497         priv->transports = g_list_prepend (priv->transports, trans);
1498       } else {
1499         GST_INFO ("removing TCP %s", tr->destination);
1500         priv->transports = g_list_remove (priv->transports, trans);
1501       }
1502       break;
1503     default:
1504       goto unknown_transport;
1505   }
1506   return TRUE;
1507
1508   /* ERRORS */
1509 unknown_transport:
1510   {
1511     GST_INFO ("Unknown transport %d", tr->lower_transport);
1512     return FALSE;
1513   }
1514 }
1515
1516
1517 /**
1518  * gst_rtsp_stream_add_transport:
1519  * @stream: a #GstRTSPStream
1520  * @trans: a #GstRTSPStreamTransport
1521  *
1522  * Add the transport in @trans to @stream. The media of @stream will
1523  * then also be send to the values configured in @trans.
1524  *
1525  * @stream must be joined to a bin.
1526  *
1527  * @trans must contain a valid #GstRTSPTransport.
1528  *
1529  * Returns: %TRUE if @trans was added
1530  */
1531 gboolean
1532 gst_rtsp_stream_add_transport (GstRTSPStream * stream,
1533     GstRTSPStreamTransport * trans)
1534 {
1535   GstRTSPStreamPrivate *priv;
1536   gboolean res;
1537
1538   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), FALSE);
1539   priv = stream->priv;
1540   g_return_val_if_fail (GST_IS_RTSP_STREAM_TRANSPORT (trans), FALSE);
1541   g_return_val_if_fail (priv->is_joined, FALSE);
1542
1543   g_mutex_lock (&priv->lock);
1544   res = update_transport (stream, trans, TRUE);
1545   g_mutex_unlock (&priv->lock);
1546
1547   return res;
1548 }
1549
1550 /**
1551  * gst_rtsp_stream_remove_transport:
1552  * @stream: a #GstRTSPStream
1553  * @trans: a #GstRTSPStreamTransport
1554  *
1555  * Remove the transport in @trans from @stream. The media of @stream will
1556  * not be sent to the values configured in @trans.
1557  *
1558  * @stream must be joined to a bin.
1559  *
1560  * @trans must contain a valid #GstRTSPTransport.
1561  *
1562  * Returns: %TRUE if @trans was removed
1563  */
1564 gboolean
1565 gst_rtsp_stream_remove_transport (GstRTSPStream * stream,
1566     GstRTSPStreamTransport * trans)
1567 {
1568   GstRTSPStreamPrivate *priv;
1569   gboolean res;
1570
1571   g_return_val_if_fail (GST_IS_RTSP_STREAM (stream), FALSE);
1572   priv = stream->priv;
1573   g_return_val_if_fail (GST_IS_RTSP_STREAM_TRANSPORT (trans), FALSE);
1574   g_return_val_if_fail (priv->is_joined, FALSE);
1575
1576   g_mutex_lock (&priv->lock);
1577   res = update_transport (stream, trans, FALSE);
1578   g_mutex_unlock (&priv->lock);
1579
1580   return res;
1581 }