]> git.sesse.net Git - vlc/blob - modules/services_discovery/sap.c
a328596948e9892a14255ed082a266727b9d24df
[vlc] / modules / services_discovery / sap.c
1 /*****************************************************************************
2  * sap.c :  SAP interface module
3  *****************************************************************************
4  * Copyright (C) 2004-2005 the VideoLAN team
5  * Copyright © 2007 Rémi Denis-Courmont
6  * $Id$
7  *
8  * Authors: Clément Stenac <zorglub@videolan.org>
9  *          Rémi Denis-Courmont
10  *
11  * This program is free software; you can redistribute it and/or modify
12  * it under the terms of the GNU General Public License as published by
13  * the Free Software Foundation; either version 2 of the License, or
14  * (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *****************************************************************************/
25
26 /*****************************************************************************
27  * Includes
28  *****************************************************************************/
29 #ifdef HAVE_CONFIG_H
30 # include "config.h"
31 #endif
32
33 #include <vlc_common.h>
34 #include <vlc_plugin.h>
35 #include <assert.h>
36
37 #include <vlc_demux.h>
38 #include <vlc_services_discovery.h>
39
40 #include <vlc_network.h>
41 #include <vlc_charset.h>
42
43 #ifdef HAVE_UNISTD_H
44 #    include <unistd.h>
45 #endif
46 #ifdef HAVE_ARPA_INET_H
47 # include <arpa/inet.h>
48 #endif
49 #ifdef HAVE_POLL
50 # include <poll.h>
51 #endif
52
53 #ifdef HAVE_ZLIB_H
54 #   include <zlib.h>
55 #endif
56
57 #ifndef WIN32
58 #   include <net/if.h>
59 #endif
60
61 /************************************************************************
62  * Macros and definitions
63  ************************************************************************/
64
65 #define MAX_LINE_LENGTH 256
66
67 /* SAP is always on that port */
68 #define SAP_PORT 9875
69 /* Global-scope SAP address */
70 #define SAP_V4_GLOBAL_ADDRESS   "224.2.127.254"
71 /* Organization-local SAP address */
72 #define SAP_V4_ORG_ADDRESS      "239.195.255.255"
73 /* Local (smallest non-link-local scope) SAP address */
74 #define SAP_V4_LOCAL_ADDRESS    "239.255.255.255"
75 /* Link-local SAP address */
76 #define SAP_V4_LINK_ADDRESS     "224.0.0.255"
77 #define ADD_SESSION 1
78
79 /*****************************************************************************
80  * Module descriptor
81  *****************************************************************************/
82 #define SAP_ADDR_TEXT N_( "SAP multicast address" )
83 #define SAP_ADDR_LONGTEXT N_( "The SAP module normally chooses itself the " \
84                               "right addresses to listen to. However, you " \
85                               "can specify a specific address." )
86 #define SAP_TIMEOUT_TEXT N_( "SAP timeout (seconds)" )
87 #define SAP_TIMEOUT_LONGTEXT N_( \
88        "Delay after which SAP items get deleted if no new announcement " \
89        "is received." )
90 #define SAP_PARSE_TEXT N_( "Try to parse the announce" )
91 #define SAP_PARSE_LONGTEXT N_( \
92        "This enables actual parsing of the announces by the SAP module. " \
93        "Otherwise, all announcements are parsed by the \"live555\" " \
94        "(RTP/RTSP) module." )
95 #define SAP_STRICT_TEXT N_( "SAP Strict mode" )
96 #define SAP_STRICT_LONGTEXT N_( \
97        "When this is set, the SAP parser will discard some non-compliant " \
98        "announcements." )
99
100 /* Callbacks */
101     static int  Open ( vlc_object_t * );
102     static void Close( vlc_object_t * );
103     static int  OpenDemux ( vlc_object_t * );
104     static void CloseDemux ( vlc_object_t * );
105
106 VLC_SD_PROBE_HELPER("sap", "Network streams (SAP)", SD_CAT_LAN)
107
108 vlc_module_begin ()
109     set_shortname( N_("SAP"))
110     set_description( N_("Network streams (SAP)") )
111     set_category( CAT_PLAYLIST )
112     set_subcategory( SUBCAT_PLAYLIST_SD )
113
114     add_string( "sap-addr", NULL,
115                 SAP_ADDR_TEXT, SAP_ADDR_LONGTEXT, true )
116     add_obsolete_bool( "sap-ipv4" ) /* since 2.0.0 */
117     add_obsolete_bool( "sap-ipv6" ) /* since 2.0.0 */
118     add_integer( "sap-timeout", 1800,
119                  SAP_TIMEOUT_TEXT, SAP_TIMEOUT_LONGTEXT, true )
120     add_bool( "sap-parse", true,
121                SAP_PARSE_TEXT,SAP_PARSE_LONGTEXT, true )
122     add_bool( "sap-strict", false,
123                SAP_STRICT_TEXT,SAP_STRICT_LONGTEXT, true )
124     add_obsolete_bool( "sap-timeshift" ) /* Redumdant since 1.0.0 */
125
126     set_capability( "services_discovery", 0 )
127     set_callbacks( Open, Close )
128
129     VLC_SD_PROBE_SUBMODULE
130
131     add_submodule ()
132         set_description( N_("SDP Descriptions parser") )
133         add_shortcut( "sdp" )
134         set_capability( "demux", 51 )
135         set_callbacks( OpenDemux, CloseDemux )
136 vlc_module_end ()
137
138
139 /*****************************************************************************
140  * Local structures
141  *****************************************************************************/
142
143 typedef struct sdp_t sdp_t;
144 typedef struct attribute_t attribute_t;
145 typedef struct sap_announce_t sap_announce_t;
146
147
148 struct sdp_media_t
149 {
150     struct sdp_t           *parent;
151     char                   *fmt;
152     struct sockaddr_storage addr;
153     socklen_t               addrlen;
154     unsigned                n_addr;
155     int           i_attributes;
156     attribute_t  **pp_attributes;
157 };
158
159
160 /* The structure that contains sdp information */
161 struct  sdp_t
162 {
163     const char *psz_sdp;
164
165     /* o field */
166     char     username[64];
167     uint64_t session_id;
168     uint64_t session_version;
169     unsigned orig_ip_version;
170     char     orig_host[1024];
171
172     /* s= field */
173     char *psz_sessionname;
174
175     /* i= field */
176     char *psz_sessioninfo;
177
178     /* old cruft */
179     /* "computed" URI */
180     char *psz_uri;
181     int           i_media_type;
182     unsigned rtcp_port;
183
184     /* a= global attributes */
185     int           i_attributes;
186     attribute_t  **pp_attributes;
187
188     /* medias (well, we only support one atm) */
189     unsigned            mediac;
190     struct sdp_media_t *mediav;
191 };
192
193 struct attribute_t
194 {
195     const char *value;
196     char name[];
197 };
198
199 struct sap_announce_t
200 {
201     mtime_t i_last;
202     mtime_t i_period;
203     uint8_t i_period_trust;
204
205     uint16_t    i_hash;
206     uint32_t    i_source[4];
207
208     /* SAP annnounces must only contain one SDP */
209     sdp_t       *p_sdp;
210
211     input_item_t * p_item;
212 };
213
214 struct services_discovery_sys_t
215 {
216     vlc_thread_t thread;
217
218     /* Socket descriptors */
219     int i_fd;
220     int *pi_fd;
221
222     /* Table of announces */
223     int i_announces;
224     struct sap_announce_t **pp_announces;
225
226     /* Modes */
227     bool  b_strict;
228     bool  b_parse;
229
230     int i_timeout;
231 };
232
233 struct demux_sys_t
234 {
235     sdp_t *p_sdp;
236 };
237
238 /*****************************************************************************
239  * Local prototypes
240  *****************************************************************************/
241
242
243 /* Main functions */
244     static int Demux( demux_t *p_demux );
245     static int Control( demux_t *, int, va_list );
246     static void *Run  ( void *p_sd );
247
248 /* Main parsing functions */
249     static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp );
250     static int ParseSAP( services_discovery_t *p_sd, const uint8_t *p_buffer, size_t i_read );
251     static sdp_t *ParseSDP (vlc_object_t *p_sd, const char *psz_sdp);
252     static sap_announce_t *CreateAnnounce( services_discovery_t *, uint32_t *, uint16_t, sdp_t * );
253     static int RemoveAnnounce( services_discovery_t *p_sd, sap_announce_t *p_announce );
254
255 /* Helper functions */
256     static inline attribute_t *MakeAttribute (const char *str);
257     static const char *GetAttribute (attribute_t **tab, unsigned n, const char *name);
258     static inline void FreeAttribute (attribute_t *a);
259     static const char *FindAttribute (const sdp_t *sdp, unsigned media,
260                                       const char *name);
261
262     static bool IsSameSession( sdp_t *p_sdp1, sdp_t *p_sdp2 );
263     static int InitSocket( services_discovery_t *p_sd, const char *psz_address, int i_port );
264     static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i_len );
265     static void FreeSDP( sdp_t *p_sdp );
266
267 static inline int min_int( int a, int b )
268 {
269     return a > b ? b : a;
270 }
271
272 static bool IsWellKnownPayload (int type)
273 {
274     switch (type)
275     {   /* Should be in sync with modules/demux/rtp.c */
276         case  0: /* PCMU/8000 */
277         case  3:
278         case  8: /* PCMA/8000 */
279         case 10: /* L16/44100/2 */
280         case 11: /* L16/44100 */
281         case 12:
282         case 14: /* MPA/90000 */
283         case 32: /* MPV/90000 */
284         case 33: /* MP2/90000 */
285             return true;
286    }
287    return false;
288 }
289
290 /*****************************************************************************
291  * Open: initialize and create stuff
292  *****************************************************************************/
293 static int Open( vlc_object_t *p_this )
294 {
295     services_discovery_t *p_sd = ( services_discovery_t* )p_this;
296     services_discovery_sys_t *p_sys  = (services_discovery_sys_t *)
297                                 malloc( sizeof( services_discovery_sys_t ) );
298     if( !p_sys )
299         return VLC_ENOMEM;
300
301     p_sys->i_timeout = var_CreateGetInteger( p_sd, "sap-timeout" );
302
303     p_sd->p_sys  = p_sys;
304
305     p_sys->pi_fd = NULL;
306     p_sys->i_fd = 0;
307
308     p_sys->b_strict = var_CreateGetBool( p_sd, "sap-strict");
309     p_sys->b_parse = var_CreateGetBool( p_sd, "sap-parse" );
310
311     p_sys->i_announces = 0;
312     p_sys->pp_announces = NULL;
313     /* TODO: create sockets here, and fix racy sockets table */
314     if (vlc_clone (&p_sys->thread, Run, p_sd, VLC_THREAD_PRIORITY_LOW))
315     {
316         free (p_sys);
317         return VLC_EGENERIC;
318     }
319
320     return VLC_SUCCESS;
321 }
322
323 /*****************************************************************************
324  * OpenDemux: initialize and create stuff
325  *****************************************************************************/
326 static int OpenDemux( vlc_object_t *p_this )
327 {
328     demux_t *p_demux = (demux_t *)p_this;
329     const uint8_t *p_peek;
330     char *psz_sdp = NULL;
331     sdp_t *p_sdp = NULL;
332     int errval = VLC_EGENERIC;
333     size_t i_len;
334
335     if( !var_CreateGetBool( p_demux, "sap-parse" ) )
336     {
337         /* We want livedotcom module to parse this SDP file */
338         return VLC_EGENERIC;
339     }
340
341     assert( p_demux->s ); /* this is NOT an access_demux */
342
343     /* Probe for SDP */
344     if( stream_Peek( p_demux->s, &p_peek, 7 ) < 7 )
345         return VLC_EGENERIC;
346
347     if( memcmp( p_peek, "v=0\r\no=", 7 ) && memcmp( p_peek, "v=0\no=", 6 ) )
348         return VLC_EGENERIC;
349
350     /* Gather the complete sdp file */
351     for( i_len = 0, psz_sdp = NULL; i_len < 65536; )
352     {
353         const int i_read_max = 1024;
354         char *psz_sdp_new = realloc( psz_sdp, i_len + i_read_max );
355         size_t i_read;
356         if( psz_sdp_new == NULL )
357         {
358             errval = VLC_ENOMEM;
359             goto error;
360         }
361         psz_sdp = psz_sdp_new;
362
363         i_read = stream_Read( p_demux->s, &psz_sdp[i_len], i_read_max );
364         if( (int)i_read < 0 )
365         {
366             msg_Err( p_demux, "cannot read SDP" );
367             goto error;
368         }
369         i_len += i_read;
370
371         psz_sdp[i_len] = '\0';
372
373         if( (int)i_read < i_read_max )
374             break; // EOF
375     }
376
377     p_sdp = ParseSDP( VLC_OBJECT(p_demux), psz_sdp );
378
379     if( !p_sdp )
380     {
381         msg_Warn( p_demux, "invalid SDP");
382         goto error;
383     }
384
385     if( ParseConnection( VLC_OBJECT( p_demux ), p_sdp ) )
386     {
387         p_sdp->psz_uri = NULL;
388     }
389     if (!IsWellKnownPayload (p_sdp->i_media_type))
390         goto error;
391     if( p_sdp->psz_uri == NULL ) goto error;
392
393     p_demux->p_sys = (demux_sys_t *)malloc( sizeof(demux_sys_t) );
394     if( unlikely( !p_demux->p_sys ) )
395         goto error;
396     p_demux->p_sys->p_sdp = p_sdp;
397     p_demux->pf_control = Control;
398     p_demux->pf_demux = Demux;
399
400     FREENULL( psz_sdp );
401     return VLC_SUCCESS;
402
403 error:
404     FREENULL( psz_sdp );
405     if( p_sdp ) FreeSDP( p_sdp ); p_sdp = NULL;
406     stream_Seek( p_demux->s, 0 );
407     return errval;
408 }
409
410 /*****************************************************************************
411  * Close:
412  *****************************************************************************/
413 static void Close( vlc_object_t *p_this )
414 {
415     services_discovery_t *p_sd = ( services_discovery_t* )p_this;
416     services_discovery_sys_t    *p_sys  = p_sd->p_sys;
417     int i;
418
419     vlc_cancel (p_sys->thread);
420     vlc_join (p_sys->thread, NULL);
421
422     for( i = p_sys->i_fd-1 ; i >= 0 ; i-- )
423     {
424         net_Close( p_sys->pi_fd[i] );
425     }
426     FREENULL( p_sys->pi_fd );
427
428     for( i = p_sys->i_announces  - 1;  i>= 0; i-- )
429     {
430         RemoveAnnounce( p_sd, p_sys->pp_announces[i] );
431     }
432     FREENULL( p_sys->pp_announces );
433
434     free( p_sys );
435 }
436
437 /*****************************************************************************
438  * CloseDemux: Close the demuxer
439  *****************************************************************************/
440 static void CloseDemux( vlc_object_t *p_this )
441 {
442     demux_t *p_demux = (demux_t *)p_this;
443
444     if( p_demux->p_sys->p_sdp )
445         FreeSDP( p_demux->p_sys->p_sdp );
446     free( p_demux->p_sys );
447 }
448
449 /*****************************************************************************
450  * Run: main SAP thread
451  *****************************************************************************
452  * Listens to SAP packets, and sends them to packet_handle
453  *****************************************************************************/
454 #define MAX_SAP_BUFFER 5000
455
456 static void *Run( void *data )
457 {
458     services_discovery_t *p_sd = data;
459     char *psz_addr;
460     int i;
461     int timeout = -1;
462     int canc = vlc_savecancel ();
463
464     /* Braindead Winsock DNS resolver will get stuck over 2 seconds per failed
465      * DNS queries, even if the DNS server returns an error with milliseconds.
466      * You don't want to know why the bug (as of XP SP2) wasn't fixed since
467      * Winsock 1.1 from Windows 95, if not Windows 3.1.
468      * Anyway, to avoid a 30 seconds delay for failed IPv6 socket creation,
469      * we have to open sockets in Run() rather than Open(). */
470     InitSocket( p_sd, SAP_V4_GLOBAL_ADDRESS, SAP_PORT );
471     InitSocket( p_sd, SAP_V4_ORG_ADDRESS, SAP_PORT );
472     InitSocket( p_sd, SAP_V4_LOCAL_ADDRESS, SAP_PORT );
473     InitSocket( p_sd, SAP_V4_LINK_ADDRESS, SAP_PORT );
474
475     char psz_address[NI_MAXNUMERICHOST] = "ff02::2:7ffe%";
476 #ifndef WIN32
477     struct if_nameindex *l = if_nameindex ();
478     if (l != NULL)
479     {
480         char *ptr = strchr (psz_address, '%') + 1;
481         for (unsigned i = 0; l[i].if_index; i++)
482         {
483             strcpy (ptr, l[i].if_name);
484             InitSocket (p_sd, psz_address, SAP_PORT);
485         }
486         if_freenameindex (l);
487     }
488 #else
489         /* this is the Winsock2 equivalant of SIOCGIFCONF on BSD stacks,
490            which if_nameindex uses internally anyway */
491
492         // first create a dummy socket to pin down the protocol family
493         SOCKET s = socket(PF_INET6, SOCK_DGRAM, IPPROTO_UDP);
494         if( s != INVALID_SOCKET )
495         {
496             INTERFACE_INFO ifaces[10]; // Assume there will be no more than 10 IP interfaces
497             DWORD len = sizeof(ifaces);
498
499             if( SOCKET_ERROR != WSAIoctl(s, SIO_GET_INTERFACE_LIST, NULL, 0, &ifaces, len, &len, NULL, NULL) )
500             {
501                 unsigned ifcount = len/sizeof(INTERFACE_INFO);
502                 char *ptr = strchr (psz_address, '%') + 1;
503                 for(unsigned i = 1; i<=ifcount; ++i )
504                 {
505                     // append link-local zone identifier
506                     sprintf(ptr, "%d", i);
507                 }
508             }
509             closesocket(s);
510         }
511 #endif
512     *strchr (psz_address, '%') = '\0';
513
514     static const char ipv6_scopes[] = "1456789ABCDE";
515     for (const char *c_scope = ipv6_scopes; *c_scope; c_scope++)
516     {
517         psz_address[3] = *c_scope;
518         InitSocket( p_sd, psz_address, SAP_PORT );
519     }
520
521     psz_addr = var_CreateGetString( p_sd, "sap-addr" );
522     if( psz_addr && *psz_addr )
523         InitSocket( p_sd, psz_addr, SAP_PORT );
524     free( psz_addr );
525
526     if( p_sd->p_sys->i_fd == 0 )
527     {
528         msg_Err( p_sd, "unable to listen on any address" );
529         return NULL;
530     }
531
532     /* read SAP packets */
533     for (;;)
534     {
535         vlc_restorecancel (canc);
536         unsigned n = p_sd->p_sys->i_fd;
537         struct pollfd ufd[n];
538
539         for (unsigned i = 0; i < n; i++)
540         {
541             ufd[i].fd = p_sd->p_sys->pi_fd[i];
542             ufd[i].events = POLLIN;
543             ufd[i].revents = 0;
544         }
545
546         int val = poll (ufd, n, timeout);
547         canc = vlc_savecancel ();
548         if (val > 0)
549         {
550             for (unsigned i = 0; i < n; i++)
551             {
552                 if (ufd[i].revents)
553                 {
554                     uint8_t p_buffer[MAX_SAP_BUFFER+1];
555                     ssize_t i_read;
556
557                     i_read = net_Read (p_sd, ufd[i].fd, NULL, p_buffer,
558                                        MAX_SAP_BUFFER, false);
559                     if (i_read < 0)
560                         msg_Warn (p_sd, "receive error: %m");
561                     if (i_read > 6)
562                     {
563                         /* Parse the packet */
564                         p_buffer[i_read] = '\0';
565                         ParseSAP (p_sd, p_buffer, i_read);
566                     }
567                 }
568             }
569         }
570
571         mtime_t now = mdate();
572
573         /* A 1 hour timeout correspond to the RFC Implicit timeout.
574          * This timeout is tuned in the following loop. */
575         timeout = 1000 * 60 * 60;
576
577         /* Check for items that need deletion */
578         for( i = 0; i < p_sd->p_sys->i_announces; i++ )
579         {
580             mtime_t i_timeout = ( mtime_t ) 1000000 * p_sd->p_sys->i_timeout;
581             sap_announce_t * p_announce = p_sd->p_sys->pp_announces[i];
582             mtime_t i_last_period = now - p_announce->i_last;
583
584             /* Remove the announcement, if the last announcement was 1 hour ago
585              * or if the last packet emitted was 3 times the average time
586              * between two packets */
587             if( ( p_announce->i_period_trust > 5 && i_last_period > 3 * p_announce->i_period ) ||
588                 i_last_period > i_timeout )
589             {
590                 RemoveAnnounce( p_sd, p_announce );
591             }
592             else
593             {
594                 /* Compute next timeout */
595                 if( p_announce->i_period_trust > 5 )
596                     timeout = min_int((3 * p_announce->i_period - i_last_period) / 1000, timeout);
597                 timeout = min_int((i_timeout - i_last_period)/1000, timeout);
598             }
599         }
600
601         if( !p_sd->p_sys->i_announces )
602             timeout = -1; /* We can safely poll indefinitely. */
603         else if( timeout < 200 )
604             timeout = 200; /* Don't wakeup too fast. */
605     }
606     assert (0);
607 }
608
609 /**********************************************************************
610  * Demux: reads and demuxes data packets
611  * Return -1 if error, 0 if EOF, 1 else
612  **********************************************************************/
613 static int Demux( demux_t *p_demux )
614 {
615     sdp_t *p_sdp = p_demux->p_sys->p_sdp;
616     input_thread_t *p_input;
617     input_item_t *p_parent_input;
618
619     p_input = demux_GetParentInput( p_demux );
620     assert( p_input );
621     if( !p_input )
622     {
623         msg_Err( p_demux, "parent input could not be found" );
624         return VLC_EGENERIC;
625     }
626
627     /* This item hasn't been held by input_GetItem
628      * don't release it */
629     p_parent_input = input_GetItem( p_input );
630
631     input_item_SetURI( p_parent_input, p_sdp->psz_uri );
632     input_item_SetName( p_parent_input, p_sdp->psz_sessionname );
633     if( p_sdp->rtcp_port )
634     {
635         char *rtcp;
636         if( asprintf( &rtcp, ":rtcp-port=%u", p_sdp->rtcp_port ) != -1 )
637         {
638             input_item_AddOption( p_parent_input, rtcp, VLC_INPUT_OPTION_TRUSTED );
639             free( rtcp );
640         }
641     }
642
643     vlc_mutex_lock( &p_parent_input->lock );
644
645     p_parent_input->i_type = ITEM_TYPE_NET;
646
647     vlc_mutex_unlock( &p_parent_input->lock );
648     vlc_object_release( p_input );
649     return VLC_SUCCESS;
650 }
651
652 static int Control( demux_t *p_demux, int i_query, va_list args )
653 {
654     VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args);
655     return VLC_EGENERIC;
656 }
657
658 /**************************************************************
659  * Local functions
660  **************************************************************/
661
662 /* i_read is at least > 6 */
663 static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
664                      size_t len )
665 {
666     int i;
667     const char          *psz_sdp;
668     const uint8_t *end = buf + len;
669     sdp_t               *p_sdp;
670     uint32_t            i_source[4];
671
672     assert (buf[len] == '\0');
673
674     if (len < 4)
675         return VLC_EGENERIC;
676
677     uint8_t flags = buf[0];
678     uint8_t auth_len = buf[1];
679
680     /* First, check the sap announce is correct */
681     if ((flags >> 5) != 1)
682         return VLC_EGENERIC;
683
684     bool b_ipv6 = (flags & 0x10) != 0;
685     bool b_need_delete = (flags & 0x04) != 0;
686
687     if (flags & 0x02)
688     {
689         msg_Dbg( p_sd, "encrypted packet, unsupported" );
690         return VLC_EGENERIC;
691     }
692
693     bool b_compressed = (flags & 0x01) != 0;
694
695     uint16_t i_hash = U16_AT (buf + 2);
696
697     if( p_sd->p_sys->b_strict && i_hash == 0 )
698     {
699         msg_Dbg( p_sd, "strict mode, discarding announce with null id hash");
700         return VLC_EGENERIC;
701     }
702
703     buf += 4;
704     if( b_ipv6 )
705     {
706         for( int i = 0; i < 4; i++,buf+=4)
707             i_source[i] = U32_AT(buf);
708     }
709     else
710     {
711         memset(i_source, 0, sizeof(i_source));
712         i_source[3] = U32_AT(buf);
713         buf+=4;
714     }
715     // Skips auth data
716     buf += auth_len;
717     if (buf > end)
718         return VLC_EGENERIC;
719
720     uint8_t *decomp = NULL;
721     if( b_compressed )
722     {
723         int newsize = Decompress (buf, &decomp, end - buf);
724         if (newsize < 0)
725         {
726             msg_Dbg( p_sd, "decompression of SAP packet failed" );
727             return VLC_EGENERIC;
728         }
729
730         decomp = realloc (decomp, newsize + 1);
731         decomp[newsize] = '\0';
732         psz_sdp = (const char *)decomp;
733         len = newsize;
734     }
735     else
736     {
737         psz_sdp = (const char *)buf;
738         len = end - buf;
739     }
740
741     /* len is a strlen here here. both buf and decomp are len+1 where the 1 should be a \0 */
742     assert( psz_sdp[len] == '\0');
743
744     /* Skip payload type */
745     /* SAPv1 has implicit "application/sdp" payload type: first line is v=0 */
746     if (strncmp (psz_sdp, "v=0", 3))
747     {
748         size_t clen = strlen (psz_sdp) + 1;
749
750         if (strcmp (psz_sdp, "application/sdp"))
751         {
752             msg_Dbg (p_sd, "unsupported content type: %s", psz_sdp);
753             return VLC_EGENERIC;
754         }
755
756         // skips content type
757         if (len <= clen)
758             return VLC_EGENERIC;
759
760         len -= clen;
761         psz_sdp += clen;
762     }
763
764     /* Parse SDP info */
765     p_sdp = ParseSDP( VLC_OBJECT(p_sd), psz_sdp );
766
767     if( p_sdp == NULL )
768         return VLC_EGENERIC;
769
770     p_sdp->psz_sdp = psz_sdp;
771
772     /* Decide whether we should add a playlist item for this SDP */
773     /* Parse connection information (c= & m= ) */
774     if( ParseConnection( VLC_OBJECT(p_sd), p_sdp ) )
775         p_sdp->psz_uri = NULL;
776
777     /* Multi-media or no-parse -> pass to LIVE.COM */
778     if( !IsWellKnownPayload( p_sdp->i_media_type ) || !p_sd->p_sys->b_parse )
779     {
780         free( p_sdp->psz_uri );
781         if (asprintf( &p_sdp->psz_uri, "sdp://%s", p_sdp->psz_sdp ) == -1)
782             p_sdp->psz_uri = NULL;
783     }
784
785     if( p_sdp->psz_uri == NULL )
786     {
787         FreeSDP( p_sdp );
788         return VLC_EGENERIC;
789     }
790
791     for( i = 0 ; i< p_sd->p_sys->i_announces ; i++ )
792     {
793         sap_announce_t * p_announce = p_sd->p_sys->pp_announces[i];
794         /* FIXME: slow */
795         if( ( !i_hash && IsSameSession( p_announce->p_sdp, p_sdp ) )
796             || ( i_hash && p_announce->i_hash == i_hash
797                  && !memcmp(p_announce->i_source, i_source, sizeof(i_source)) ) )
798         {
799             /* We don't support delete announcement as they can easily
800              * Be used to highjack an announcement by a third party.
801              * Instead we cleverly implement Implicit Announcement removal.
802              *
803              * if( b_need_delete )
804              *    RemoveAnnounce( p_sd, p_sd->p_sys->pp_announces[i]);
805              * else
806              */
807
808             if( !b_need_delete )
809             {
810                 /* No need to go after six, as we start to trust the
811                  * average period at six */
812                 if( p_announce->i_period_trust <= 5 )
813                     p_announce->i_period_trust++;
814
815                 /* Compute the average period */
816                 mtime_t now = mdate();
817                 p_announce->i_period = ( p_announce->i_period * (p_announce->i_period_trust-1) + (now - p_announce->i_last) ) / p_announce->i_period_trust;
818                 p_announce->i_last = now;
819             }
820             FreeSDP( p_sdp ); p_sdp = NULL;
821             return VLC_SUCCESS;
822         }
823     }
824
825     CreateAnnounce( p_sd, i_source, i_hash, p_sdp );
826
827     FREENULL (decomp);
828     return VLC_SUCCESS;
829 }
830
831 sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint32_t *i_source, uint16_t i_hash,
832                                 sdp_t *p_sdp )
833 {
834     input_item_t *p_input;
835     const char *psz_value;
836     sap_announce_t *p_sap = (sap_announce_t *)malloc(
837                                         sizeof(sap_announce_t ) );
838     services_discovery_sys_t *p_sys;
839     if( p_sap == NULL )
840         return NULL;
841
842     p_sys = p_sd->p_sys;
843
844     p_sap->i_last = mdate();
845     p_sap->i_period = 0;
846     p_sap->i_period_trust = 0;
847     p_sap->i_hash = i_hash;
848     memcpy (p_sap->i_source, i_source, sizeof(p_sap->i_source));
849     p_sap->p_sdp = p_sdp;
850
851     /* Released in RemoveAnnounce */
852     p_input = input_item_NewWithType( p_sap->p_sdp->psz_uri,
853                                       p_sdp->psz_sessionname,
854                                       0, NULL, 0, -1, ITEM_TYPE_NET );
855     vlc_meta_t *p_meta = vlc_meta_New();
856     vlc_meta_Set( p_meta, vlc_meta_Description, p_sdp->psz_sessioninfo );
857     p_input->p_meta = p_meta;
858     p_sap->p_item = p_input;
859     if( !p_input )
860     {
861         free( p_sap );
862         return NULL;
863     }
864
865     if( p_sdp->rtcp_port )
866     {
867         char *rtcp;
868         if( asprintf( &rtcp, ":rtcp-port=%u", p_sdp->rtcp_port ) != -1 )
869         {
870             input_item_AddOption( p_input, rtcp, VLC_INPUT_OPTION_TRUSTED );
871             free( rtcp );
872         }
873     }
874
875     psz_value = GetAttribute( p_sap->p_sdp->pp_attributes, p_sap->p_sdp->i_attributes, "tool" );
876     if( psz_value != NULL )
877     {
878         input_item_AddInfo( p_input, _("Session"), _("Tool"), "%s", psz_value );
879     }
880     if( strcmp( p_sdp->username, "-" ) )
881     {
882         input_item_AddInfo( p_input, _("Session"), _("User"), "%s",
883                            p_sdp->username );
884     }
885
886     /* Handle category */
887     psz_value = GetAttribute(p_sap->p_sdp->pp_attributes,
888                              p_sap->p_sdp->i_attributes, "cat");
889     if (psz_value != NULL)
890     {
891         /* a=cat provides a dot-separated hierarchy.
892          * For the time being only replace dots with pipe. TODO: FIXME */
893         char *str = strdup(psz_value);
894         if (likely(str != NULL))
895             for (char *p = strchr(str, '.'); p != NULL; p = strchr(p, '.'))
896                 *(p++) = '|';
897         services_discovery_AddItem(p_sd, p_input, str ? str : psz_value);
898         free(str);
899     }
900     else
901     {
902         /* backward compatibility with VLC 0.7.3-2.0.0 senders */
903         psz_value = GetAttribute(p_sap->p_sdp->pp_attributes,
904                                  p_sap->p_sdp->i_attributes, "x-plgroup");
905         services_discovery_AddItem(p_sd, p_input, psz_value);
906     }
907
908     TAB_APPEND( p_sys->i_announces, p_sys->pp_announces, p_sap );
909
910     return p_sap;
911 }
912
913
914 static const char *FindAttribute (const sdp_t *sdp, unsigned media,
915                                   const char *name)
916 {
917     /* Look for media attribute, and fallback to session */
918     const char *attr = GetAttribute (sdp->mediav[media].pp_attributes,
919                                      sdp->mediav[media].i_attributes, name);
920     if (attr == NULL)
921         attr = GetAttribute (sdp->pp_attributes, sdp->i_attributes, name);
922     return attr;
923 }
924
925
926 /* Fill p_sdp->psz_uri */
927 static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
928 {
929     if (p_sdp->mediac == 0)
930     {
931         msg_Dbg (p_obj, "Ignoring SDP with no media");
932         return VLC_EGENERIC;
933     }
934
935     for (unsigned i = 1; i < p_sdp->mediac; i++)
936     {
937         if ((p_sdp->mediav[i].n_addr != p_sdp->mediav->n_addr)
938          || (p_sdp->mediav[i].addrlen != p_sdp->mediav->addrlen)
939          || memcmp (&p_sdp->mediav[i].addr, &p_sdp->mediav->addr,
940                     p_sdp->mediav->addrlen))
941         {
942             msg_Dbg (p_obj, "Multiple media ports not supported -> live555");
943             return VLC_EGENERIC;
944         }
945     }
946
947     if (p_sdp->mediav->n_addr != 1)
948     {
949         msg_Dbg (p_obj, "Layered encoding not supported -> live555");
950         return VLC_EGENERIC;
951     }
952
953     char psz_uri[1026];
954     const char *host;
955     int port;
956
957     psz_uri[0] = '[';
958     if (vlc_getnameinfo ((struct sockaddr *)&(p_sdp->mediav->addr),
959                          p_sdp->mediav->addrlen, psz_uri + 1,
960                          sizeof (psz_uri) - 2, &port, NI_NUMERICHOST))
961         return VLC_EGENERIC;
962
963     if (strchr (psz_uri + 1, ':'))
964     {
965         host = psz_uri;
966         strcat (psz_uri, "]");
967     }
968     else
969         host = psz_uri + 1;
970
971     /* Parse m= field */
972     char *sdp_proto = strdup (p_sdp->mediav[0].fmt);
973     if (sdp_proto == NULL)
974         return VLC_ENOMEM;
975
976     char *subtype = strchr (sdp_proto, ' ');
977     if (subtype == NULL)
978     {
979         msg_Dbg (p_obj, "missing SDP media subtype: %s", sdp_proto);
980         free (sdp_proto);
981         return VLC_EGENERIC;
982     }
983     else
984     {
985         *subtype++ = '\0';
986         /* FIXME: check for multiple payload types in RTP/AVP case.
987          * FIXME: check for "mpeg" subtype in raw udp case. */
988         if (!strcasecmp (sdp_proto, "udp"))
989             p_sdp->i_media_type = 33;
990         else
991             p_sdp->i_media_type = atoi (subtype);
992     }
993
994     /* RTP protocol, nul, VLC shortcut, nul, flags byte as follow:
995      * 0x1: Connection-Oriented media. */
996     static const char proto_match[] =
997         "udp\0"             "udp\0\0"
998         "RTP/AVP\0"         "rtp\0\0"
999         "UDPLite/RTP/AVP\0" "udplite\0\0"
1000         "DCCP/RTP/AVP\0"    "dccp\0\1"
1001         "TCP/RTP/AVP\0"     "rtptcp\0\1"
1002         "\0";
1003
1004     const char *vlc_proto = NULL;
1005     uint8_t flags = 0;
1006     for (const char *proto = proto_match; *proto;)
1007     {
1008         if (strcasecmp (proto, sdp_proto) == 0)
1009         {
1010             vlc_proto = proto + strlen (proto) + 1;
1011             flags = vlc_proto[strlen (vlc_proto) + 1];
1012             break;
1013         }
1014         proto += strlen (proto) + 1;
1015         proto += strlen (proto) + 2;
1016     }
1017
1018     free (sdp_proto);
1019     if (vlc_proto == NULL)
1020     {
1021         msg_Dbg (p_obj, "unknown SDP media protocol: %s",
1022                  p_sdp->mediav[0].fmt);
1023         return VLC_EGENERIC;
1024     }
1025
1026     if (!strcmp (vlc_proto, "udp") || FindAttribute (p_sdp, 0, "rtcp-mux"))
1027         p_sdp->rtcp_port = 0;
1028     else
1029     {
1030         const char *rtcp = FindAttribute (p_sdp, 0, "rtcp");
1031         if (rtcp)
1032             p_sdp->rtcp_port = atoi (rtcp);
1033         else
1034         if (port & 1) /* odd port -> RTCP; next even port -> RTP */
1035             p_sdp->rtcp_port = port++;
1036         else /* even port -> RTP; next odd port -> RTCP */
1037             p_sdp->rtcp_port = port + 1;
1038     }
1039
1040     if (flags & 1)
1041     {
1042         /* Connection-oriented media */
1043         const char *setup = FindAttribute (p_sdp, 0, "setup");
1044         if (setup == NULL)
1045             setup = "active"; /* default value */
1046
1047         if (strcmp (setup, "actpass") && strcmp (setup, "passive"))
1048         {
1049             msg_Dbg (p_obj, "unsupported COMEDIA mode: %s", setup);
1050             return VLC_EGENERIC;
1051         }
1052
1053         if (asprintf (&p_sdp->psz_uri, "%s://%s:%d", vlc_proto,
1054                       host, port) == -1)
1055             return VLC_ENOMEM;
1056     }
1057     else
1058     {
1059         /* Non-connected (normally multicast) media */
1060         char psz_source[258] = "";
1061         const char *sfilter = FindAttribute (p_sdp, 0, "source-filter");
1062         if (sfilter != NULL)
1063         {
1064             char psz_source_ip[256];
1065             unsigned ipv;
1066
1067             if (sscanf (sfilter, " incl IN IP%u %*s %255s ", &ipv,
1068                         psz_source_ip) == 2)
1069             {
1070                 /* According to RFC4570, FQDNs can be used for source-filters,
1071                  * but -seriously- this is impractical */
1072                 switch (ipv)
1073                 {
1074 #ifdef AF_INET6
1075                     case 6:
1076                     {
1077                         struct in6_addr addr;
1078                         if ((inet_pton (AF_INET6, psz_source_ip, &addr) > 0)
1079                         && (inet_ntop (AF_INET6, &addr, psz_source + 1,
1080                                         sizeof (psz_source) - 2) != NULL))
1081                         {
1082                             psz_source[0] = '[';
1083                             psz_source[strlen (psz_source)] = ']';
1084                         }
1085                         break;
1086                     }
1087 #endif
1088                     case 4:
1089                     {
1090                         struct in_addr addr;
1091                         if ((inet_pton (AF_INET, psz_source_ip, &addr) > 0)
1092                         && (inet_ntop (AF_INET, &addr, psz_source,
1093                                         sizeof (psz_source)) == NULL))
1094                             *psz_source = '\0';
1095                         break;
1096                     }
1097                 }
1098             }
1099         }
1100
1101         if (asprintf (&p_sdp->psz_uri, "%s://%s@%s:%i", vlc_proto, psz_source,
1102                      host, port) == -1)
1103             return VLC_ENOMEM;
1104     }
1105
1106     return VLC_SUCCESS;
1107 }
1108
1109
1110 static int ParseSDPConnection (const char *str, struct sockaddr_storage *addr,
1111                                socklen_t *addrlen, unsigned *number)
1112 {
1113     char host[60];
1114     unsigned fam, n1, n2;
1115
1116     int res = sscanf (str, "IN IP%u %59[^/]/%u/%u", &fam, host, &n1, &n2);
1117     if (res < 2)
1118         return -1;
1119
1120     switch (fam)
1121     {
1122 #ifdef AF_INET6
1123         case 6:
1124             addr->ss_family = AF_INET6;
1125 # ifdef HAVE_SA_LEN
1126             addr->ss_len =
1127 # endif
1128            *addrlen = sizeof (struct sockaddr_in6);
1129
1130             if (inet_pton (AF_INET6, host,
1131                            &((struct sockaddr_in6 *)addr)->sin6_addr) <= 0)
1132                 return -1;
1133
1134             *number = (res >= 3) ? n1 : 1;
1135             break;
1136 #endif
1137
1138         case 4:
1139             addr->ss_family = AF_INET;
1140 # ifdef HAVE_SA_LEN
1141             addr->ss_len =
1142 # endif
1143            *addrlen = sizeof (struct sockaddr_in);
1144
1145             if (inet_pton (AF_INET, host,
1146                            &((struct sockaddr_in *)addr)->sin_addr) <= 0)
1147                 return -1;
1148
1149             *number = (res >= 4) ? n2 : 1;
1150             break;
1151
1152         default:
1153             return -1;
1154     }
1155     return 0;
1156 }
1157
1158
1159 /***********************************************************************
1160  * ParseSDP : SDP parsing
1161  * *********************************************************************
1162  * Validate SDP and parse all fields
1163  ***********************************************************************/
1164 static sdp_t *ParseSDP (vlc_object_t *p_obj, const char *psz_sdp)
1165 {
1166     if( psz_sdp == NULL )
1167         return NULL;
1168
1169     sdp_t *p_sdp = calloc (1, sizeof (*p_sdp));
1170     if (p_sdp == NULL)
1171         return NULL;
1172
1173     char expect = 'V';
1174     struct sockaddr_storage glob_addr;
1175     memset (&glob_addr, 0, sizeof (glob_addr));
1176     socklen_t glob_len = 0;
1177     unsigned glob_count = 1;
1178     int port = 0;
1179
1180     /* TODO: use iconv and charset attribute instead of EnsureUTF8 */
1181     while (*psz_sdp)
1182     {
1183         /* Extract one line */
1184         char *eol = strchr (psz_sdp, '\n');
1185         size_t linelen = eol ? (size_t)(eol - psz_sdp) : strlen (psz_sdp);
1186         char line[linelen + 1];
1187         memcpy (line, psz_sdp, linelen);
1188         line[linelen] = '\0';
1189
1190         psz_sdp += linelen + 1;
1191
1192         /* Remove carriage return if present */
1193         eol = strchr (line, '\r');
1194         if (eol != NULL)
1195         {
1196             linelen = eol - line;
1197             line[linelen] = '\0';
1198         }
1199
1200         /* Validate line */
1201         char cat = line[0], *data = line + 2;
1202         if (!cat || (strchr ("vosiuepcbtrzkam", cat) == NULL))
1203         {
1204             /* MUST ignore SDP with unknown line type */
1205             msg_Dbg (p_obj, "unknown SDP line type: 0x%02x", (int)cat);
1206             goto error;
1207         }
1208         if (line[1] != '=')
1209         {
1210             msg_Dbg (p_obj, "invalid SDP line: %s", line);
1211             goto error;
1212         }
1213
1214         assert (linelen >= 2);
1215
1216         /* SDP parsing state machine
1217          * We INTERNALLY use uppercase for session, lowercase for media
1218          */
1219         switch (expect)
1220         {
1221             /* Session description */
1222             case 'V':
1223                 expect = 'O';
1224                 if (cat != 'v')
1225                 {
1226                     msg_Dbg (p_obj, "missing SDP version");
1227                     goto error;
1228                 }
1229                 if (strcmp (data, "0"))
1230                 {
1231                     msg_Dbg (p_obj, "unknown SDP version: %s", data);
1232                     goto error;
1233                 }
1234                 break;
1235
1236             case 'O':
1237             {
1238                 expect = 'S';
1239                 if (cat != 'o')
1240                 {
1241                     msg_Dbg (p_obj, "missing SDP originator");
1242                     goto error;
1243                 }
1244
1245                 if ((sscanf (data, "%63s %"SCNu64" %"SCNu64" IN IP%u %1023s",
1246                              p_sdp->username, &p_sdp->session_id,
1247                              &p_sdp->session_version, &p_sdp->orig_ip_version,
1248                              p_sdp->orig_host) != 5)
1249                  || ((p_sdp->orig_ip_version != 4)
1250                   && (p_sdp->orig_ip_version != 6)))
1251                 {
1252                     msg_Dbg (p_obj, "SDP origin not supported: %s", data);
1253                     /* Or maybe out-of-range, but this looks suspicious */
1254                     return NULL;
1255                 }
1256                 EnsureUTF8 (p_sdp->orig_host);
1257                 break;
1258             }
1259
1260             case 'S':
1261             {
1262                 expect = 'I';
1263                 if ((cat != 's') || !*data)
1264                 {
1265                     /* MUST be present AND non-empty */
1266                     msg_Dbg (p_obj, "missing SDP session name");
1267                     goto error;
1268                 }
1269                 assert (p_sdp->psz_sessionname == NULL); // no memleak here
1270                 p_sdp->psz_sessionname = strdup (data);
1271                 if (p_sdp->psz_sessionname == NULL)
1272                     goto error;
1273                 EnsureUTF8 (p_sdp->psz_sessionname);
1274                 break;
1275             }
1276
1277             case 'I':
1278             {
1279                 expect = 'U';
1280                 /* optional (and may be empty) */
1281                 if (cat == 'i')
1282                 {
1283                     assert (p_sdp->psz_sessioninfo == NULL);
1284                     p_sdp->psz_sessioninfo = strdup (data);
1285                     if (p_sdp->psz_sessioninfo == NULL)
1286                         goto error;
1287                     EnsureUTF8 (p_sdp->psz_sessioninfo);
1288                     break;
1289                 }
1290             }
1291
1292             case 'U':
1293                 expect = 'E';
1294                 if (cat == 'u')
1295                     break;
1296             case 'E':
1297                 expect = 'E';
1298                 if (cat == 'e')
1299                     break;
1300             case 'P':
1301                 expect = 'P';
1302                 if (cat == 'p')
1303                     break;
1304             case 'C':
1305                 expect = 'B';
1306                 if (cat == 'c')
1307                 {
1308                     if (ParseSDPConnection (data, &glob_addr, &glob_len,
1309                                             &glob_count))
1310                     {
1311                         msg_Dbg (p_obj, "SDP connection infos not supported: "
1312                                  "%s", data);
1313                         goto error;
1314                     }
1315                     break;
1316                 }
1317             case 'B':
1318                 assert (expect == 'B');
1319                 if (cat == 'b')
1320                     break;
1321             case 'T':
1322                 expect = 'R';
1323                 if (cat != 't')
1324                 {
1325                     msg_Dbg (p_obj, "missing SDP time description");
1326                     goto error;
1327                 }
1328                 break;
1329
1330             case 'R':
1331                 if ((cat == 't') || (cat == 'r'))
1332                     break;
1333
1334             case 'Z':
1335                 expect = 'K';
1336                 if (cat == 'z')
1337                     break;
1338             case 'K':
1339                 expect = 'A';
1340                 if (cat == 'k')
1341                     break;
1342             case 'A':
1343                 //expect = 'A';
1344                 if (cat == 'a')
1345                 {
1346                     attribute_t *p_attr = MakeAttribute (data);
1347                     TAB_APPEND( p_sdp->i_attributes, p_sdp->pp_attributes, p_attr );
1348                     break;
1349                 }
1350
1351             /* Media description */
1352             case 'm':
1353             media:
1354             {
1355                 expect = 'i';
1356                 if (cat != 'm')
1357                 {
1358                     msg_Dbg (p_obj, "missing SDP media description");
1359                     goto error;
1360                 }
1361                 struct sdp_media_t *m;
1362                 m = realloc (p_sdp->mediav, (p_sdp->mediac + 1) * sizeof (*m));
1363                 if (m == NULL)
1364                     goto error;
1365
1366                 p_sdp->mediav = m;
1367                 m += p_sdp->mediac;
1368                 p_sdp->mediac++;
1369
1370                 memset (m, 0, sizeof (*m));
1371                 memcpy (&m->addr, &glob_addr, m->addrlen = glob_len);
1372                 m->n_addr = glob_count;
1373
1374                 /* TODO: remember media type (if we need multiple medias) */
1375                 data = strchr (data, ' ');
1376                 if (data == NULL)
1377                 {
1378                     msg_Dbg (p_obj, "missing SDP media port");
1379                     goto error;
1380                 }
1381                 port = atoi (++data);
1382                 if (port <= 0 || port >= 65536)
1383                 {
1384                     msg_Dbg (p_obj, "invalid transport port %d", port);
1385                     goto error;
1386                 }
1387                 net_SetPort ((struct sockaddr *)&m->addr, htons (port));
1388
1389                 data = strchr (data, ' ');
1390                 if (data == NULL)
1391                 {
1392                     msg_Dbg (p_obj, "missing SDP media format");
1393                     goto error;
1394                 }
1395                 m->fmt = strdup (++data);
1396                 if (m->fmt == NULL)
1397                     goto error;
1398
1399                 break;
1400             }
1401             case 'i':
1402                 expect = 'c';
1403                 if (cat == 'i')
1404                     break;
1405             case 'c':
1406                 expect = 'b';
1407                 if (cat == 'c')
1408                 {
1409                     struct sdp_media_t *m = p_sdp->mediav + p_sdp->mediac - 1;
1410                     if (ParseSDPConnection (data, &m->addr, &m->addrlen,
1411                                             &m->n_addr))
1412                     {
1413                         msg_Dbg (p_obj, "SDP connection infos not supported: "
1414                                  "%s", data);
1415                         goto error;
1416                     }
1417                     net_SetPort ((struct sockaddr *)&m->addr, htons (port));
1418                     break;
1419                 }
1420             case 'b':
1421                 expect = 'b';
1422                 if (cat == 'b')
1423                     break;
1424             case 'k':
1425                 expect = 'a';
1426                 if (cat == 'k')
1427                     break;
1428             case 'a':
1429                 assert (expect == 'a');
1430                 if (cat == 'a')
1431                 {
1432                     attribute_t *p_attr = MakeAttribute (data);
1433                     if (p_attr == NULL)
1434                         goto error;
1435
1436                     TAB_APPEND (p_sdp->mediav[p_sdp->mediac - 1].i_attributes,
1437                                 p_sdp->mediav[p_sdp->mediac - 1].pp_attributes, p_attr);
1438                     break;
1439                 }
1440
1441                 if (cat == 'm')
1442                     goto media;
1443
1444                 if (cat != 'm')
1445                 {
1446                     msg_Dbg (p_obj, "unexpected SDP line: 0x%02x", (int)cat);
1447                     goto error;
1448                 }
1449                 break;
1450
1451             default:
1452                 msg_Err (p_obj, "*** BUG in SDP parser! ***");
1453                 goto error;
1454         }
1455     }
1456
1457     return p_sdp;
1458
1459 error:
1460     FreeSDP (p_sdp);
1461     return NULL;
1462 }
1463
1464 static int InitSocket( services_discovery_t *p_sd, const char *psz_address,
1465                        int i_port )
1466 {
1467     int i_fd = net_ListenUDP1 ((vlc_object_t *)p_sd, psz_address, i_port);
1468     if (i_fd == -1)
1469         return VLC_EGENERIC;
1470
1471     shutdown( i_fd, SHUT_WR );
1472     INSERT_ELEM (p_sd->p_sys->pi_fd, p_sd->p_sys->i_fd,
1473                  p_sd->p_sys->i_fd, i_fd);
1474     return VLC_SUCCESS;
1475 }
1476
1477 static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i_len )
1478 {
1479 #ifdef HAVE_ZLIB_H
1480     int i_result, i_dstsize, n = 0;
1481     unsigned char *psz_dst = NULL;
1482     z_stream d_stream;
1483
1484     memset (&d_stream, 0, sizeof (d_stream));
1485
1486     i_result = inflateInit(&d_stream);
1487     if( i_result != Z_OK )
1488         return( -1 );
1489
1490     d_stream.next_in = (Bytef *)psz_src;
1491     d_stream.avail_in = i_len;
1492
1493     do
1494     {
1495         n++;
1496         psz_dst = (unsigned char *)realloc( psz_dst, n * 1000 );
1497         d_stream.next_out = (Bytef *)&psz_dst[(n - 1) * 1000];
1498         d_stream.avail_out = 1000;
1499
1500         i_result = inflate(&d_stream, Z_NO_FLUSH);
1501         if( ( i_result != Z_OK ) && ( i_result != Z_STREAM_END ) )
1502         {
1503             inflateEnd( &d_stream );
1504             free( psz_dst );
1505             return( -1 );
1506         }
1507     }
1508     while( ( d_stream.avail_out == 0 ) && ( d_stream.avail_in != 0 ) &&
1509            ( i_result != Z_STREAM_END ) );
1510
1511     i_dstsize = d_stream.total_out;
1512     inflateEnd( &d_stream );
1513
1514     *_dst = (unsigned char *)realloc( psz_dst, i_dstsize );
1515
1516     return i_dstsize;
1517 #else
1518     (void)psz_src;
1519     (void)_dst;
1520     (void)i_len;
1521     return -1;
1522 #endif
1523 }
1524
1525
1526 static void FreeSDP( sdp_t *p_sdp )
1527 {
1528     free( p_sdp->psz_sessionname );
1529     free( p_sdp->psz_sessioninfo );
1530     free( p_sdp->psz_uri );
1531
1532     for (unsigned j = 0; j < p_sdp->mediac; j++)
1533     {
1534         free (p_sdp->mediav[j].fmt);
1535         for (int i = 0; i < p_sdp->mediav[j].i_attributes; i++)
1536             FreeAttribute (p_sdp->mediav[j].pp_attributes[i]);
1537         free (p_sdp->mediav[j].pp_attributes);
1538     }
1539     free (p_sdp->mediav);
1540
1541     for (int i = 0; i < p_sdp->i_attributes; i++)
1542         FreeAttribute (p_sdp->pp_attributes[i]);
1543
1544     free (p_sdp->pp_attributes);
1545     free (p_sdp);
1546 }
1547
1548 static int RemoveAnnounce( services_discovery_t *p_sd,
1549                            sap_announce_t *p_announce )
1550 {
1551     int i;
1552
1553     if( p_announce->p_sdp )
1554     {
1555         FreeSDP( p_announce->p_sdp );
1556         p_announce->p_sdp = NULL;
1557     }
1558
1559     if( p_announce->p_item )
1560     {
1561         services_discovery_RemoveItem( p_sd, p_announce->p_item );
1562         vlc_gc_decref( p_announce->p_item );
1563         p_announce->p_item = NULL;
1564     }
1565
1566     for( i = 0; i< p_sd->p_sys->i_announces; i++)
1567     {
1568         if( p_sd->p_sys->pp_announces[i] == p_announce )
1569         {
1570             REMOVE_ELEM( p_sd->p_sys->pp_announces, p_sd->p_sys->i_announces,
1571                          i);
1572             break;
1573         }
1574     }
1575
1576     free( p_announce );
1577
1578     return VLC_SUCCESS;
1579 }
1580
1581 /*
1582  * Compare two sessions, when hash is not set (SAP v0)
1583  */
1584 static bool IsSameSession( sdp_t *p_sdp1, sdp_t *p_sdp2 )
1585 {
1586     /* A session is identified by
1587      * - username,
1588      * - session_id,
1589      * - network type (which is always IN),
1590      * - address type (currently, this means IP version),
1591      * - and hostname.
1592      */
1593     if (strcmp (p_sdp1->username, p_sdp2->username)
1594      || (p_sdp1->session_id != p_sdp2->session_id)
1595      || (p_sdp1->orig_ip_version != p_sdp2->orig_ip_version)
1596      || strcmp (p_sdp1->orig_host, p_sdp2->orig_host))
1597         return false;
1598
1599     return true;
1600 }
1601
1602 static inline attribute_t *MakeAttribute (const char *str)
1603 {
1604     attribute_t *a = malloc (sizeof (*a) + strlen (str) + 1);
1605     if (a == NULL)
1606         return NULL;
1607
1608     strcpy (a->name, str);
1609     EnsureUTF8 (a->name);
1610     char *value = strchr (a->name, ':');
1611     if (value != NULL)
1612     {
1613         *value++ = '\0';
1614         a->value = value;
1615     }
1616     else
1617         a->value = "";
1618     return a;
1619 }
1620
1621
1622 static const char *GetAttribute (attribute_t **tab, unsigned n,
1623                                  const char *name)
1624 {
1625     for (unsigned i = 0; i < n; i++)
1626         if (strcasecmp (tab[i]->name, name) == 0)
1627             return tab[i]->value;
1628     return NULL;
1629 }
1630
1631
1632 static inline void FreeAttribute (attribute_t *a)
1633 {
1634     free (a);
1635 }