Mercurial > mplayer.hg
annotate stream/tcp.c @ 29084:88512570a249
Add config.h #include, necessary for HAVE_MALLOC_H check.
author | diego |
---|---|
date | Wed, 01 Apr 2009 16:28:25 +0000 |
parents | 613f075b4ef0 |
children | 0f1b5b68af32 |
rev | line source |
---|---|
19542 | 1 /* |
2 * Network layer for MPlayer | |
3 * by Bertrand BAUDET <bertrand_baudet@yahoo.com> | |
4 * (C) 2001, MPlayer team. | |
5 */ | |
6 | |
7 #include <stdlib.h> | |
8 #include <string.h> | |
9 #include <unistd.h> | |
10 | |
11 #include <errno.h> | |
12 #include <ctype.h> | |
13 | |
14 #include <fcntl.h> | |
15 #include <sys/time.h> | |
16 #include <sys/types.h> | |
17 | |
18 #include "config.h" | |
19 | |
20 #include "mp_msg.h" | |
21 #include "help_mp.h" | |
27472
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
22 |
28402 | 23 #if !HAVE_WINSOCK2_H |
27472
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
24 #include <netdb.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
25 #include <netinet/in.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
26 #include <sys/socket.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
27 #include <arpa/inet.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
28 #else |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
29 #include <winsock2.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
30 #include <ws2tcpip.h> |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
31 #endif |
c0b233cd30ca
Revert moving closesocket definition and network headers to network.h.
diego
parents:
27464
diff
changeset
|
32 |
27473
ae5da477539e
Move '#define closesocket close' preprocessor directive to a common place
diego
parents:
27472
diff
changeset
|
33 #include "network.h" |
26751 | 34 #include "stream.h" |
19542 | 35 #include "tcp.h" |
36 | |
37 /* IPv6 options */ | |
38 int network_prefer_ipv4 = 0; | |
39 | |
40 // Converts an address family constant to a string | |
41 | |
24841
44540930bf94
Make functions static if they aren't referenced externally.
zuxy
parents:
24559
diff
changeset
|
42 static const char *af2String(int af) { |
19542 | 43 switch (af) { |
44 case AF_INET: return "AF_INET"; | |
45 | |
46 #ifdef HAVE_AF_INET6 | |
47 case AF_INET6: return "AF_INET6"; | |
48 #endif | |
49 default: return "Unknown address family!"; | |
50 } | |
51 } | |
52 | |
53 | |
54 | |
55 // Connect to a server using a TCP connection, with specified address family | |
56 // return -2 for fatal error, like unable to resolve name, connection timeout... | |
57 // return -1 is unable to connect to a particular port | |
58 | |
24841
44540930bf94
Make functions static if they aren't referenced externally.
zuxy
parents:
24559
diff
changeset
|
59 static int |
19542 | 60 connect2Server_with_af(char *host, int port, int af,int verb) { |
61 int socket_server_fd; | |
62 int err; | |
63 socklen_t err_len; | |
64 int ret,count = 0; | |
65 fd_set set; | |
66 struct timeval tv; | |
67 union { | |
68 struct sockaddr_in four; | |
69 #ifdef HAVE_AF_INET6 | |
70 struct sockaddr_in6 six; | |
71 #endif | |
72 } server_address; | |
73 size_t server_address_size; | |
74 void *our_s_addr; // Pointer to sin_addr or sin6_addr | |
75 struct hostent *hp=NULL; | |
76 char buf[255]; | |
77 | |
28402 | 78 #if HAVE_WINSOCK2_H |
28162 | 79 unsigned long val; |
22379
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
80 int to; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
81 #else |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
82 struct timeval to; |
19542 | 83 #endif |
84 | |
85 socket_server_fd = socket(af, SOCK_STREAM, 0); | |
86 | |
87 | |
88 if( socket_server_fd==-1 ) { | |
89 // mp_msg(MSGT_NETWORK,MSGL_ERR,"Failed to create %s socket:\n", af2String(af)); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
90 return TCP_ERROR_FATAL; |
19542 | 91 } |
92 | |
22114 | 93 #if defined(SO_RCVTIMEO) && defined(SO_SNDTIMEO) |
28402 | 94 #if HAVE_WINSOCK2_H |
22379
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
95 /* timeout in milliseconds */ |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
96 to = 10 * 1000; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
97 #else |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
98 to.tv_sec = 10; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
99 to.tv_usec = 0; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
100 #endif |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
101 setsockopt(socket_server_fd, SOL_SOCKET, SO_RCVTIMEO, &to, sizeof(to)); |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
102 setsockopt(socket_server_fd, SOL_SOCKET, SO_SNDTIMEO, &to, sizeof(to)); |
22113
50c9dc00154d
Add timeout to tcp connections, avoid hanging forever.
rtogni
parents:
19543
diff
changeset
|
103 #endif |
50c9dc00154d
Add timeout to tcp connections, avoid hanging forever.
rtogni
parents:
19543
diff
changeset
|
104 |
19542 | 105 switch (af) { |
106 case AF_INET: our_s_addr = (void *) &server_address.four.sin_addr; break; | |
107 #ifdef HAVE_AF_INET6 | |
108 case AF_INET6: our_s_addr = (void *) &server_address.six.sin6_addr; break; | |
109 #endif | |
110 default: | |
111 mp_msg(MSGT_NETWORK,MSGL_ERR, MSGTR_MPDEMUX_NW_UnknownAF, af); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
112 return TCP_ERROR_FATAL; |
19542 | 113 } |
114 | |
115 | |
116 memset(&server_address, 0, sizeof(server_address)); | |
117 | |
28404
613f075b4ef0
Restructure network tests: Always check for both inet_aton and inet_pton.
diego
parents:
28402
diff
changeset
|
118 #if HAVE_INET_PTON |
613f075b4ef0
Restructure network tests: Always check for both inet_aton and inet_pton.
diego
parents:
28402
diff
changeset
|
119 if (inet_pton(af, host, our_s_addr)!=1) |
613f075b4ef0
Restructure network tests: Always check for both inet_aton and inet_pton.
diego
parents:
28402
diff
changeset
|
120 #elif HAVE_INET_ATON |
19542 | 121 if (inet_aton(host, our_s_addr)!=1) |
28404
613f075b4ef0
Restructure network tests: Always check for both inet_aton and inet_pton.
diego
parents:
28402
diff
changeset
|
122 #elif HAVE_WINSOCK2_H |
19542 | 123 if ( inet_addr(host)==INADDR_NONE ) |
124 #endif | |
125 { | |
126 if(verb) mp_msg(MSGT_NETWORK,MSGL_STATUS,MSGTR_MPDEMUX_NW_ResolvingHostForAF, host, af2String(af)); | |
127 | |
128 #ifdef HAVE_GETHOSTBYNAME2 | |
129 hp=(struct hostent*)gethostbyname2( host, af ); | |
130 #else | |
131 hp=(struct hostent*)gethostbyname( host ); | |
132 #endif | |
133 if( hp==NULL ) { | |
134 if(verb) mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_CantResolv, af2String(af), host); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
135 return TCP_ERROR_FATAL; |
19542 | 136 } |
137 | |
138 memcpy( our_s_addr, (void*)hp->h_addr_list[0], hp->h_length ); | |
139 } | |
28402 | 140 #if HAVE_WINSOCK2_H |
19542 | 141 else { |
142 unsigned long addr = inet_addr(host); | |
143 memcpy( our_s_addr, (void*)&addr, sizeof(addr) ); | |
144 } | |
145 #endif | |
146 | |
147 switch (af) { | |
148 case AF_INET: | |
149 server_address.four.sin_family=af; | |
150 server_address.four.sin_port=htons(port); | |
151 server_address_size = sizeof(server_address.four); | |
152 break; | |
153 #ifdef HAVE_AF_INET6 | |
154 case AF_INET6: | |
155 server_address.six.sin6_family=af; | |
156 server_address.six.sin6_port=htons(port); | |
157 server_address_size = sizeof(server_address.six); | |
158 break; | |
159 #endif | |
160 default: | |
161 mp_msg(MSGT_NETWORK,MSGL_ERR, MSGTR_MPDEMUX_NW_UnknownAF, af); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
162 return TCP_ERROR_FATAL; |
19542 | 163 } |
164 | |
28400
9202b9245819
HAVE_ATON --> HAVE_INET_ATON to match FFmpeg and give it a 0/1 value.
diego
parents:
28162
diff
changeset
|
165 #if HAVE_INET_ATON || defined(HAVE_WINSOCK2_H) |
19542 | 166 strncpy( buf, inet_ntoa( *((struct in_addr*)our_s_addr) ), 255); |
167 #else | |
168 inet_ntop(af, our_s_addr, buf, 255); | |
169 #endif | |
170 if(verb) mp_msg(MSGT_NETWORK,MSGL_STATUS,MSGTR_MPDEMUX_NW_ConnectingToServer, host, buf , port ); | |
171 | |
172 // Turn the socket as non blocking so we can timeout on the connection | |
28402 | 173 #if !HAVE_WINSOCK2_H |
19542 | 174 fcntl( socket_server_fd, F_SETFL, fcntl(socket_server_fd, F_GETFL) | O_NONBLOCK ); |
175 #else | |
176 val = 1; | |
177 ioctlsocket( socket_server_fd, FIONBIO, &val ); | |
178 #endif | |
179 if( connect( socket_server_fd, (struct sockaddr*)&server_address, server_address_size )==-1 ) { | |
28402 | 180 #if !HAVE_WINSOCK2_H |
19542 | 181 if( errno!=EINPROGRESS ) { |
182 #else | |
183 if( (WSAGetLastError() != WSAEINPROGRESS) && (WSAGetLastError() != WSAEWOULDBLOCK) ) { | |
184 #endif | |
185 if(verb) mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_CantConnect2Server, af2String(af)); | |
186 closesocket(socket_server_fd); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
187 return TCP_ERROR_PORT; |
19542 | 188 } |
189 } | |
190 tv.tv_sec = 0; | |
191 tv.tv_usec = 500000; | |
192 FD_ZERO( &set ); | |
193 FD_SET( socket_server_fd, &set ); | |
24558 | 194 // When the connection will be made, we will have a writeable fd |
19542 | 195 while((ret = select(socket_server_fd+1, NULL, &set, NULL, &tv)) == 0) { |
26326
5bfc1d8bece9
Remove the need for code using stream to export an mp_input_check_interrupt()
albeu
parents:
24841
diff
changeset
|
196 if(count > 30 || stream_check_interrupt(500)) { |
19542 | 197 if(count > 30) |
198 mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_ConnTimeout); | |
199 else | |
24558 | 200 mp_msg(MSGT_NETWORK,MSGL_V,"Connection interrupted by user\n"); |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
201 return TCP_ERROR_TIMEOUT; |
19542 | 202 } |
203 count++; | |
204 FD_ZERO( &set ); | |
205 FD_SET( socket_server_fd, &set ); | |
206 tv.tv_sec = 0; | |
207 tv.tv_usec = 500000; | |
208 } | |
24559
7b6a3948f5f2
(Re)move idiotic checks, ret can't be < 0 or > 0 if the loop condition
reimar
parents:
24558
diff
changeset
|
209 if (ret < 0) mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_SelectFailed); |
19542 | 210 |
211 // Turn back the socket as blocking | |
28402 | 212 #if !HAVE_WINSOCK2_H |
19542 | 213 fcntl( socket_server_fd, F_SETFL, fcntl(socket_server_fd, F_GETFL) & ~O_NONBLOCK ); |
214 #else | |
215 val = 0; | |
216 ioctlsocket( socket_server_fd, FIONBIO, &val ); | |
217 #endif | |
24558 | 218 // Check if there were any errors |
19542 | 219 err_len = sizeof(int); |
220 ret = getsockopt(socket_server_fd,SOL_SOCKET,SO_ERROR,&err,&err_len); | |
221 if(ret < 0) { | |
222 mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_GetSockOptFailed,strerror(errno)); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
223 return TCP_ERROR_FATAL; |
19542 | 224 } |
225 if(err > 0) { | |
226 mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_ConnectError,strerror(err)); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
227 return TCP_ERROR_PORT; |
19542 | 228 } |
229 | |
230 return socket_server_fd; | |
231 } | |
232 | |
233 // Connect to a server using a TCP connection | |
234 // return -2 for fatal error, like unable to resolve name, connection timeout... | |
235 // return -1 is unable to connect to a particular port | |
236 | |
237 | |
238 int | |
239 connect2Server(char *host, int port, int verb) { | |
240 #ifdef HAVE_AF_INET6 | |
241 int r; | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
242 int s = TCP_ERROR_FATAL; |
19542 | 243 |
244 r = connect2Server_with_af(host, port, network_prefer_ipv4 ? AF_INET:AF_INET6,verb); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
245 if (r >= 0) return r; |
19542 | 246 |
247 s = connect2Server_with_af(host, port, network_prefer_ipv4 ? AF_INET6:AF_INET,verb); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
248 if (s == TCP_ERROR_FATAL) return r; |
19542 | 249 return s; |
250 #else | |
251 return connect2Server_with_af(host, port, AF_INET,verb); | |
252 #endif | |
253 | |
254 | |
255 } |