]> git.sesse.net Git - vlc/blob - modules/services_discovery/sap.c
- win32 replacements for inet_pton/inet_ntop
[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 #define _GNU_SOURCE
30 #include <stdlib.h>                                      /* malloc(), free() */
31
32 #include <vlc/vlc.h>
33 #include <vlc_playlist.h>
34 #include <vlc_demux.h>
35
36 #include <vlc_network.h>
37 #include <vlc_charset.h>
38
39 #include <ctype.h>
40 #include <errno.h>
41
42 #ifdef HAVE_UNISTD_H
43 #    include <unistd.h>
44 #endif
45 #ifdef HAVE_SYS_TIME_H
46 #    include <sys/time.h>
47 #endif
48
49 #ifdef HAVE_ZLIB_H
50 #   include <zlib.h>
51 #endif
52
53 /************************************************************************
54  * Macros and definitions
55  ************************************************************************/
56
57 #define MAX_LINE_LENGTH 256
58
59 /* SAP is always on that port */
60 #define SAP_PORT 9875
61 /* Global-scope SAP address */
62 #define SAP_V4_GLOBAL_ADDRESS   "224.2.127.254"
63 /* Organization-local SAP address */
64 #define SAP_V4_ORG_ADDRESS      "239.195.255.255"
65 /* Local (smallest non-link-local scope) SAP address */
66 #define SAP_V4_LOCAL_ADDRESS    "239.255.255.255"
67 /* Link-local SAP address */
68 #define SAP_V4_LINK_ADDRESS     "224.0.0.255"
69 #define ADD_SESSION 1
70
71 #define SAP_V6_1 "FF0"
72 /* Scope is inserted between them */
73 #define SAP_V6_2 "::2:7FFE"
74 /* See RFC3513 for list of valid scopes */
75 /* FIXME: find a way to listen to link-local scope */
76 static const char ipv6_scopes[] = "1456789ABCDE";
77
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_IPV4_TEXT N_( "IPv4 SAP" )
87 #define SAP_IPV4_LONGTEXT N_( \
88       "Listen to IPv4 announcements " \
89       "on the standard address." )
90 #define SAP_IPV6_TEXT N_( "IPv6 SAP" )
91 #define SAP_IPV6_LONGTEXT N_( \
92       "Listen to IPv6 announcements " \
93       "on the standard addresses." )
94 #define SAP_SCOPE_TEXT N_( "IPv6 SAP scope" )
95 #define SAP_SCOPE_LONGTEXT N_( \
96        "Scope for IPv6 announcements (default is 8)." )
97 #define SAP_TIMEOUT_TEXT N_( "SAP timeout (seconds)" )
98 #define SAP_TIMEOUT_LONGTEXT N_( \
99        "Delay after which SAP items get deleted if no new announcement " \
100        "is received." )
101 #define SAP_PARSE_TEXT N_( "Try to parse the announce" )
102 #define SAP_PARSE_LONGTEXT N_( \
103        "This enables actual parsing of the announces by the SAP module. " \
104        "Otherwise, all announcements are parsed by the \"livedotcom\" " \
105        "(RTP/RTSP) module." )
106 #define SAP_STRICT_TEXT N_( "SAP Strict mode" )
107 #define SAP_STRICT_LONGTEXT N_( \
108        "When this is set, the SAP parser will discard some non-compliant " \
109        "announcements." )
110 #define SAP_CACHE_TEXT N_("Use SAP cache")
111 #define SAP_CACHE_LONGTEXT N_( \
112        "This enables a SAP caching mechanism. " \
113        "This will result in lower SAP startup time, but you could end up " \
114        "with items corresponding to legacy streams." )
115 #define SAP_TIMESHIFT_TEXT N_("Allow timeshifting")
116 #define SAP_TIMESHIFT_LONGTEXT N_( "This automatically enables timeshifting " \
117         "for streams discovered through SAP announcements." )
118
119 /* Callbacks */
120     static int  Open ( vlc_object_t * );
121     static void Close( vlc_object_t * );
122     static int  OpenDemux ( vlc_object_t * );
123     static void CloseDemux ( vlc_object_t * );
124
125 vlc_module_begin();
126     set_shortname( _("SAP"));
127     set_description( _("SAP Announcements") );
128     set_category( CAT_PLAYLIST );
129     set_subcategory( SUBCAT_PLAYLIST_SD );
130
131     add_string( "sap-addr", NULL, NULL,
132                 SAP_ADDR_TEXT, SAP_ADDR_LONGTEXT, VLC_TRUE );
133     add_bool( "sap-ipv4", 1 , NULL,
134                SAP_IPV4_TEXT,SAP_IPV4_LONGTEXT, VLC_TRUE );
135     add_bool( "sap-ipv6", 1 , NULL,
136               SAP_IPV6_TEXT, SAP_IPV6_LONGTEXT, VLC_TRUE );
137     add_integer( "sap-timeout", 1800, NULL,
138                  SAP_TIMEOUT_TEXT, SAP_TIMEOUT_LONGTEXT, VLC_TRUE );
139     add_bool( "sap-parse", 1 , NULL,
140                SAP_PARSE_TEXT,SAP_PARSE_LONGTEXT, VLC_TRUE );
141     add_bool( "sap-strict", 0 , NULL,
142                SAP_STRICT_TEXT,SAP_STRICT_LONGTEXT, VLC_TRUE );
143 #if 0
144     add_bool( "sap-cache", 0 , NULL,
145                SAP_CACHE_TEXT,SAP_CACHE_LONGTEXT, VLC_TRUE );
146 #endif
147     add_bool( "sap-timeshift", 0 , NULL,
148               SAP_TIMESHIFT_TEXT,SAP_TIMESHIFT_LONGTEXT, VLC_TRUE );
149
150     set_capability( "services_discovery", 0 );
151     set_callbacks( Open, Close );
152
153     add_submodule();
154         set_description( _("SDP file parser for UDP") );
155         add_shortcut( "sdp" );
156         set_capability( "demux2", 51 );
157         set_callbacks( OpenDemux, CloseDemux );
158 vlc_module_end();
159
160
161 /*****************************************************************************
162  * Local structures
163  *****************************************************************************/
164
165 typedef struct sdp_t sdp_t;
166 typedef struct attribute_t attribute_t;
167 typedef struct sap_announce_t sap_announce_t;
168
169
170 struct sdp_media_t
171 {
172     struct sdp_t           *parent;
173     char                   *fmt;
174     struct sockaddr_storage addr;
175     socklen_t               addrlen;
176     unsigned                n_addr;
177     int           i_attributes;
178     attribute_t  **pp_attributes;
179 };
180
181
182 /* The structure that contains sdp information */
183 struct  sdp_t
184 {
185     char *psz_sdp;
186
187     /* o field */
188     char     username[64];
189     uint64_t session_id;
190     uint64_t session_version;
191     unsigned orig_ip_version;
192     char     orig_host[1024];
193
194     /* s= field */
195     char *psz_sessionname;
196
197     /* old cruft */
198     /* "computed" URI */
199     char *psz_uri;
200     int           i_media_type;
201
202     /* a= global attributes */
203     int           i_attributes;
204     attribute_t  **pp_attributes;
205
206     /* medias (well, we only support one atm) */
207     unsigned            mediac;
208     struct sdp_media_t  mediav[1];
209 };
210
211 struct attribute_t
212 {
213     const char *value;
214     char name[0];
215 };
216
217 struct sap_announce_t
218 {
219     mtime_t i_last;
220
221     uint16_t    i_hash;
222     uint32_t    i_source[4];
223
224     /* SAP annnounces must only contain one SDP */
225     sdp_t       *p_sdp;
226
227     int i_input_id;
228     int i_item_id_cat;
229     int i_item_id_one;
230 };
231
232 struct services_discovery_sys_t
233 {
234     /* Socket descriptors */
235     int i_fd;
236     int *pi_fd;
237
238     /* playlist node */
239     playlist_item_t *p_node_cat;
240     playlist_item_t *p_node_one;
241
242     /* Table of announces */
243     int i_announces;
244     struct sap_announce_t **pp_announces;
245
246     /* Modes */
247     vlc_bool_t  b_strict;
248     vlc_bool_t  b_parse;
249     vlc_bool_t  b_timeshift;
250
251     int i_timeout;
252 };
253
254 struct demux_sys_t
255 {
256     sdp_t *p_sdp;
257 };
258
259 /*****************************************************************************
260  * Local prototypes
261  *****************************************************************************/
262
263
264 /* Main functions */
265     static int Demux( demux_t *p_demux );
266     static int Control( demux_t *, int, va_list );
267     static void Run    ( services_discovery_t *p_sd );
268
269 /* Main parsing functions */
270     static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp );
271     static int ParseSAP( services_discovery_t *p_sd, const uint8_t *p_buffer, size_t i_read );
272     static sdp_t *ParseSDP (vlc_object_t *p_sd, const char *psz_sdp);
273     static sap_announce_t *CreateAnnounce( services_discovery_t *, uint16_t, sdp_t * );
274     static int RemoveAnnounce( services_discovery_t *p_sd, sap_announce_t *p_announce );
275
276 /* Helper functions */
277     static inline attribute_t *MakeAttribute (const char *str);
278     static const char *GetAttribute (attribute_t **tab, unsigned n, const char *name);
279     static inline void FreeAttribute (attribute_t *a);
280
281     static vlc_bool_t IsSameSession( sdp_t *p_sdp1, sdp_t *p_sdp2 );
282     static int InitSocket( services_discovery_t *p_sd, const char *psz_address, int i_port );
283     static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i_len );
284     static void FreeSDP( sdp_t *p_sdp );
285
286 /*****************************************************************************
287  * Open: initialize and create stuff
288  *****************************************************************************/
289 static int Open( vlc_object_t *p_this )
290 {
291     services_discovery_t *p_sd = ( services_discovery_t* )p_this;
292     services_discovery_sys_t *p_sys  = (services_discovery_sys_t *)
293                                 malloc( sizeof( services_discovery_sys_t ) );
294
295     p_sys->i_timeout = var_CreateGetInteger( p_sd, "sap-timeout" );
296
297     p_sd->pf_run = Run;
298     p_sd->p_sys  = p_sys;
299
300     p_sys->pi_fd = NULL;
301     p_sys->i_fd = 0;
302
303     p_sys->b_strict = var_CreateGetInteger( p_sd, "sap-strict");
304     p_sys->b_parse = var_CreateGetInteger( p_sd, "sap-parse" );
305
306 #if 0
307     if( var_CreateGetInteger( p_sd, "sap-cache" ) )
308     {
309         CacheLoad( p_sd );
310     }
311 #endif
312
313     /* Cache sap_timeshift value */
314     p_sys->b_timeshift = var_CreateGetInteger( p_sd, "sap-timeshift" )
315             ? VLC_TRUE : VLC_FALSE;
316
317     /* Create our playlist node */
318     pl_Yield( p_sd );
319
320     playlist_NodesPairCreate( pl_Get( p_sd ), _("SAP sessions"),
321                               &p_sys->p_node_cat, &p_sys->p_node_one,
322                               VLC_TRUE );
323     p_sys->p_node_cat->p_input->b_prefers_tree = VLC_TRUE;
324     p_sys->i_announces = 0;
325     p_sys->pp_announces = NULL;
326
327     return VLC_SUCCESS;
328 }
329
330 /*****************************************************************************
331  * OpenDemux: initialize and create stuff
332  *****************************************************************************/
333 static int OpenDemux( vlc_object_t *p_this )
334 {
335     demux_t *p_demux = (demux_t *)p_this;
336     uint8_t *p_peek;
337     int i_max_sdp = 1024;
338     int i_sdp = 0;
339     char *psz_sdp = NULL;
340     sdp_t *p_sdp = NULL;
341
342     if( !var_CreateGetInteger( p_demux, "sap-parse" ) )
343     {
344         /* We want livedotcom module to parse this SDP file */
345         return VLC_EGENERIC;
346     }
347
348     /* Probe for SDP */
349     if( p_demux->s )
350     {
351         if( stream_Peek( p_demux->s, &p_peek, 7 ) < 7 ) return VLC_EGENERIC;
352
353         if( strncmp( (char*)p_peek, "v=0\r\n", 5 ) &&
354             strncmp( (char*)p_peek, "v=0\n", 4 ) &&
355             ( p_peek[0] < 'a' || p_peek[0] > 'z' || p_peek[1] != '=' ) )
356         {
357             return VLC_EGENERIC;
358         }
359     }
360
361     psz_sdp = (char *)malloc( i_max_sdp );
362     if( !psz_sdp ) return VLC_EGENERIC;
363
364     /* Gather the complete sdp file */
365     for( ;; )
366     {
367         int i_read = stream_Read( p_demux->s,
368                                   &psz_sdp[i_sdp], i_max_sdp - i_sdp - 1 );
369
370         if( i_read < 0 )
371         {
372             msg_Err( p_demux, "failed to read SDP" );
373             goto error;
374         }
375
376         i_sdp += i_read;
377
378         if( i_read < i_max_sdp - i_sdp - 1 )
379         {
380             psz_sdp[i_sdp] = '\0';
381             break;
382         }
383
384         i_max_sdp += 1000;
385         psz_sdp = (char *)realloc( psz_sdp, i_max_sdp );
386     }
387
388     p_sdp = ParseSDP( VLC_OBJECT(p_demux), psz_sdp );
389
390     if( !p_sdp )
391     {
392         msg_Warn( p_demux, "invalid SDP");
393         goto error;
394     }
395
396     if( ParseConnection( VLC_OBJECT( p_demux ), p_sdp ) )
397     {
398         p_sdp->psz_uri = NULL;
399     }
400     if( p_sdp->i_media_type != 33 && p_sdp->i_media_type != 32 &&
401         p_sdp->i_media_type != 14 )
402         goto error;
403
404     if( p_sdp->psz_uri == NULL ) goto error;
405
406     p_demux->p_sys = (demux_sys_t *)malloc( sizeof(demux_sys_t) );
407     p_demux->p_sys->p_sdp = p_sdp;
408     p_demux->pf_control = Control;
409     p_demux->pf_demux = Demux;
410
411     FREENULL( psz_sdp );
412     return VLC_SUCCESS;
413
414 error:
415     FREENULL( psz_sdp );
416     if( p_sdp ) FreeSDP( p_sdp ); p_sdp = NULL;
417     stream_Seek( p_demux->s, 0 );
418     return VLC_EGENERIC;
419 }
420
421 /*****************************************************************************
422  * Close:
423  *****************************************************************************/
424 static void Close( vlc_object_t *p_this )
425 {
426     services_discovery_t *p_sd = ( services_discovery_t* )p_this;
427     services_discovery_sys_t    *p_sys  = p_sd->p_sys;
428
429     int i;
430
431     for( i = p_sys->i_fd-1 ; i >= 0 ; i-- )
432     {
433         net_Close( p_sys->pi_fd[i] );
434     }
435     FREENULL( p_sys->pi_fd );
436
437 #if 0
438     if( config_GetInt( p_sd, "sap-cache" ) )
439     {
440         CacheSave( p_sd );
441     }
442 #endif
443
444     for( i = p_sys->i_announces  - 1;  i>= 0; i-- )
445     {
446         RemoveAnnounce( p_sd, p_sys->pp_announces[i] );
447     }
448     FREENULL( p_sys->pp_announces );
449
450     playlist_NodeDelete( pl_Get(p_sd), p_sys->p_node_cat, VLC_TRUE,
451                          VLC_TRUE );
452     playlist_NodeDelete( pl_Get(p_sd), p_sys->p_node_one, VLC_TRUE,
453                          VLC_TRUE );
454     pl_Release( p_sd );
455     free( p_sys );
456 }
457
458 /*****************************************************************************
459  * CloseDemux: Close the demuxer
460  *****************************************************************************/
461 static void CloseDemux( vlc_object_t *p_this )
462 {
463     demux_t *p_demux = (demux_t *)p_this;
464     if( p_demux->p_sys )
465     {
466         if( p_demux->p_sys->p_sdp ) { FreeSDP( p_demux->p_sys->p_sdp ); p_demux->p_sys->p_sdp = NULL; }
467         free( p_demux->p_sys );
468     }
469 }
470
471 /*****************************************************************************
472  * Run: main SAP thread
473  *****************************************************************************
474  * Listens to SAP packets, and sends them to packet_handle
475  *****************************************************************************/
476 #define MAX_SAP_BUFFER 5000
477
478 static void Run( services_discovery_t *p_sd )
479 {
480     char *psz_addr;
481     int i;
482
483     /* Braindead Winsock DNS resolver will get stuck over 2 seconds per failed
484      * DNS queries, even if the DNS server returns an error with milliseconds.
485      * You don't want to know why the bug (as of XP SP2) wasn't fixed since
486      * Winsock 1.1 from Windows 95, if not Windows 3.1.
487      * Anyway, to avoid a 30 seconds delay for failed IPv6 socket creation,
488      * we have to open sockets in Run() rather than Open(). */
489     if( var_CreateGetInteger( p_sd, "sap-ipv4" ) )
490     {
491         InitSocket( p_sd, SAP_V4_GLOBAL_ADDRESS, SAP_PORT );
492         InitSocket( p_sd, SAP_V4_ORG_ADDRESS, SAP_PORT );
493         InitSocket( p_sd, SAP_V4_LOCAL_ADDRESS, SAP_PORT );
494         InitSocket( p_sd, SAP_V4_LINK_ADDRESS, SAP_PORT );
495     }
496     if( var_CreateGetInteger( p_sd, "sap-ipv6" ) )
497     {
498         char psz_address[] = SAP_V6_1"0"SAP_V6_2;
499         const char *c_scope;
500
501         for( c_scope = ipv6_scopes; *c_scope; c_scope++ )
502         {
503             psz_address[sizeof(SAP_V6_1) - 1] = *c_scope;
504             InitSocket( p_sd, psz_address, SAP_PORT );
505         }
506     }
507
508     psz_addr = var_CreateGetString( p_sd, "sap-addr" );
509     if( psz_addr && *psz_addr )
510     {
511         InitSocket( p_sd, psz_addr, SAP_PORT );
512         free( psz_addr );
513     }
514
515     if( p_sd->p_sys->i_fd == 0 )
516     {
517         msg_Err( p_sd, "unable to listen on any address" );
518         return;
519     }
520
521     /* read SAP packets */
522     while( !p_sd->b_die )
523     {
524         int i_read;
525         uint8_t p_buffer[MAX_SAP_BUFFER+1];
526
527         i_read = net_Select( p_sd, p_sd->p_sys->pi_fd,
528                              p_sd->p_sys->i_fd, p_buffer,
529                              MAX_SAP_BUFFER, 500000 );
530
531         /* Check for items that need deletion */
532         for( i = 0; i < p_sd->p_sys->i_announces; i++ )
533         {
534             mtime_t i_timeout = ( mtime_t ) 1000000 * p_sd->p_sys->i_timeout;
535
536             if( mdate() - p_sd->p_sys->pp_announces[i]->i_last > i_timeout )
537             {
538                 RemoveAnnounce( p_sd, p_sd->p_sys->pp_announces[i] );
539             }
540         }
541
542         /* Minimum length is > 6 */
543         if( i_read <= 6 )
544         {
545             if( i_read < 0 )
546             {
547                 msg_Warn( p_sd, "socket read error" );
548             }
549             continue;
550         }
551
552         p_buffer[i_read] = '\0';
553
554         /* Parse the packet */
555         ParseSAP( p_sd, p_buffer, i_read );
556     }
557 }
558
559 /**********************************************************************
560  * Demux: reads and demuxes data packets
561  * Return -1 if error, 0 if EOF, 1 else
562  **********************************************************************/
563 static int Demux( demux_t *p_demux )
564 {
565     sdp_t *p_sdp = p_demux->p_sys->p_sdp;
566     input_thread_t *p_input;
567     input_item_t *p_parent_input;
568
569     playlist_t *p_playlist = pl_Yield( p_demux );
570     p_input = (input_thread_t *)vlc_object_find( p_demux, VLC_OBJECT_INPUT,
571                                                  FIND_PARENT );
572     assert( p_input );
573     if( !p_input )
574     {
575         msg_Err( p_demux, "parent input could not be found" );
576         return VLC_EGENERIC;
577     }
578
579     p_parent_input = input_GetItem(p_input);
580
581     vlc_mutex_lock( &p_parent_input->lock );
582     FREENULL( p_parent_input->psz_uri );
583     p_parent_input->psz_uri = strdup( p_sdp->psz_uri );
584     FREENULL( p_parent_input->psz_name );
585     p_parent_input->psz_name = strdup( p_sdp->psz_sessionname );
586     p_parent_input->i_type = ITEM_TYPE_NET;
587
588     if( p_playlist->status.p_item &&
589              p_playlist->status.p_item->p_input == p_parent_input )
590     {
591         playlist_Control( p_playlist, PLAYLIST_VIEWPLAY, VLC_TRUE,
592                           p_playlist->status.p_node, p_playlist->status.p_item );
593     }
594
595     vlc_mutex_unlock( &p_parent_input->lock );
596     vlc_object_release( p_input );
597     vlc_object_release( p_playlist );
598
599     return VLC_SUCCESS;
600 }
601
602 static int Control( demux_t *p_demux, int i_query, va_list args )
603 {
604     return VLC_EGENERIC;
605 }
606
607 /**************************************************************
608  * Local functions
609  **************************************************************/
610
611 /* i_read is at least > 6 */
612 static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
613                      size_t len )
614 {
615     int i;
616     const char          *psz_sdp;
617     const uint8_t *end = buf + len;
618     sdp_t               *p_sdp;
619
620     assert (buf[len] == '\0');
621
622     if (len < 4)
623         return VLC_EGENERIC;
624
625     uint8_t flags = buf[0];
626
627     /* First, check the sap announce is correct */
628     if ((flags >> 5) != 1)
629         return VLC_EGENERIC;
630
631     vlc_bool_t b_ipv6 = (flags & 0x10) != 0;
632     vlc_bool_t b_need_delete = (flags & 0x04) != 0;
633
634     if (flags & 0x02)
635     {
636         msg_Dbg( p_sd, "encrypted packet, unsupported" );
637         return VLC_EGENERIC;
638     }
639
640     vlc_bool_t b_compressed = (flags & 0x01) != 0;
641
642     uint16_t i_hash = U16_AT (buf + 2);
643
644     if( p_sd->p_sys->b_strict && i_hash == 0 )
645     {
646         msg_Dbg( p_sd, "strict mode, discarding announce with null id hash");
647         return VLC_EGENERIC;
648     }
649
650     // Skips source address and auth data
651     buf += 4 + (b_ipv6 ? 16 : 4) + buf[1];
652     if (buf > end)
653         return VLC_EGENERIC;
654
655     uint8_t *decomp = NULL;
656     if( b_compressed )
657     {
658         int newsize = Decompress (buf, &decomp, end - buf);
659         if (newsize < 0)
660         {
661             msg_Dbg( p_sd, "decompression of SAP packet failed" );
662             return VLC_EGENERIC;
663         }
664
665         decomp = realloc (decomp, newsize + 1);
666         decomp[newsize] = '\0';
667         psz_sdp = (const char *)decomp;
668         len = newsize;
669     }
670     else
671     {
672         psz_sdp = (const char *)buf;
673         len = end - buf;
674     }
675
676     /* len is a strlen here here. both buf and decomp are len+1 where the 1 should be a \0 */
677     assert( psz_sdp[len] == '\0');
678
679     /* Skip payload type */
680     /* SAPv1 has implicit "application/sdp" payload type: first line is v=0 */
681     if (strncmp (psz_sdp, "v=0", 3))
682     {
683         size_t clen = strlen (psz_sdp) + 1;
684
685         if (strcmp (psz_sdp, "application/sdp"))
686         {
687             msg_Dbg (p_sd, "unsupported content type: %s", psz_sdp);
688             return VLC_EGENERIC;
689         }
690
691         // skips content type
692         if (len <= clen)
693             return VLC_EGENERIC;
694
695         len -= clen;
696         psz_sdp += clen;
697     }
698
699     /* Parse SDP info */
700     p_sdp = ParseSDP( VLC_OBJECT(p_sd), psz_sdp );
701
702     if( p_sdp == NULL )
703         return VLC_EGENERIC;
704
705     p_sdp->psz_sdp = psz_sdp;
706
707     /* Decide whether we should add a playlist item for this SDP */
708     /* Parse connection information (c= & m= ) */
709     if( ParseConnection( VLC_OBJECT(p_sd), p_sdp ) )
710         p_sdp->psz_uri = NULL;
711
712     /* Multi-media or no-parse -> pass to LIVE.COM */
713     if( ( p_sdp->i_media_type != 14
714        && p_sdp->i_media_type != 32
715        && p_sdp->i_media_type != 33)
716      || p_sd->p_sys->b_parse == VLC_FALSE )
717     {
718         free( p_sdp->psz_uri );
719         if (asprintf( &p_sdp->psz_uri, "sdp://%s", p_sdp->psz_sdp ) == -1)
720             p_sdp->psz_uri = NULL;
721     }
722
723     if( p_sdp->psz_uri == NULL ) return VLC_EGENERIC;
724
725     for( i = 0 ; i< p_sd->p_sys->i_announces ; i++ )
726     {
727         /* FIXME: slow */
728         /* FIXME: we create a new announce each time the sdp changes */
729         if( IsSameSession( p_sd->p_sys->pp_announces[i]->p_sdp, p_sdp ) )
730         {
731             if( b_need_delete )
732             {
733                 RemoveAnnounce( p_sd, p_sd->p_sys->pp_announces[i]);
734             }
735             else
736             {
737                 p_sd->p_sys->pp_announces[i]->i_last = mdate();
738             }
739             FreeSDP( p_sdp ); p_sdp = NULL;
740             return VLC_SUCCESS;
741         }
742     }
743
744     CreateAnnounce( p_sd, i_hash, p_sdp );
745
746     FREENULL (decomp);
747     return VLC_SUCCESS;
748 }
749
750 sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
751                                 sdp_t *p_sdp )
752 {
753     input_item_t *p_input;
754     playlist_item_t     *p_item, *p_child;
755     const char *psz_value;
756     sap_announce_t *p_sap = (sap_announce_t *)malloc(
757                                         sizeof(sap_announce_t ) );
758     services_discovery_sys_t *p_sys;
759     if( p_sap == NULL )
760         return NULL;
761
762     p_sys = p_sd->p_sys;
763
764     p_sap->i_last = mdate();
765     p_sap->i_hash = i_hash;
766     p_sap->p_sdp = p_sdp;
767
768     /* Create the actual playlist item here */
769     p_input = input_ItemNewWithType( VLC_OBJECT(p_sd),
770                                      p_sap->p_sdp->psz_uri,
771                                      p_sdp->psz_sessionname,
772                                      0, NULL, -1, ITEM_TYPE_NET );
773     p_sap->i_input_id = p_input->i_id;
774     if( !p_input )
775     {
776         free( p_sap );
777         return NULL;
778     }
779
780     if( p_sys->b_timeshift )
781         input_ItemAddOption( p_input, ":access-filter=timeshift" );
782
783     psz_value = GetAttribute( p_sap->p_sdp->pp_attributes, p_sap->p_sdp->i_attributes, "tool" );
784     if( psz_value != NULL )
785     {
786         input_ItemAddInfo( p_input, _("Session"),_("Tool"), psz_value );
787     }
788     if( strcmp( p_sdp->username, "-" ) )
789     {
790         input_ItemAddInfo( p_input, _("Session"),
791                                 _("User"), p_sdp->username );
792     }
793
794     /* Handle group */
795     psz_value = GetAttribute( p_sap->p_sdp->pp_attributes, p_sap->p_sdp->i_attributes, "x-plgroup" );
796     if( psz_value == NULL )
797         psz_value = GetAttribute( p_sap->p_sdp->pp_attributes, p_sap->p_sdp->i_attributes, "plgroup" );
798
799     if( psz_value != NULL )
800     {
801         p_child = playlist_ChildSearchName( p_sys->p_node_cat, psz_value );
802
803         if( p_child == NULL )
804         {
805             p_child = playlist_NodeCreate( pl_Get( p_sd ), psz_value,
806                                            p_sys->p_node_cat, 0 );
807             p_child->i_flags &= ~PLAYLIST_SKIP_FLAG;
808         }
809     }
810     else
811     {
812         p_child = p_sys->p_node_cat;
813     }
814
815     p_item = playlist_NodeAddInput( pl_Get( p_sd ), p_input, p_child,
816                                     PLAYLIST_APPEND, PLAYLIST_END );
817     p_item->i_flags &= ~PLAYLIST_SKIP_FLAG;
818     p_item->i_flags &= ~PLAYLIST_SAVE_FLAG;
819     p_sap->i_item_id_cat = p_item->i_id;
820
821     p_item = playlist_NodeAddInput( pl_Get( p_sd ), p_input,
822                         p_sys->p_node_one, PLAYLIST_APPEND, PLAYLIST_END );
823     p_item->i_flags &= ~PLAYLIST_SKIP_FLAG;
824     p_item->i_flags &= ~PLAYLIST_SAVE_FLAG;
825     p_sap->i_item_id_one = p_item->i_id;
826
827     TAB_APPEND( p_sys->i_announces, p_sys->pp_announces, p_sap );
828
829     return p_sap;
830 }
831
832 /* Fill p_sdp->psz_uri */
833 static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
834 {
835     if (p_sdp->mediac != 1)
836         return VLC_EGENERIC;
837
838     char psz_uri[1026];
839     const char *host;
840     int port;
841
842     psz_uri[0] = '[';
843     if (vlc_getnameinfo ((struct sockaddr *)&(p_sdp->mediav[0].addr),
844                          p_sdp->mediav[0].addrlen, psz_uri + 1,
845                          sizeof (psz_uri) - 2, &port, NI_NUMERICHOST))
846         return VLC_EGENERIC;
847
848     if (strchr (psz_uri + 1, ':'))
849     {
850         host = psz_uri;
851         psz_uri[strlen (psz_uri)] = ']';
852     }
853     else
854         host = psz_uri + 1;
855
856     /* Parse m= field */
857     char *sdp_proto = strdup (p_sdp->mediav[0].fmt);
858     if (sdp_proto == NULL)
859         return VLC_ENOMEM;
860
861     char *subtype = strchr (sdp_proto, ' ');
862     if (sdp_proto == NULL)
863     {
864         msg_Dbg (p_obj, "missing SDP media subtype: %s", sdp_proto);
865         p_sdp->i_media_type = 0;
866     }
867     else
868     {
869         *subtype++ = '\0';
870         p_sdp->i_media_type = atoi (subtype);
871     }
872     if (p_sdp->i_media_type == 0)
873          p_sdp->i_media_type = 33;
874
875     static const char proto_match[] =
876         "udp\0"             "udp\0"
877         "RTP/AVP\0"         "rtp\0"
878         "UDPLite/RTP/AVP\0" "udplite\0"
879         "DCCP/RTP/AVP\0"    "dccp\0"
880         "TCP/RTP/AVP\0"     "tcp\0"
881         "\0";
882
883     const char *vlc_proto = NULL;
884     for (const char *proto = proto_match; *proto;)
885     {
886         if (strcasecmp (proto, sdp_proto))
887         {
888             vlc_proto = proto + strlen (proto) + 1;
889             break;
890         }
891         proto += strlen (proto) + 1;
892         proto += strlen (proto) + 1;
893     }
894
895     free (sdp_proto);
896     if (vlc_proto == NULL)
897     {
898         msg_Dbg (p_obj, "unknown SDP media protocol: %s",
899                  p_sdp->mediav[0].fmt);
900         return VLC_EGENERIC;
901     }
902
903     /* handle SSM case */
904     const char *sfilter = GetAttribute( p_sdp->mediav[0].pp_attributes,
905                                         p_sdp->mediav[0].i_attributes,
906                                         "source-filter" );
907     if (sfilter == NULL)
908         sfilter = GetAttribute( p_sdp->pp_attributes, p_sdp->i_attributes,
909                                 "source-filter" );
910
911     char psz_source[258] = "";
912     if (sfilter != NULL)
913     {
914         char psz_source_ip[256];
915
916         if (sscanf (sfilter, " incl IN IP%*c %*s %255s ", psz_source_ip) == 1)
917         {
918             if (strchr (psz_source_ip, ':') != NULL)
919                 sprintf (psz_source, "[%s]", psz_source_ip);
920             else
921                 strcpy (psz_source, psz_source_ip);
922         }
923     }
924
925     asprintf( &p_sdp->psz_uri, "%s://%s@%s:%i", vlc_proto, psz_source,
926               host, port );
927
928     return VLC_SUCCESS;
929 }
930
931
932 static int ParseSDPConnection (const char *str, struct sockaddr_storage *addr,
933                                socklen_t *addrlen, unsigned *number)
934 {
935     char host[60];
936     unsigned fam, n1, n2;
937
938     int res = sscanf (str, "IN IP%u %59[^/]/%u/%u", &fam, host, &n1, &n2);
939     if (res < 2)
940         return -1;
941
942     switch (fam)
943     {
944 #ifdef AF_INET6
945         case 6:
946             addr->ss_family = AF_INET6;
947 # ifdef HAVE_SA_LEN
948             addr->ss_len =
949 # endif
950            *addrlen = sizeof (struct sockaddr_in6);
951
952 #if defined(WIN32) || defined(UNDER_CE)
953             if( WSAStringToAddressA(host, AF_INET6, NULL,
954                     (LPSOCKADDR)addr, addrlen) )
955                 return -1;
956 #else
957             if (inet_pton (AF_INET6, host,
958                            &((struct sockaddr_in6 *)addr)->sin6_addr) <= 0)
959                 return -1;
960 #endif
961             *number = (res >= 3) ? n1 : 1;
962             break;
963 #endif
964
965         case 4:
966             addr->ss_family = AF_INET;
967 # ifdef HAVE_SA_LEN
968             addr->ss_len =
969 # endif
970            *addrlen = sizeof (struct sockaddr_in);
971
972 #if defined(WIN32) || defined(UNDER_CE)
973             if( WSAStringToAddressA(host, AF_INET, NULL,
974                     (LPSOCKADDR)addr, addrlen) )
975                 return -1;
976 #else
977             if (inet_pton (AF_INET, host,
978                            &((struct sockaddr_in *)addr)->sin_addr) <= 0)
979                 return -1;
980 #endif
981
982             *number = (res >= 4) ? n2 : 1;
983             break;
984
985         default:
986             return -1;
987     }
988     return 0;
989 }
990
991
992 /***********************************************************************
993  * ParseSDP : SDP parsing
994  * *********************************************************************
995  * Validate SDP and parse all fields
996  ***********************************************************************/
997 static sdp_t *ParseSDP (vlc_object_t *p_obj, const char *psz_sdp)
998 {
999     if( psz_sdp == NULL )
1000         return NULL;
1001
1002     sdp_t *p_sdp = calloc (1, sizeof (*p_sdp));
1003     if (p_sdp == NULL)
1004         return NULL;
1005
1006     char expect = 'V';
1007     struct sockaddr_storage glob_addr;
1008     memset (&glob_addr, 0, sizeof (glob_addr));
1009     socklen_t glob_len = 0;
1010     unsigned glob_count = 1;
1011
1012     /* TODO: use iconv and charset attribute instead of EnsureUTF8 */
1013     while (*psz_sdp)
1014     {
1015         /* Extract one line */
1016         char *eol = strchr (psz_sdp, '\n');
1017         size_t linelen = eol ? (size_t)(eol - psz_sdp) : strlen (psz_sdp);
1018         char line[linelen + 1];
1019         memcpy (line, psz_sdp, linelen);
1020         line[linelen] = '\0';
1021
1022         psz_sdp += linelen + 1;
1023
1024         /* Remove carriage return if present */
1025         eol = strchr (line, '\r');
1026         if (eol != NULL)
1027         {
1028             linelen = eol - line;
1029             line[linelen] = '\0';
1030         }
1031
1032         /* Validate line */
1033         char cat = line[0], *data = line + 2;
1034         if (!cat || (strchr ("vosiuepcbtrzkam", cat) == NULL))
1035         {
1036             /* MUST ignore SDP with unknown line type */
1037             msg_Dbg (p_obj, "unknown SDP line type: 0x%02x", (int)cat);
1038             goto error;
1039         }
1040         if (line[1] != '=')
1041         {
1042             msg_Dbg (p_obj, "invalid SDP line: %s", line);
1043             goto error;
1044         }
1045
1046         assert (linelen >= 2);
1047
1048         /* SDP parsing state machine
1049          * We INTERNALLY use uppercase for session, lowercase for media
1050          */
1051         switch (expect)
1052         {
1053             /* Session description */
1054             case 'V':
1055                 expect = 'O';
1056                 if (cat != 'v')
1057                 {
1058                     msg_Dbg (p_obj, "missing SDP version");
1059                     goto error;
1060                 }
1061                 if (strcmp (data, "0"))
1062                 {
1063                     msg_Dbg (p_obj, "unknown SDP version: %s", data);
1064                     goto error;
1065                 }
1066                 break;
1067
1068             case 'O':
1069             {
1070                 expect = 'S';
1071                 if (cat != 'o')
1072                 {
1073                     msg_Dbg (p_obj, "missing SDP originator");
1074                     goto error;
1075                 }
1076
1077                 if ((sscanf (data, "%63s "I64Fu" "I64Fu" IN IP%u %1023s",
1078                              p_sdp->username, &p_sdp->session_id,
1079                              &p_sdp->session_version, &p_sdp->orig_ip_version,
1080                              p_sdp->orig_host) != 5)
1081                  || ((p_sdp->orig_ip_version != 4)
1082                   && (p_sdp->orig_ip_version != 6)))
1083                 {
1084                     msg_Dbg (p_obj, "SDP origin not supported: %s\n", data);
1085                     /* Or maybe out-of-range, but this looks suspicious */
1086                     return NULL;
1087                 }
1088                 EnsureUTF8 (p_sdp->orig_host);
1089                 break;
1090             }
1091
1092             case 'S':
1093             {
1094                 expect = 'I';
1095                 if ((cat != 's') || !*data)
1096                 {
1097                     /* MUST be present AND non-empty */
1098                     msg_Dbg (p_obj, "missing SDP session name");
1099                     goto error;
1100                 }
1101                 assert (p_sdp->psz_sessionname == NULL); // no memleak here
1102                 p_sdp->psz_sessionname = strdup (data);
1103                 EnsureUTF8 (p_sdp->psz_sessionname);
1104                 if (p_sdp->psz_sessionname == NULL)
1105                     goto error;
1106                 break;
1107             }
1108
1109             case 'I':
1110                 expect = 'U';
1111                 if (cat == 'i')
1112                     break;
1113             case 'U':
1114                 expect = 'E';
1115                 if (cat == 'u')
1116                     break;
1117             case 'E':
1118                 expect = 'E';
1119                 if (cat == 'e')
1120                     break;
1121             case 'P':
1122                 expect = 'P';
1123                 if (cat == 'p')
1124                     break;
1125             case 'C':
1126                 expect = 'B';
1127                 if (cat == 'c')
1128                 {
1129                     if (ParseSDPConnection (data, &glob_addr, &glob_len,
1130                                             &glob_count))
1131                     {
1132                         msg_Dbg (p_obj, "SDP connection infos not supported: "
1133                                  "%s", data);
1134                         goto error;
1135                     }
1136                     break;
1137                 }
1138             case 'B':
1139                 assert (expect == 'B');
1140                 if (cat == 'b')
1141                     break;
1142             case 'T':
1143                 expect = 'R';
1144                 if (cat != 't')
1145                 {
1146                     msg_Dbg (p_obj, "missing SDP time description");
1147                     goto error;
1148                 }
1149                 break;
1150
1151             case 'R':
1152                 if ((cat == 't') || (cat == 'r'))
1153                     break;
1154
1155             case 'Z':
1156                 expect = 'K';
1157                 if (cat == 'z')
1158                     break;
1159             case 'K':
1160                 expect = 'A';
1161                 if (cat == 'k')
1162                     break;
1163             case 'A':
1164                 //expect = 'A';
1165                 if (cat == 'a')
1166                 {
1167                     attribute_t *p_attr = MakeAttribute (data);
1168                     TAB_APPEND( p_sdp->i_attributes, p_sdp->pp_attributes, p_attr );
1169                     break;
1170                 }
1171
1172             /* Media description */
1173             case 'm':
1174             {
1175                 expect = 'i';
1176                 if (cat != 'm')
1177                 {
1178                     msg_Dbg (p_obj, "missing SDP media description");
1179                     goto error;
1180                 }
1181                 struct sdp_media_t *m = p_sdp->mediav + p_sdp->mediac;
1182
1183                 memcpy (&m->addr, &glob_addr, m->addrlen = glob_len);
1184                 m->n_addr = glob_count;
1185
1186                 /* TODO: remember media type (if we need multiple medias) */
1187                 data = strchr (data, ' ');
1188                 if (data == NULL)
1189                 {
1190                     msg_Dbg (p_obj, "missing SDP media port");
1191                     goto error;
1192                 }
1193                 int port = atoi (++data);
1194                 if (port <= 0 || port >= 65536)
1195                 {
1196                     msg_Dbg (p_obj, "invalid transport port %d", port);
1197                     goto error;
1198                 }
1199                 net_SetPort ((struct sockaddr *)&m->addr, htons (port));
1200
1201                 data = strchr (data, ' ');
1202                 if (data == NULL)
1203                 {
1204                     msg_Dbg (p_obj, "missing SDP media format");
1205                     goto error;
1206                 }
1207                 m->fmt = strdup (++data);
1208                 if (m->fmt == NULL)
1209                     goto error;
1210
1211                 p_sdp->mediac++;
1212                 break;
1213             }
1214             case 'i':
1215                 expect = 'c';
1216                 if (cat == 'i')
1217                     break;
1218             case 'c':
1219                 expect = 'b';
1220                 if (cat == 'c')
1221                 {
1222                     struct sdp_media_t *m = p_sdp->mediav + p_sdp->mediac;
1223                     if (ParseSDPConnection (data, &m->addr, &m->addrlen,
1224                                             &m->n_addr))
1225                     {
1226                         msg_Dbg (p_obj, "SDP connection infos not supported: "
1227                                  "%s", data);
1228                         goto error;
1229                     }
1230                     break;
1231                 }
1232             case 'b':
1233                 expect = 'b';
1234                 if (cat == 'b')
1235                     break;
1236             case 'k':
1237                 expect = 'a';
1238                 if (cat == 'k')
1239                     break;
1240             case 'a':
1241                 assert (expect == 'a');
1242                 if (cat == 'a')
1243                 {
1244                     attribute_t *p_attr = MakeAttribute (data);
1245                     if (p_attr == NULL)
1246                         goto error;
1247
1248                     TAB_APPEND (p_sdp->mediav[p_sdp->mediac - 1].i_attributes,
1249                                 p_sdp->mediav[p_sdp->mediac - 1].pp_attributes, p_attr);
1250                     break;
1251                 }
1252
1253                 if (cat == 'm')
1254                 {
1255                     /* TODO */
1256                     msg_Dbg (p_obj, "multi-media SDP not implemented -> live555");
1257                     goto error;
1258                 }
1259
1260                 if (cat != 'm')
1261                 {
1262                     msg_Dbg (p_obj, "unexpected SDP line: 0x%02x", (int)cat);
1263                     goto error;
1264                 }
1265                 break;
1266
1267             default:
1268                 msg_Err (p_obj, "*** BUG in SDP parser! ***");
1269                 goto error;
1270         }
1271     }
1272
1273     return p_sdp;
1274
1275 error:
1276     FreeSDP (p_sdp);
1277     return NULL;
1278 }
1279
1280 static int InitSocket( services_discovery_t *p_sd, const char *psz_address,
1281                        int i_port )
1282 {
1283     int i_fd = net_ListenUDP1 ((vlc_object_t *)p_sd, psz_address, i_port);
1284     if (i_fd == -1)
1285         return VLC_EGENERIC;
1286
1287     net_StopSend( i_fd );
1288     INSERT_ELEM (p_sd->p_sys->pi_fd, p_sd->p_sys->i_fd,
1289                  p_sd->p_sys->i_fd, i_fd);
1290     return VLC_SUCCESS;
1291 }
1292
1293 static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i_len )
1294 {
1295 #ifdef HAVE_ZLIB_H
1296     int i_result, i_dstsize, n = 0;
1297     unsigned char *psz_dst = NULL;
1298     z_stream d_stream;
1299
1300     memset (&d_stream, 0, sizeof (d_stream));
1301
1302     i_result = inflateInit(&d_stream);
1303     if( i_result != Z_OK )
1304         return( -1 );
1305
1306     d_stream.next_in = (Bytef *)psz_src;
1307     d_stream.avail_in = i_len;
1308
1309     do
1310     {
1311         n++;
1312         psz_dst = (unsigned char *)realloc( psz_dst, n * 1000 );
1313         d_stream.next_out = (Bytef *)&psz_dst[(n - 1) * 1000];
1314         d_stream.avail_out = 1000;
1315
1316         i_result = inflate(&d_stream, Z_NO_FLUSH);
1317         if( ( i_result != Z_OK ) && ( i_result != Z_STREAM_END ) )
1318         {
1319             inflateEnd( &d_stream );
1320             return( -1 );
1321         }
1322     }
1323     while( ( d_stream.avail_out == 0 ) && ( d_stream.avail_in != 0 ) &&
1324            ( i_result != Z_STREAM_END ) );
1325
1326     i_dstsize = d_stream.total_out;
1327     inflateEnd( &d_stream );
1328
1329     *_dst = (unsigned char *)realloc( psz_dst, i_dstsize );
1330
1331     return i_dstsize;
1332 #else
1333     (void)psz_src;
1334     (void)_dst;
1335     (void)i_len;
1336     return -1;
1337 #endif
1338 }
1339
1340
1341 static void FreeSDP( sdp_t *p_sdp )
1342 {
1343     free( p_sdp->psz_sessionname );
1344     free( p_sdp->psz_uri );
1345
1346     for (unsigned j = 0; j < p_sdp->mediac; j++)
1347     {
1348         free (p_sdp->mediav[j].fmt);
1349         for (int i = 0; i < p_sdp->mediav[j].i_attributes; i++)
1350             FreeAttribute (p_sdp->mediav[j].pp_attributes[i]);
1351     }
1352
1353     for (int i = 0; i < p_sdp->i_attributes; i++)
1354         FreeAttribute (p_sdp->pp_attributes[i]);
1355
1356     free (p_sdp->pp_attributes);
1357     free (p_sdp);
1358 }
1359
1360 static int RemoveAnnounce( services_discovery_t *p_sd,
1361                            sap_announce_t *p_announce )
1362 {
1363     int i;
1364
1365     if( p_announce->p_sdp )
1366     {
1367         FreeSDP( p_announce->p_sdp );
1368         p_announce->p_sdp = NULL;
1369     }
1370
1371     if( p_announce->i_input_id > -1 )
1372         playlist_DeleteFromInput( pl_Get(p_sd), p_announce->i_input_id, VLC_FALSE );
1373
1374     for( i = 0; i< p_sd->p_sys->i_announces; i++)
1375     {
1376         if( p_sd->p_sys->pp_announces[i] == p_announce )
1377         {
1378             REMOVE_ELEM( p_sd->p_sys->pp_announces, p_sd->p_sys->i_announces,
1379                          i);
1380             break;
1381         }
1382     }
1383
1384     free( p_announce );
1385
1386     return VLC_SUCCESS;
1387 }
1388
1389 static vlc_bool_t IsSameSession( sdp_t *p_sdp1, sdp_t *p_sdp2 )
1390 {
1391     /* A session is identified by
1392      * - username,
1393      * - session_id,
1394      * - network type (which is always IN),
1395      * - address type (currently, this means IP version),
1396      * - and hostname.
1397      */
1398     if (strcmp (p_sdp1->username, p_sdp2->username)
1399      || (p_sdp1->session_id != p_sdp2->session_id)
1400      || (p_sdp1->orig_ip_version != p_sdp2->orig_ip_version)
1401      || strcmp (p_sdp1->orig_host, p_sdp2->orig_host))
1402         return VLC_FALSE;
1403
1404     return VLC_TRUE;
1405 }
1406
1407
1408 static inline attribute_t *MakeAttribute (const char *str)
1409 {
1410     attribute_t *a = malloc (sizeof (*a) + strlen (str) + 1);
1411     if (a == NULL)
1412         return NULL;
1413
1414     strcpy (a->name, str);
1415     EnsureUTF8 (a->name);
1416     char *value = strchr (a->name, ':');
1417     if (value != NULL)
1418     {
1419         *value++ = '\0';
1420         a->value = value;
1421     }
1422     else
1423         a->value = "";
1424     return a;
1425 }
1426
1427
1428 static const char *GetAttribute (attribute_t **tab, unsigned n,
1429                                  const char *name)
1430 {
1431     for (unsigned i = 0; i < n; i++)
1432         if (strcasecmp (tab[i]->name, name) == 0)
1433             return tab[i]->value;
1434     return NULL;
1435 }
1436
1437
1438 static inline void FreeAttribute (attribute_t *a)
1439 {
1440     free (a);
1441 }