Mercurial > mplayer.hg
annotate stream/tcp.c @ 26480:b9ced5a4c5ff
consistently print fps with three digits of precision
author | corey |
---|---|
date | Wed, 23 Apr 2008 20:20:25 +0000 |
parents | 5bfc1d8bece9 |
children | ba4a01eb4475 |
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" | |
22 | |
23 #ifndef HAVE_WINSOCK2 | |
24 #include <netdb.h> | |
25 #include <netinet/in.h> | |
26 #include <sys/socket.h> | |
27 #include <arpa/inet.h> | |
28 #define closesocket close | |
29 #else | |
30 #include <winsock2.h> | |
31 #include <ws2tcpip.h> | |
32 #endif | |
33 | |
34 #include "tcp.h" | |
35 | |
36 /* IPv6 options */ | |
37 int network_prefer_ipv4 = 0; | |
38 | |
39 // Converts an address family constant to a string | |
40 | |
24841
44540930bf94
Make functions static if they aren't referenced externally.
zuxy
parents:
24559
diff
changeset
|
41 static const char *af2String(int af) { |
19542 | 42 switch (af) { |
43 case AF_INET: return "AF_INET"; | |
44 | |
45 #ifdef HAVE_AF_INET6 | |
46 case AF_INET6: return "AF_INET6"; | |
47 #endif | |
48 default: return "Unknown address family!"; | |
49 } | |
50 } | |
51 | |
52 | |
53 | |
54 // Connect to a server using a TCP connection, with specified address family | |
55 // return -2 for fatal error, like unable to resolve name, connection timeout... | |
56 // return -1 is unable to connect to a particular port | |
57 | |
24841
44540930bf94
Make functions static if they aren't referenced externally.
zuxy
parents:
24559
diff
changeset
|
58 static int |
19542 | 59 connect2Server_with_af(char *host, int port, int af,int verb) { |
60 int socket_server_fd; | |
61 int err; | |
62 socklen_t err_len; | |
63 int ret,count = 0; | |
64 fd_set set; | |
65 struct timeval tv; | |
66 union { | |
67 struct sockaddr_in four; | |
68 #ifdef HAVE_AF_INET6 | |
69 struct sockaddr_in6 six; | |
70 #endif | |
71 } server_address; | |
72 size_t server_address_size; | |
73 void *our_s_addr; // Pointer to sin_addr or sin6_addr | |
74 struct hostent *hp=NULL; | |
75 char buf[255]; | |
76 | |
77 #ifdef HAVE_WINSOCK2 | |
78 u_long val; | |
22379
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
79 int to; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
80 #else |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
81 struct timeval to; |
19542 | 82 #endif |
83 | |
84 socket_server_fd = socket(af, SOCK_STREAM, 0); | |
85 | |
86 | |
87 if( socket_server_fd==-1 ) { | |
88 // 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
|
89 return TCP_ERROR_FATAL; |
19542 | 90 } |
91 | |
22114 | 92 #if defined(SO_RCVTIMEO) && defined(SO_SNDTIMEO) |
22379
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
93 #ifdef HAVE_WINSOCK2 |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
94 /* timeout in milliseconds */ |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
95 to = 10 * 1000; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
96 #else |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
97 to.tv_sec = 10; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
98 to.tv_usec = 0; |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
99 #endif |
6b245d6e56a9
winsocks expects an int in milliseconds instead of struct timeval to set
ivo
parents:
22142
diff
changeset
|
100 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
|
101 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
|
102 #endif |
50c9dc00154d
Add timeout to tcp connections, avoid hanging forever.
rtogni
parents:
19543
diff
changeset
|
103 |
19542 | 104 switch (af) { |
105 case AF_INET: our_s_addr = (void *) &server_address.four.sin_addr; break; | |
106 #ifdef HAVE_AF_INET6 | |
107 case AF_INET6: our_s_addr = (void *) &server_address.six.sin6_addr; break; | |
108 #endif | |
109 default: | |
110 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
|
111 return TCP_ERROR_FATAL; |
19542 | 112 } |
113 | |
114 | |
115 memset(&server_address, 0, sizeof(server_address)); | |
116 | |
117 #ifndef HAVE_WINSOCK2 | |
118 #ifdef USE_ATON | |
119 if (inet_aton(host, our_s_addr)!=1) | |
120 #else | |
121 if (inet_pton(af, host, our_s_addr)!=1) | |
122 #endif | |
123 #else | |
124 if ( inet_addr(host)==INADDR_NONE ) | |
125 #endif | |
126 { | |
127 if(verb) mp_msg(MSGT_NETWORK,MSGL_STATUS,MSGTR_MPDEMUX_NW_ResolvingHostForAF, host, af2String(af)); | |
128 | |
129 #ifdef HAVE_GETHOSTBYNAME2 | |
130 hp=(struct hostent*)gethostbyname2( host, af ); | |
131 #else | |
132 hp=(struct hostent*)gethostbyname( host ); | |
133 #endif | |
134 if( hp==NULL ) { | |
135 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
|
136 return TCP_ERROR_FATAL; |
19542 | 137 } |
138 | |
139 memcpy( our_s_addr, (void*)hp->h_addr_list[0], hp->h_length ); | |
140 } | |
141 #ifdef HAVE_WINSOCK2 | |
142 else { | |
143 unsigned long addr = inet_addr(host); | |
144 memcpy( our_s_addr, (void*)&addr, sizeof(addr) ); | |
145 } | |
146 #endif | |
147 | |
148 switch (af) { | |
149 case AF_INET: | |
150 server_address.four.sin_family=af; | |
151 server_address.four.sin_port=htons(port); | |
152 server_address_size = sizeof(server_address.four); | |
153 break; | |
154 #ifdef HAVE_AF_INET6 | |
155 case AF_INET6: | |
156 server_address.six.sin6_family=af; | |
157 server_address.six.sin6_port=htons(port); | |
158 server_address_size = sizeof(server_address.six); | |
159 break; | |
160 #endif | |
161 default: | |
162 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
|
163 return TCP_ERROR_FATAL; |
19542 | 164 } |
165 | |
166 #if defined(USE_ATON) || defined(HAVE_WINSOCK2) | |
167 strncpy( buf, inet_ntoa( *((struct in_addr*)our_s_addr) ), 255); | |
168 #else | |
169 inet_ntop(af, our_s_addr, buf, 255); | |
170 #endif | |
171 if(verb) mp_msg(MSGT_NETWORK,MSGL_STATUS,MSGTR_MPDEMUX_NW_ConnectingToServer, host, buf , port ); | |
172 | |
173 // Turn the socket as non blocking so we can timeout on the connection | |
174 #ifndef HAVE_WINSOCK2 | |
175 fcntl( socket_server_fd, F_SETFL, fcntl(socket_server_fd, F_GETFL) | O_NONBLOCK ); | |
176 #else | |
177 val = 1; | |
178 ioctlsocket( socket_server_fd, FIONBIO, &val ); | |
179 #endif | |
180 if( connect( socket_server_fd, (struct sockaddr*)&server_address, server_address_size )==-1 ) { | |
181 #ifndef HAVE_WINSOCK2 | |
182 if( errno!=EINPROGRESS ) { | |
183 #else | |
184 if( (WSAGetLastError() != WSAEINPROGRESS) && (WSAGetLastError() != WSAEWOULDBLOCK) ) { | |
185 #endif | |
186 if(verb) mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_CantConnect2Server, af2String(af)); | |
187 closesocket(socket_server_fd); | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
188 return TCP_ERROR_PORT; |
19542 | 189 } |
190 } | |
191 tv.tv_sec = 0; | |
192 tv.tv_usec = 500000; | |
193 FD_ZERO( &set ); | |
194 FD_SET( socket_server_fd, &set ); | |
24558 | 195 // When the connection will be made, we will have a writeable fd |
19542 | 196 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
|
197 if(count > 30 || stream_check_interrupt(500)) { |
19542 | 198 if(count > 30) |
199 mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_ConnTimeout); | |
200 else | |
24558 | 201 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
|
202 return TCP_ERROR_TIMEOUT; |
19542 | 203 } |
204 count++; | |
205 FD_ZERO( &set ); | |
206 FD_SET( socket_server_fd, &set ); | |
207 tv.tv_sec = 0; | |
208 tv.tv_usec = 500000; | |
209 } | |
24559
7b6a3948f5f2
(Re)move idiotic checks, ret can't be < 0 or > 0 if the loop condition
reimar
parents:
24558
diff
changeset
|
210 if (ret < 0) mp_msg(MSGT_NETWORK,MSGL_ERR,MSGTR_MPDEMUX_NW_SelectFailed); |
19542 | 211 |
212 // Turn back the socket as blocking | |
213 #ifndef HAVE_WINSOCK2 | |
214 fcntl( socket_server_fd, F_SETFL, fcntl(socket_server_fd, F_GETFL) & ~O_NONBLOCK ); | |
215 #else | |
216 val = 0; | |
217 ioctlsocket( socket_server_fd, FIONBIO, &val ); | |
218 #endif | |
24558 | 219 // Check if there were any errors |
19542 | 220 err_len = sizeof(int); |
221 ret = getsockopt(socket_server_fd,SOL_SOCKET,SO_ERROR,&err,&err_len); | |
222 if(ret < 0) { | |
223 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
|
224 return TCP_ERROR_FATAL; |
19542 | 225 } |
226 if(err > 0) { | |
227 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
|
228 return TCP_ERROR_PORT; |
19542 | 229 } |
230 | |
231 return socket_server_fd; | |
232 } | |
233 | |
234 // Connect to a server using a TCP connection | |
235 // return -2 for fatal error, like unable to resolve name, connection timeout... | |
236 // return -1 is unable to connect to a particular port | |
237 | |
238 | |
239 int | |
240 connect2Server(char *host, int port, int verb) { | |
241 #ifdef HAVE_AF_INET6 | |
242 int r; | |
19543
e053647fbeec
Cosmetics: recommit patch changing return values to defines
reimar
parents:
19542
diff
changeset
|
243 int s = TCP_ERROR_FATAL; |
19542 | 244 |
245 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
|
246 if (r >= 0) return r; |
19542 | 247 |
248 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
|
249 if (s == TCP_ERROR_FATAL) return r; |
19542 | 250 return s; |
251 #else | |
252 return connect2Server_with_af(host, port, AF_INET,verb); | |
253 #endif | |
254 | |
255 | |
256 } |