]> git.sesse.net Git - vlc/blob - src/stream_output/announce.c
* Stream output now support IPv6 SAP announces
[vlc] / src / stream_output / announce.c
1 /*****************************************************************************
2  * announce.c : Session announcement
3  *****************************************************************************
4  * Copyright (C) 2002 VideoLAN
5  *
6  * Authors: ClĂ©ment Stenac <zorglub@via.ecp.fr>
7  *          Damien Lucas <nitrox@via.ecp.fr>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #include <stdlib.h>                                                /* free() */
28 #include <stdio.h>                                              /* sprintf() */
29 #include <string.h>                                            /* strerror() */
30
31 #include <unistd.h>
32
33 #include <vlc/vlc.h>
34
35 #include <vlc/sout.h>
36 #undef DEBUG_BUFFER
37
38 #include <announce.h>
39 #include <network.h>
40
41 #define SAP_IPV4_ADDR "224.2.127.254" /* Standard port and address for SAP */
42 #define SAP_PORT 9875
43
44 #define SAP_IPV6_ADDR_1 "FF0"
45 #define SAP_IPV6_ADDR_2 "::2:7FFE"
46
47 /*****************************************************************************
48  * sout_SAPNew: Creates a SAP Session
49  *****************************************************************************/
50 sap_session_t * sout_SAPNew ( sout_instance_t *p_sout , 
51                 char * psz_url_arg , char *psz_port_arg , 
52                 char * psz_name_arg, int ip_version, 
53                 char * psz_v6_scope )
54 {
55       
56     sap_session_t       *p_new; /* The SAP structure */
57     module_t            *p_network; /* Network module */
58     network_socket_t    socket_desc; /* Socket descriptor */
59     char                psz_network[6]; /* IPv4 or IPv6 */
60     struct              sockaddr_in addr; /* IPv4 connection structure */
61     struct              sockaddr_in6 addr6;/* IPv6 connection structure */
62     char                *sap_ipv6_addr=NULL; /* IPv6 built address */     
63     void                *net_ipv6_addr=NULL; /* IPv6 address in net-format */
64     int                 i_status=0;  /* Problems on system calls*/
65     
66     /* Allocate the SAP structure */ 
67     p_new = (sap_session_t *)malloc( sizeof ( sap_session_t ) ) ;
68     if ( !p_new )
69     {
70         msg_Err( p_sout, "No memory left" );        
71         return NULL;
72     }
73                 
74     /* Fill the information in the structure */
75     sprintf ( p_new->psz_url , "%s" , psz_url_arg );
76     sprintf ( p_new->psz_name , "%s" , psz_name_arg );
77     /* Port is not implemented in sout */
78     sprintf ( p_new->psz_port, "%s" , psz_port_arg );
79
80     p_new->i_ip_version = ip_version;
81                         
82     /* Only "6" triggers IPv6. IPv4 is default */
83     if(ip_version != 6)
84     {
85         msg_Dbg( p_sout , "Creating IPv4 SAP socket" );
86         
87         /* Fill the socket descriptor */
88         socket_desc.i_type            = NETWORK_UDP;
89         socket_desc.psz_bind_addr     = "";
90         socket_desc.i_bind_port       = 0;
91         socket_desc.psz_server_addr   = SAP_IPV4_ADDR;
92         socket_desc.i_server_port     = SAP_PORT;
93         socket_desc.i_handle          = 0;
94        
95         /* Call the network module */ 
96         sprintf ( psz_network, "ipv4" ); 
97         p_sout->p_private=(void*) &socket_desc;
98         if( !( p_network = module_Need( p_sout, "network", psz_network ) ) )
99         {
100              msg_Warn( p_sout, "failed to open a connection (udp)" );
101         }
102         module_Unneed( p_sout, p_network );
103               
104         p_new->socket   =       socket_desc.i_handle;
105         if(p_new->socket <= 0 )
106         {
107             msg_Warn( p_sout, "Unable to initialize SAP" );
108             return NULL;
109         }
110         
111         /* Fill the sockaddr_in structure */ 
112         memset( &addr , 0 , sizeof(addr) );
113         addr.sin_family      = AF_INET;
114         addr.sin_addr.s_addr = inet_addr(SAP_IPV4_ADDR);
115         addr.sin_port        = htons( SAP_PORT );
116         p_new->addr = addr;
117     }
118     else
119     {
120         msg_Dbg(p_sout , "Creating IPv6 SAP socket with scope %s" 
121                         , psz_v6_scope );
122
123         /* Initialize and build the IPv6 address to broadcast to */
124         sap_ipv6_addr = (char *)malloc(28*sizeof(char));
125         if ( !sap_ipv6_addr )
126         {
127             msg_Err( p_sout, "No memory left" );        
128             return NULL;
129         }
130         sprintf(sap_ipv6_addr,"%s%c%s",
131                          SAP_IPV6_ADDR_1,
132                          psz_v6_scope[0],
133                          SAP_IPV6_ADDR_2); 
134
135         /* Convert it to network format */
136         net_ipv6_addr = (struct in6_addr *)malloc( sizeof(struct in6_addr) );
137         if ( !net_ipv6_addr )
138         {
139             msg_Err( p_sout, "No memory left" );        
140             return NULL;
141         }
142         
143         i_status         = inet_pton(AF_INET6,sap_ipv6_addr,net_ipv6_addr);
144         if(i_status < 0 )
145         {
146            msg_Warn(p_sout,"Unable to convert address to network format");
147            return NULL;
148         }
149         else if(i_status == 0)
150         {
151             msg_Warn(p_sout,"Adresse de diffusion SAP invalide"); 
152             return NULL;
153         }
154         
155         /* Fill the socket descriptor */
156         socket_desc.i_type        = NETWORK_UDP;
157         socket_desc.psz_bind_addr = ""; 
158         socket_desc.i_bind_port   = 0;
159         socket_desc.psz_server_addr = sap_ipv6_addr;
160         socket_desc.i_server_port     = SAP_PORT;
161         socket_desc.i_handle          = 0;
162             
163         sprintf ( psz_network, "ipv6" ); 
164         
165         /* Call the network module */
166         p_sout->p_private=(void*) &socket_desc;
167         if( !( p_network = module_Need( p_sout, "network", psz_network ) ) )
168         {
169             msg_Warn( p_sout, "failed to open a connection (udp)" );
170         }
171         module_Unneed( p_sout, p_network );
172                
173         p_new->socket   =       socket_desc.i_handle;
174
175         if(p_new->socket <= 0 )
176         {
177              msg_Warn( p_sout, "Unable to initialize SAP" );
178              return NULL;
179          }
180                 
181         /* Fill the sockaddr_in structure */
182         memset( &addr6 , 0 , sizeof(addr6) );
183         addr6.sin6_family      = AF_INET6;
184         memcpy( &addr6.sin6_addr.s6_addr , net_ipv6_addr ,
185                    sizeof(addr6.sin6_addr.s6_addr) );
186             
187         addr6.sin6_port        = htons( SAP_PORT );
188         p_new->addr6 = addr6;
189
190         /* Free what we allocated */
191         if( sap_ipv6_addr ) free(sap_ipv6_addr);
192         if( net_ipv6_addr ) free(net_ipv6_addr);
193     }
194             
195     msg_Dbg (p_sout,"SAP initialization complete");
196   
197     return(p_new);
198 }
199
200 /*****************************************************************************
201  * sout_SAPDelete: Deletes a SAP Session 
202  *****************************************************************************/
203 void sout_SAPDelete( sout_instance_t *p_sout , sap_session_t * p_this )
204 {
205         if( close(p_this->socket) )
206                 msg_Err ( p_sout, "Unable to close SAP socket");
207         
208         if( p_this ) free( p_this );        
209 }       
210
211 /*****************************************************************************
212  * sout_SAPSend: Sends a SAP packet 
213  *****************************************************************************/
214 void sout_SAPSend( sout_instance_t *p_sout, sap_session_t * p_this)
215 {
216         
217     char *sap_head;                         /* SAP header */
218     char sap_msg[1000];                     /* SDP content */
219     char *sap_send;                         /* What we send */
220     char *payload_type="application/sdp";   
221     int i_send_result=0;                    /* Result of send */
222     int i;
223     int i_header_size;                      /* SAP header size */
224     int i_msg_size;                         /* SDP content size */
225     int i_size;                             /* Total size */
226     
227     /* We send a packet every 24 calls to the function */
228     if( p_this->sendnow == 24 )
229     {
230         i_header_size = 9 + strlen( payload_type );
231         sap_head = ( char * )malloc( i_header_size * sizeof( char ) );
232         
233         if( ! sap_head )         
234         {
235             msg_Warn( p_sout , "No memory left");
236             return;
237         }       
238          
239         /* Create the SAP headers */
240         sap_head[0]=0x20; /* Means IPv4, not encrypted, not compressed */
241         sap_head[1]=0x00; /* No authentification */
242         sap_head[2]=0x42; /* Version */
243         sap_head[3]=0x12; /* Version */
244           
245         sap_head[4]=0x01; /* Source IP  FIXME: we should get the real address */
246         sap_head[5]=0x02; /* idem */
247         sap_head[6]=0x03; /* idem */
248         sap_head[7]=0x04; /* idem */
249           
250         strncpy( sap_head+8 , payload_type , 15 );
251         sap_head[ i_header_size-1 ] = '\0'; 
252        
253         /* Create the SDP content */
254         /* Do not add spaces at beginning of the lines ! */  
255           sprintf(sap_msg,"v=0\n\
256 o=VideoLAN 3247692199 3247895918 IN IP4 VideoLAN\n\
257 s=%s\n\
258 u=VideoLAN\n\
259 t=0 0\n\
260 m=audio %s udp 14\n\
261 c=IN IP4 %s/15\n\
262 a=type:test\n", p_this->psz_name , p_this->psz_port , p_this->psz_url );
263      
264        i_msg_size = strlen( sap_msg );     
265        i_size = i_msg_size + i_header_size;
266      
267        /* Create the message */
268        sap_send = ( char* )malloc( i_size*sizeof(char) ); 
269        if(! sap_send)
270        {
271            msg_Err( p_sout ,  "No memory left") ;
272           return;
273        } 
274       
275        for(i=0 ; i<i_header_size ; i++)
276        {
277            sap_send[i] = sap_head[i];
278        }
279        
280        for( ;  i<i_size; i++)
281        {
282            sap_send[i] = sap_msg[i-i_header_size];
283        }
284     
285       if(i_size<1024) /* We mustn't send packets larger than 1024B */     
286       {
287           if( p_this->i_ip_version == 6)
288           {
289                i_send_result =  sendto( p_this->socket , sap_send ,
290                            i_size , 0 , 
291                            (struct sockaddr *)&p_this->addr6, 
292                            sizeof(p_this->addr6) );
293           } 
294           else
295           {
296                 i_send_result =  sendto( p_this->socket , sap_send ,
297                              i_size , 0 , 
298                              (struct sockaddr *)&p_this->addr , 
299                              sizeof(p_this->addr) );
300            }
301       }
302      
303       if(i_send_result == -1)
304       {
305           msg_Warn(p_sout , "SAP Send failed on socket %i. " ,
306                           p_this->socket );
307           perror("sendto"); 
308        }
309       
310        p_this->sendnow = 0;
311        
312        /* Free what we allocated */
313        if(sap_send) free(sap_send);
314        if(sap_head) free(sap_head);
315    }
316     
317    p_this->sendnow++;  
318 }