Mercurial > libavformat.hg
annotate tcp.c @ 6378:a5b3f5d68db6 libavformat
Print error messages in case of connection failure or name resolution failure
in tcp.c.
| author | rbultje |
|---|---|
| date | Sat, 14 Aug 2010 20:34:51 +0000 |
| parents | 4fc5e0e4e1cd |
| children | 3e7384f85f1d |
| rev | line source |
|---|---|
| 0 | 1 /* |
| 2 * TCP protocol | |
|
4251
77e0c7511d41
cosmetics: Remove pointless period after copyright statement non-sentences.
diego
parents:
4206
diff
changeset
|
3 * Copyright (c) 2002 Fabrice Bellard |
| 0 | 4 * |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
5 * This file is part of FFmpeg. |
|
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
6 * |
|
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
7 * FFmpeg is free software; you can redistribute it and/or |
| 0 | 8 * modify it under the terms of the GNU Lesser General Public |
| 9 * License as published by the Free Software Foundation; either | |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
10 * version 2.1 of the License, or (at your option) any later version. |
| 0 | 11 * |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
12 * FFmpeg is distributed in the hope that it will be useful, |
| 0 | 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
| 15 * Lesser General Public License for more details. | |
| 16 * | |
| 17 * You should have received a copy of the GNU Lesser General Public | |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
923
diff
changeset
|
18 * License along with FFmpeg; if not, write to the Free Software |
|
896
edbe5c3717f9
Update licensing information: The FSF changed postal address.
diego
parents:
885
diff
changeset
|
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| 0 | 20 */ |
| 21 #include "avformat.h" | |
| 22 #include <unistd.h> | |
|
5837
d605f589f0be
move ff_url_split() and ff_url_join() declarations to internal.h
aurel
parents:
5775
diff
changeset
|
23 #include "internal.h" |
| 1754 | 24 #include "network.h" |
| 2773 | 25 #include "os_support.h" |
| 4206 | 26 #if HAVE_SYS_SELECT_H |
|
3937
c0667b2aa3ab
include sys/select.h to get select, according to posix 2001,
bcoudurier
parents:
3765
diff
changeset
|
27 #include <sys/select.h> |
|
3941
53c5b89b8dff
only include sys/select.h if present, fix mingw compilation
bcoudurier
parents:
3937
diff
changeset
|
28 #endif |
| 180 | 29 #include <sys/time.h> |
| 0 | 30 |
| 31 typedef struct TCPContext { | |
| 32 int fd; | |
| 33 } TCPContext; | |
| 34 | |
| 35 /* return non zero if error */ | |
| 36 static int tcp_open(URLContext *h, const char *uri, int flags) | |
| 37 { | |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
38 struct addrinfo hints, *ai, *cur_ai; |
| 0 | 39 int port, fd = -1; |
|
683
095009fc2f35
kill warnings patch by (M?ns Rullg?rd <mru inprovide com>)
michael
parents:
511
diff
changeset
|
40 TCPContext *s = NULL; |
| 180 | 41 fd_set wfds; |
| 42 int fd_max, ret; | |
| 43 struct timeval tv; | |
| 44 socklen_t optlen; | |
|
3765
d55fb12134d1
Remove check for @ in tcp.c which removes the authorization data from the
rbultje
parents:
3756
diff
changeset
|
45 char hostname[1024],proto[1024],path[1024]; |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
46 char portstr[10]; |
|
511
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
47 |
| 6182 | 48 av_url_split(proto, sizeof(proto), NULL, 0, hostname, sizeof(hostname), |
| 3756 | 49 &port, path, sizeof(path), uri); |
| 50 if (strcmp(proto,"tcp") || port <= 0 || port >= 65536) | |
|
3755
b9c1bcc6c1ca
On failure, return directly because the fail: case does nothing. This also
rbultje
parents:
3754
diff
changeset
|
51 return AVERROR(EINVAL); |
| 885 | 52 |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
53 memset(&hints, 0, sizeof(hints)); |
|
5595
0980c09b05c1
Fix BSD compile (PF_UNSPEC is not a standard define, AF_UNSPEC is).
rbultje
parents:
5518
diff
changeset
|
54 hints.ai_family = AF_UNSPEC; |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
55 hints.ai_socktype = SOCK_STREAM; |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
56 snprintf(portstr, sizeof(portstr), "%d", port); |
|
6378
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
57 ret = getaddrinfo(hostname, portstr, &hints, &ai); |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
58 if (ret) { |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
59 av_log(NULL, AV_LOG_ERROR, |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
60 "Failed to resolve hostname %s: %s\n", |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
61 hostname, gai_strerror(ret)); |
|
3755
b9c1bcc6c1ca
On failure, return directly because the fail: case does nothing. This also
rbultje
parents:
3754
diff
changeset
|
62 return AVERROR(EIO); |
|
6378
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
63 } |
| 0 | 64 |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
65 cur_ai = ai; |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
66 |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
67 restart: |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
68 fd = socket(cur_ai->ai_family, cur_ai->ai_socktype, cur_ai->ai_protocol); |
| 0 | 69 if (fd < 0) |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
70 goto fail; |
|
2057
857fbfeb2fa0
implement ff_socket_nonblock and use it in networking code
alex
parents:
2056
diff
changeset
|
71 ff_socket_nonblock(fd, 1); |
| 885 | 72 |
| 180 | 73 redo: |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
74 ret = connect(fd, cur_ai->ai_addr, cur_ai->ai_addrlen); |
| 180 | 75 if (ret < 0) { |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
76 if (ff_neterrno() == FF_NETERROR(EINTR)) |
| 180 | 77 goto redo; |
| 2321 | 78 if (ff_neterrno() != FF_NETERROR(EINPROGRESS) && |
| 79 ff_neterrno() != FF_NETERROR(EAGAIN)) | |
| 180 | 80 goto fail; |
| 0 | 81 |
| 180 | 82 /* wait until we are connected or until abort */ |
| 83 for(;;) { | |
| 84 if (url_interrupt_cb()) { | |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
85 ret = AVERROR(EINTR); |
| 180 | 86 goto fail1; |
| 87 } | |
| 88 fd_max = fd; | |
| 89 FD_ZERO(&wfds); | |
| 90 FD_SET(fd, &wfds); | |
| 91 tv.tv_sec = 0; | |
| 92 tv.tv_usec = 100 * 1000; | |
| 93 ret = select(fd_max + 1, NULL, &wfds, NULL, &tv); | |
| 94 if (ret > 0 && FD_ISSET(fd, &wfds)) | |
| 95 break; | |
| 96 } | |
| 885 | 97 |
| 180 | 98 /* test error */ |
| 99 optlen = sizeof(ret); | |
| 100 getsockopt (fd, SOL_SOCKET, SO_ERROR, &ret, &optlen); | |
|
6378
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
101 if (ret != 0) { |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
102 av_log(NULL, AV_LOG_ERROR, |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
103 "TCP connection to %s:%d failed: %s\n", |
|
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
104 hostname, port, strerror(ret)); |
| 180 | 105 goto fail; |
|
6378
a5b3f5d68db6
Print error messages in case of connection failure or name resolution failure
rbultje
parents:
6182
diff
changeset
|
106 } |
| 180 | 107 } |
|
3754
8d267b43eaba
Move malloc() down until after all initializations, so that the resource is
rbultje
parents:
3753
diff
changeset
|
108 s = av_malloc(sizeof(TCPContext)); |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
109 if (!s) { |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
110 freeaddrinfo(ai); |
|
3754
8d267b43eaba
Move malloc() down until after all initializations, so that the resource is
rbultje
parents:
3753
diff
changeset
|
111 return AVERROR(ENOMEM); |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
112 } |
|
3754
8d267b43eaba
Move malloc() down until after all initializations, so that the resource is
rbultje
parents:
3753
diff
changeset
|
113 h->priv_data = s; |
|
8d267b43eaba
Move malloc() down until after all initializations, so that the resource is
rbultje
parents:
3753
diff
changeset
|
114 h->is_streamed = 1; |
| 0 | 115 s->fd = fd; |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
116 freeaddrinfo(ai); |
| 0 | 117 return 0; |
| 118 | |
| 119 fail: | |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
120 if (cur_ai->ai_next) { |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
121 /* Retry with the next sockaddr */ |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
122 cur_ai = cur_ai->ai_next; |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
123 if (fd >= 0) |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
124 closesocket(fd); |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
125 goto restart; |
|
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
126 } |
|
2274
b21c2af60bc9
Replace all occurrences of AVERROR_IO with AVERROR(EIO).
takis
parents:
2057
diff
changeset
|
127 ret = AVERROR(EIO); |
| 180 | 128 fail1: |
| 0 | 129 if (fd >= 0) |
| 1670 | 130 closesocket(fd); |
|
5518
0dc71c7f6653
Use getaddrinfo() instead of resolve_host(). Patch by Martin Storsj?
rbultje
parents:
5102
diff
changeset
|
131 freeaddrinfo(ai); |
| 180 | 132 return ret; |
| 0 | 133 } |
| 134 | |
| 65 | 135 static int tcp_read(URLContext *h, uint8_t *buf, int size) |
| 0 | 136 { |
| 137 TCPContext *s = h->priv_data; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
138 int len, fd_max, ret; |
| 180 | 139 fd_set rfds; |
| 140 struct timeval tv; | |
| 0 | 141 |
|
385
2f56d366a787
no read loop tcp/http and http CRLF fix by (Leon van Stuivenberg <l dot vanstuivenberg at chello dot nl>)
michael
parents:
261
diff
changeset
|
142 for (;;) { |
| 180 | 143 if (url_interrupt_cb()) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
144 return AVERROR(EINTR); |
| 180 | 145 fd_max = s->fd; |
| 146 FD_ZERO(&rfds); | |
| 147 FD_SET(s->fd, &rfds); | |
| 148 tv.tv_sec = 0; | |
| 149 tv.tv_usec = 100 * 1000; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
150 ret = select(fd_max + 1, &rfds, NULL, NULL, &tv); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
151 if (ret > 0 && FD_ISSET(s->fd, &rfds)) { |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
152 len = recv(s->fd, buf, size, 0); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
153 if (len < 0) { |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
154 if (ff_neterrno() != FF_NETERROR(EINTR) && |
|
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
155 ff_neterrno() != FF_NETERROR(EAGAIN)) |
| 5102 | 156 return AVERROR(ff_neterrno()); |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
157 } else return len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
158 } else if (ret < 0) { |
|
5896
395592984ef0
Don't report EINTR from select as an error, retry select instead
mstorsjo
parents:
5837
diff
changeset
|
159 if (ff_neterrno() == FF_NETERROR(EINTR)) |
|
395592984ef0
Don't report EINTR from select as an error, retry select instead
mstorsjo
parents:
5837
diff
changeset
|
160 continue; |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
161 return -1; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
162 } |
| 0 | 163 } |
| 164 } | |
| 165 | |
| 6068 | 166 static int tcp_write(URLContext *h, const uint8_t *buf, int size) |
| 0 | 167 { |
| 168 TCPContext *s = h->priv_data; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
169 int ret, size1, fd_max, len; |
| 180 | 170 fd_set wfds; |
| 171 struct timeval tv; | |
| 0 | 172 |
| 173 size1 = size; | |
| 174 while (size > 0) { | |
| 180 | 175 if (url_interrupt_cb()) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
176 return AVERROR(EINTR); |
| 180 | 177 fd_max = s->fd; |
| 178 FD_ZERO(&wfds); | |
| 179 FD_SET(s->fd, &wfds); | |
| 180 tv.tv_sec = 0; | |
| 181 tv.tv_usec = 100 * 1000; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
182 ret = select(fd_max + 1, NULL, &wfds, NULL, &tv); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
183 if (ret > 0 && FD_ISSET(s->fd, &wfds)) { |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
184 len = send(s->fd, buf, size, 0); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
185 if (len < 0) { |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
186 if (ff_neterrno() != FF_NETERROR(EINTR) && |
|
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
187 ff_neterrno() != FF_NETERROR(EAGAIN)) |
| 5102 | 188 return AVERROR(ff_neterrno()); |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
189 continue; |
|
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
190 } |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
191 size -= len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
192 buf += len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
193 } else if (ret < 0) { |
|
5896
395592984ef0
Don't report EINTR from select as an error, retry select instead
mstorsjo
parents:
5837
diff
changeset
|
194 if (ff_neterrno() == FF_NETERROR(EINTR)) |
|
395592984ef0
Don't report EINTR from select as an error, retry select instead
mstorsjo
parents:
5837
diff
changeset
|
195 continue; |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
196 return -1; |
|
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
197 } |
| 0 | 198 } |
| 199 return size1 - size; | |
| 200 } | |
| 201 | |
| 202 static int tcp_close(URLContext *h) | |
| 203 { | |
| 204 TCPContext *s = h->priv_data; | |
| 205 closesocket(s->fd); | |
| 206 av_free(s); | |
| 207 return 0; | |
| 208 } | |
| 209 | |
|
4640
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
210 static int tcp_get_file_handle(URLContext *h) |
|
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
211 { |
|
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
212 TCPContext *s = h->priv_data; |
|
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
213 return s->fd; |
|
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
214 } |
|
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
215 |
| 0 | 216 URLProtocol tcp_protocol = { |
| 217 "tcp", | |
| 218 tcp_open, | |
| 219 tcp_read, | |
| 220 tcp_write, | |
| 221 NULL, /* seek */ | |
| 222 tcp_close, | |
|
4640
b34d9614b887
Add url_get_file_handle(), which is used to get the file descriptor
rbultje
parents:
4251
diff
changeset
|
223 .url_get_file_handle = tcp_get_file_handle, |
| 0 | 224 }; |
