]> git.sesse.net Git - vlc/blob - src/network/tcp.c
Also set SO_REUSEADDR here
[vlc] / src / network / tcp.c
1 /*****************************************************************************
2  * tcp.c:
3  *****************************************************************************
4  * Copyright (C) 2004-2005 the VideoLAN team
5  * Copyright (C) 2005-2006 Rémi Denis-Courmont
6  * $Id$
7  *
8  * Authors: Laurent Aimar <fenrir@videolan.org>
9  *          Rémi Denis-Courmont <rem # videolan.org>
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  * Preamble
28  *****************************************************************************/
29 #include <stdlib.h>
30 #include <vlc/vlc.h>
31
32 #include <errno.h>
33
34 #ifdef HAVE_FCNTL_H
35 #   include <fcntl.h>
36 #endif
37 #ifdef HAVE_SYS_TIME_H
38 #    include <sys/time.h>
39 #endif
40 #ifdef HAVE_UNISTD_H
41 #   include <unistd.h>
42 #endif
43
44 #include "network.h"
45 #if defined (WIN32) || defined (UNDER_CE)
46 #   undef EINPROGRESS
47 #   define EINPROGRESS WSAEWOULDBLOCK
48 #   undef EINTR
49 #   define EINTR WSAEINTR
50 #   undef ETIMEDOUT
51 #   define ETIMEDOUT WSAETIMEDOUT
52 #endif
53
54 static int SocksNegociate( vlc_object_t *, int fd, int i_socks_version,
55                            char *psz_socks_user, char *psz_socks_passwd );
56 static int SocksHandshakeTCP( vlc_object_t *,
57                               int fd, int i_socks_version,
58                               char *psz_socks_user, char *psz_socks_passwd,
59                               const char *psz_host, int i_port );
60 extern int net_Socket( vlc_object_t *p_this, int i_family, int i_socktype,
61                        int i_protocol );
62 extern int rootwrap_bind (int family, int socktype, int protocol,
63                           const struct sockaddr *addr, size_t alen);
64
65 /*****************************************************************************
66  * __net_ConnectTCP:
67  *****************************************************************************
68  * Open a TCP connection and return a handle
69  *****************************************************************************/
70 int __net_ConnectTCP( vlc_object_t *p_this, const char *psz_host, int i_port )
71 {
72     struct addrinfo hints, *res, *ptr;
73     const char      *psz_realhost;
74     char            *psz_socks;
75     int             i_realport, i_val, i_handle = -1, i_saved_errno = 0;
76     unsigned        u_errstep = 0;
77
78     if( i_port == 0 )
79         i_port = 80; /* historical VLC thing */
80
81     memset( &hints, 0, sizeof( hints ) );
82     hints.ai_socktype = SOCK_STREAM;
83
84     psz_socks = var_CreateGetString( p_this, "socks" );
85     if( *psz_socks && *psz_socks != ':' )
86     {
87         char *psz = strchr( psz_socks, ':' );
88
89         if( psz )
90             *psz++ = '\0';
91
92         psz_realhost = psz_socks;
93         i_realport = ( psz != NULL ) ? atoi( psz ) : 1080;
94
95         msg_Dbg( p_this, "net: connecting to %s port %d for %s port %d",
96                  psz_realhost, i_realport, psz_host, i_port );
97     }
98     else
99     {
100         psz_realhost = psz_host;
101         i_realport = i_port;
102
103         msg_Dbg( p_this, "net: connecting to %s port %d", psz_realhost,
104                  i_realport );
105     }
106
107     i_val = vlc_getaddrinfo( p_this, psz_realhost, i_realport, &hints, &res );
108     if( i_val )
109     {
110         msg_Err( p_this, "cannot resolve %s port %d : %s", psz_realhost,
111                  i_realport, vlc_gai_strerror( i_val ) );
112         free( psz_socks );
113         return -1;
114     }
115
116     for( ptr = res; ptr != NULL; ptr = ptr->ai_next )
117     {
118         int fd = net_Socket( p_this, ptr->ai_family, ptr->ai_socktype,
119                              ptr->ai_protocol );
120         if( fd == -1 )
121         {
122             if( u_errstep <= 0 )
123             {
124                 u_errstep = 1;
125                 i_saved_errno = net_errno;
126             }
127             msg_Dbg( p_this, "socket error: %s", strerror( net_errno ) );
128             continue;
129         }
130
131         if( connect( fd, ptr->ai_addr, ptr->ai_addrlen ) )
132         {
133             socklen_t i_val_size = sizeof( i_val );
134             div_t d;
135             struct timeval tv;
136             vlc_value_t timeout;
137
138             if( net_errno != EINPROGRESS )
139             {
140                 if( u_errstep <= 1 )
141                 {
142                     u_errstep = 2;
143                     i_saved_errno = net_errno;
144                 }
145                 msg_Dbg( p_this, "connect error: %s", strerror( net_errno ) );
146                 goto next_ai;
147             }
148
149             var_Create( p_this, "ipv4-timeout",
150                         VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
151             var_Get( p_this, "ipv4-timeout", &timeout );
152             if( timeout.i_int < 0 )
153             {
154                 msg_Err( p_this, "invalid negative value for ipv4-timeout" );
155                 timeout.i_int = 0;
156             }
157             d = div( timeout.i_int, 100 );
158
159             msg_Dbg( p_this, "connection in progress" );
160             for (;;)
161             {
162                 fd_set fds;
163                 int i_ret;
164
165                 if( p_this->b_die )
166                 {
167                     msg_Dbg( p_this, "connection aborted" );
168                     net_Close( fd );
169                     vlc_freeaddrinfo( res );
170                     free( psz_socks );
171                     return -1;
172                 }
173
174                 /* Initialize file descriptor set */
175                 FD_ZERO( &fds );
176                 FD_SET( fd, &fds );
177
178                 /*
179                  * We'll wait 0.1 second if nothing happens
180                  * NOTE:
181                  * time out will be shortened if we catch a signal (EINTR)
182                  */
183                 tv.tv_sec = 0;
184                 tv.tv_usec = (d.quot > 0) ? 100000 : (1000 * d.rem);
185
186                 i_ret = select( fd + 1, NULL, &fds, NULL, &tv );
187                 if( i_ret == 1 )
188                     break;
189
190                 if( ( i_ret == -1 ) && ( net_errno != EINTR ) )
191                 {
192                     msg_Warn( p_this, "select error: %s",
193                               strerror( net_errno ) );
194                     goto next_ai;
195                 }
196
197                 if( d.quot <= 0 )
198                 {
199                     msg_Dbg( p_this, "select timed out" );
200                     if( u_errstep <= 2 )
201                     {
202                         u_errstep = 3;
203                         i_saved_errno = ETIMEDOUT;
204                     }
205                     goto next_ai;
206                 }
207
208                 d.quot--;
209             }
210
211 #if !defined( SYS_BEOS ) && !defined( UNDER_CE )
212             if( getsockopt( fd, SOL_SOCKET, SO_ERROR, (void*)&i_val,
213                             &i_val_size ) == -1 || i_val != 0 )
214             {
215                 u_errstep = 4;
216                 i_saved_errno = i_val;
217                 msg_Dbg( p_this, "connect error (via getsockopt): %s",
218                          net_strerror( i_val ) );
219                 goto next_ai;
220             }
221 #endif
222         }
223
224         i_handle = fd; /* success! */
225         break;
226
227 next_ai: /* failure */
228         net_Close( fd );
229         continue;
230     }
231
232     vlc_freeaddrinfo( res );
233
234     if( i_handle == -1 )
235     {
236         msg_Err( p_this, "Connection to %s port %d failed: %s", psz_host,
237                  i_port, net_strerror( i_saved_errno ) );
238         free( psz_socks );
239         return -1;
240     }
241
242     if( *psz_socks && *psz_socks != ':' )
243     {
244         char *psz_user = var_CreateGetString( p_this, "socks-user" );
245         char *psz_pwd  = var_CreateGetString( p_this, "socks-pwd" );
246
247         if( SocksHandshakeTCP( p_this, i_handle, 5, psz_user, psz_pwd,
248                                psz_host, i_port ) )
249         {
250             msg_Err( p_this, "Failed to use the SOCKS server" );
251             net_Close( i_handle );
252             i_handle = -1;
253         }
254
255         free( psz_user );
256         free( psz_pwd );
257     }
258     free( psz_socks );
259
260     return i_handle;
261 }
262
263
264 /*****************************************************************************
265  * __net_ListenTCP:
266  *****************************************************************************
267  * Open TCP passive "listening" socket(s)
268  * This function returns NULL in case of error.
269  *****************************************************************************/
270 int *__net_ListenTCP( vlc_object_t *p_this, const char *psz_host, int i_port )
271 {
272     struct addrinfo hints, *res, *ptr;
273     int             i_val, *pi_handles, i_size;
274
275     memset( &hints, 0, sizeof( hints ) );
276     hints.ai_socktype = SOCK_STREAM;
277     hints.ai_flags = AI_PASSIVE;
278
279     msg_Dbg( p_this, "net: listening to %s port %d", psz_host, i_port );
280
281     i_val = vlc_getaddrinfo( p_this, psz_host, i_port, &hints, &res );
282     if( i_val )
283     {
284         msg_Err( p_this, "Cannot resolve %s port %d : %s", psz_host, i_port,
285                  vlc_gai_strerror( i_val ) );
286         return NULL;
287     }
288
289     pi_handles = NULL;
290     i_size = 1;
291
292     for( ptr = res; ptr != NULL; ptr = ptr->ai_next )
293     {
294         int fd, *newpi;
295
296         fd = net_Socket( p_this, ptr->ai_family, ptr->ai_socktype,
297                          ptr->ai_protocol );
298         if( fd == -1 )
299         {
300             msg_Dbg( p_this, "socket error: %s", net_strerror( net_errno ) );
301             continue;
302         }
303
304         /* Bind the socket */
305         if( bind( fd, ptr->ai_addr, ptr->ai_addrlen ) )
306         {
307             int saved_errno;
308
309             saved_errno = net_errno;
310             net_Close( fd );
311 #if !defined(WIN32) && !defined(UNDER_CE)
312             fd = rootwrap_bind( ptr->ai_family, ptr->ai_socktype,
313                                 ptr->ai_protocol, ptr->ai_addr,
314                                 ptr->ai_addrlen );
315             if( fd != -1 )
316             {
317                 msg_Dbg( p_this, "got socket %d from rootwrap", fd );
318             }
319             else
320 #endif
321             {
322                 msg_Err( p_this, "cannot bind socket (%s)",
323                          net_strerror( saved_errno ) );
324                 continue;
325             }
326         }
327
328         /* Listen */
329         if( listen( fd, 100 ) == -1 )
330         {
331             msg_Err( p_this, "cannot bring the socket in listening mode (%s)",
332                      net_strerror( net_errno ) );
333             net_Close( fd );
334             continue;
335         }
336
337         newpi = (int *)realloc( pi_handles, (++i_size) * sizeof( int ) );
338         if( newpi == NULL )
339         {
340             net_Close( fd );
341             break;
342         }
343         else
344         {
345             newpi[i_size - 2] = fd;
346             pi_handles = newpi;
347         }
348     }
349
350     vlc_freeaddrinfo( res );
351
352     if( pi_handles != NULL )
353         pi_handles[i_size - 1] = -1;
354     return pi_handles;
355 }
356
357 /*****************************************************************************
358  * __net_Accept:
359  *****************************************************************************
360  * Accept a connection on a set of listening sockets and return it
361  *****************************************************************************/
362 int __net_Accept( vlc_object_t *p_this, int *pi_fd, mtime_t i_wait )
363 {
364     vlc_bool_t b_die = p_this->b_die, b_block = (i_wait < 0);
365
366     while( p_this->b_die == b_die )
367     {
368         int i_val = -1, *pi, *pi_end;
369         struct timeval timeout;
370         fd_set fds_r, fds_e;
371
372         pi = pi_fd;
373
374         /* Initialize file descriptor set */
375         FD_ZERO( &fds_r );
376         FD_ZERO( &fds_e );
377
378         for( pi = pi_fd; *pi != -1; pi++ )
379         {
380             int i_fd = *pi;
381
382             if( i_fd > i_val )
383                 i_val = i_fd;
384
385             FD_SET( i_fd, &fds_r );
386             FD_SET( i_fd, &fds_e );
387         }
388         pi_end = pi;
389
390         timeout.tv_sec = 0;
391         timeout.tv_usec = b_block ? 500000 : i_wait;
392
393         i_val = select( i_val + 1, &fds_r, NULL, &fds_e, &timeout );
394         if( ( ( i_val < 0 ) && ( net_errno == EINTR ) ) || i_val == 0 )
395         {
396             if( b_block )
397                 continue;
398             else
399                 return -1;
400         }
401         else if( i_val < 0 )
402         {
403             msg_Err( p_this, "network select error (%s)",
404                      net_strerror( net_errno ) );
405             return -1;
406         }
407
408         for( pi = pi_fd; *pi != -1; pi++ )
409         {
410             int i_fd = *pi;
411
412             if( !FD_ISSET( i_fd, &fds_r ) && !FD_ISSET( i_fd, &fds_e ) )
413                 continue;
414
415             i_val = accept( i_fd, NULL, 0 );
416             if( i_val < 0 )
417                 msg_Err( p_this, "accept failed (%s)",
418                          net_strerror( net_errno ) );
419             else if( i_val >= FD_SETSIZE )
420             {
421                 net_Close( i_val ); /* avoid future overflows in FD_SET */
422                 msg_Err( p_this, "accept failed (too many sockets opened)" );
423             }
424             else
425             {
426                 const int yes = 1;
427                 setsockopt( i_fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof( yes ));
428 #ifdef FD_CLOEXEC
429                 fcntl( i_fd, F_SETFD, FD_CLOEXEC );
430 #endif
431                 /*
432                  * This round-robin trick ensures that the first sockets in
433                  * pi_fd won't prevent the last ones from getting accept'ed.
434                  */
435                 --pi_end;
436                 memmove( pi, pi + 1, pi_end - pi );
437                 *pi_end = i_fd;
438                 return i_val;
439             }
440         }
441     }
442
443     return -1;
444 }
445
446
447 /*****************************************************************************
448  * SocksNegociate:
449  *****************************************************************************
450  * Negociate authentication with a SOCKS server.
451  *****************************************************************************/
452 static int SocksNegociate( vlc_object_t *p_obj,
453                            int fd, int i_socks_version,
454                            char *psz_socks_user,
455                            char *psz_socks_passwd )
456 {
457     uint8_t buffer[128+2*256];
458     int i_len;
459     vlc_bool_t b_auth = VLC_FALSE;
460
461     if( i_socks_version != 5 )
462         return VLC_SUCCESS;
463
464     /* We negociate authentication */
465
466     if( psz_socks_user && psz_socks_passwd &&
467         *psz_socks_user && *psz_socks_passwd )
468         b_auth = VLC_TRUE;
469
470     buffer[0] = i_socks_version;    /* SOCKS version */
471     if( b_auth )
472     {
473         buffer[1] = 2;                  /* Number of methods */
474         buffer[2] = 0x00;               /* - No auth required */
475         buffer[3] = 0x02;               /* - USer/Password */
476         i_len = 4;
477     }
478     else
479     {
480         buffer[1] = 1;                  /* Number of methods */
481         buffer[2] = 0x00;               /* - No auth required */
482         i_len = 3;
483     }
484
485     if( net_Write( p_obj, fd, NULL, buffer, i_len ) != i_len )
486         return VLC_EGENERIC;
487     if( net_Read( p_obj, fd, NULL, buffer, 2, VLC_TRUE ) != 2 )
488         return VLC_EGENERIC;
489
490     msg_Dbg( p_obj, "socks: v=%d method=%x", buffer[0], buffer[1] );
491
492     if( buffer[1] == 0x00 )
493     {
494         msg_Dbg( p_obj, "socks: no authentication required" );
495     }
496     else if( buffer[1] == 0x02 )
497     {
498         int i_len1 = __MIN( strlen(psz_socks_user), 255 );
499         int i_len2 = __MIN( strlen(psz_socks_passwd), 255 );
500         msg_Dbg( p_obj, "socks: username/password authentication" );
501
502         /* XXX: we don't support user/pwd > 255 (truncated)*/
503         buffer[0] = i_socks_version;        /* Version */
504         buffer[1] = i_len1;                 /* User length */
505         memcpy( &buffer[2], psz_socks_user, i_len1 );
506         buffer[2+i_len1] = i_len2;          /* Password length */
507         memcpy( &buffer[2+i_len1+1], psz_socks_passwd, i_len2 );
508
509         i_len = 3 + i_len1 + i_len2;
510
511         if( net_Write( p_obj, fd, NULL, buffer, i_len ) != i_len )
512             return VLC_EGENERIC;
513
514         if( net_Read( p_obj, fd, NULL, buffer, 2, VLC_TRUE ) != 2 )
515             return VLC_EGENERIC;
516
517         msg_Dbg( p_obj, "socks: v=%d status=%x", buffer[0], buffer[1] );
518         if( buffer[1] != 0x00 )
519         {
520             msg_Err( p_obj, "socks: authentication rejected" );
521             return VLC_EGENERIC;
522         }
523     }
524     else
525     {
526         if( b_auth )
527             msg_Err( p_obj, "socks: unsupported authentication method %x",
528                      buffer[0] );
529         else
530             msg_Err( p_obj, "socks: authentification needed" );
531         return VLC_EGENERIC;
532     }
533
534     return VLC_SUCCESS;
535 }
536
537 /*****************************************************************************
538  * SocksHandshakeTCP:
539  *****************************************************************************
540  * Open a TCP connection using a SOCKS server and return a handle (RFC 1928)
541  *****************************************************************************/
542 static int SocksHandshakeTCP( vlc_object_t *p_obj,
543                               int fd,
544                               int i_socks_version,
545                               char *psz_socks_user, char *psz_socks_passwd,
546                               const char *psz_host, int i_port )
547 {
548     uint8_t buffer[128+2*256];
549
550     if( i_socks_version != 4 && i_socks_version != 5 )
551     {
552         msg_Warn( p_obj, "invalid socks protocol version %d", i_socks_version );
553         i_socks_version = 5;
554     }
555
556     if( i_socks_version == 5 && 
557         SocksNegociate( p_obj, fd, i_socks_version,
558                         psz_socks_user, psz_socks_passwd ) )
559         return VLC_EGENERIC;
560
561     if( i_socks_version == 4 )
562     {
563         struct addrinfo hints, *p_res;
564
565         /* v4 only support ipv4 */
566         memset (&hints, 0, sizeof (hints));
567         hints.ai_family = AF_INET;
568         if( vlc_getaddrinfo( p_obj, psz_host, 0, &hints, &p_res ) )
569             return VLC_EGENERIC;
570
571         buffer[0] = i_socks_version;
572         buffer[1] = 0x01;               /* CONNECT */
573         SetWBE( &buffer[2], i_port );   /* Port */
574         memcpy( &buffer[4],             /* Address */
575                 &((struct sockaddr_in *)(p_res->ai_addr))->sin_addr, 4 );
576         vlc_freeaddrinfo( p_res );
577
578         buffer[8] = 0;                  /* Empty user id */
579
580         if( net_Write( p_obj, fd, NULL, buffer, 9 ) != 9 )
581             return VLC_EGENERIC;
582         if( net_Read( p_obj, fd, NULL, buffer, 8, VLC_TRUE ) != 8 )
583             return VLC_EGENERIC;
584
585         msg_Dbg( p_obj, "socks: v=%d cd=%d",
586                  buffer[0], buffer[1] );
587
588         if( buffer[1] != 90 )
589             return VLC_EGENERIC;
590     }
591     else if( i_socks_version == 5 )
592     {
593         int i_hlen = __MIN(strlen( psz_host ), 255);
594         int i_len;
595
596         buffer[0] = i_socks_version;    /* Version */
597         buffer[1] = 0x01;               /* Cmd: connect */
598         buffer[2] = 0x00;               /* Reserved */
599         buffer[3] = 3;                  /* ATYP: for now domainname */
600
601         buffer[4] = i_hlen;
602         memcpy( &buffer[5], psz_host, i_hlen );
603         SetWBE( &buffer[5+i_hlen], i_port );
604
605         i_len = 5 + i_hlen + 2;
606
607
608         if( net_Write( p_obj, fd, NULL, buffer, i_len ) != i_len )
609             return VLC_EGENERIC;
610
611         /* Read the header */
612         if( net_Read( p_obj, fd, NULL, buffer, 5, VLC_TRUE ) != 5 )
613             return VLC_EGENERIC;
614
615         msg_Dbg( p_obj, "socks: v=%d rep=%d atyp=%d",
616                  buffer[0], buffer[1], buffer[3] );
617
618         if( buffer[1] != 0x00 )
619         {
620             msg_Err( p_obj, "socks: CONNECT request failed\n" );
621             return VLC_EGENERIC;
622         }
623
624         /* Read the remaining bytes */
625         if( buffer[3] == 0x01 )
626             i_len = 4-1 + 2;
627         else if( buffer[3] == 0x03 )
628             i_len = buffer[4] + 2;
629         else if( buffer[3] == 0x04 )
630             i_len = 16-1+2;
631         else 
632             return VLC_EGENERIC;
633
634         if( net_Read( p_obj, fd, NULL, buffer, i_len, VLC_TRUE ) != i_len )
635             return VLC_EGENERIC;
636     }
637
638     return VLC_SUCCESS;
639 }
640
641 void net_ListenClose( int *pi_fd )
642 {
643     if( pi_fd != NULL )
644     {
645         int *pi;
646
647         for( pi = pi_fd; *pi != -1; pi++ )
648             net_Close( *pi );
649         free( pi_fd );
650     }
651 }