Mercurial > libavformat.hg
annotate tcp.c @ 2056:eeea52739ff3 libavformat
use ff_neterrno() and FF_NETERROR() for networking error handling
| author | alex |
|---|---|
| date | Fri, 27 Apr 2007 00:35:54 +0000 |
| parents | 612d19cb9550 |
| children | 857fbfeb2fa0 |
| rev | line source |
|---|---|
| 0 | 1 /* |
| 2 * TCP protocol | |
| 3 * Copyright (c) 2002 Fabrice Bellard. | |
| 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> | |
| 1754 | 23 #include "network.h" |
| 180 | 24 #include <sys/time.h> |
| 25 #include <fcntl.h> | |
| 0 | 26 |
| 27 typedef struct TCPContext { | |
| 28 int fd; | |
| 29 } TCPContext; | |
| 30 | |
| 31 /* return non zero if error */ | |
| 32 static int tcp_open(URLContext *h, const char *uri, int flags) | |
| 33 { | |
| 34 struct sockaddr_in dest_addr; | |
| 35 char hostname[1024], *q; | |
| 36 int port, fd = -1; | |
|
683
095009fc2f35
kill warnings patch by (M?ns Rullg?rd <mru inprovide com>)
michael
parents:
511
diff
changeset
|
37 TCPContext *s = NULL; |
| 180 | 38 fd_set wfds; |
| 39 int fd_max, ret; | |
| 40 struct timeval tv; | |
| 41 socklen_t optlen; | |
|
511
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
42 char proto[1024],path[1024],tmp[1024]; // PETR: protocol and path strings |
|
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
43 |
|
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
44 url_split(proto, sizeof(proto), NULL, 0, hostname, sizeof(hostname), |
|
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
45 &port, path, sizeof(path), uri); // PETR: use url_split |
|
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
46 if (strcmp(proto,"tcp")) goto fail; // PETR: check protocol |
|
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
47 if ((q = strchr(hostname,'@'))) { strcpy(tmp,q+1); strcpy(hostname,tmp); } // PETR: take only the part after '@' for tcp protocol |
| 885 | 48 |
| 0 | 49 s = av_malloc(sizeof(TCPContext)); |
| 50 if (!s) | |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
51 return AVERROR(ENOMEM); |
| 0 | 52 h->priv_data = s; |
| 885 | 53 |
| 0 | 54 if (port <= 0 || port >= 65536) |
| 55 goto fail; | |
| 885 | 56 |
| 0 | 57 dest_addr.sin_family = AF_INET; |
| 58 dest_addr.sin_port = htons(port); | |
| 59 if (resolve_host(&dest_addr.sin_addr, hostname) < 0) | |
| 60 goto fail; | |
| 61 | |
|
1810
d85795da84ab
change PF_INET to AF_INET to be consistent in the whole project. PF_INET is deprecated, while AF_INET is referred by the POSIX standards
alex
parents:
1787
diff
changeset
|
62 fd = socket(AF_INET, SOCK_STREAM, 0); |
| 0 | 63 if (fd < 0) |
| 64 goto fail; | |
| 180 | 65 fcntl(fd, F_SETFL, O_NONBLOCK); |
| 885 | 66 |
| 180 | 67 redo: |
| 885 | 68 ret = connect(fd, (struct sockaddr *)&dest_addr, |
| 180 | 69 sizeof(dest_addr)); |
| 70 if (ret < 0) { | |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
71 if (ff_neterrno() == FF_NETERROR(EINTR)) |
| 180 | 72 goto redo; |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
73 if (ff_neterrno() != FF_NETERROR(EINPROGRESS)) |
| 180 | 74 goto fail; |
| 0 | 75 |
| 180 | 76 /* wait until we are connected or until abort */ |
| 77 for(;;) { | |
| 78 if (url_interrupt_cb()) { | |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
79 ret = AVERROR(EINTR); |
| 180 | 80 goto fail1; |
| 81 } | |
| 82 fd_max = fd; | |
| 83 FD_ZERO(&wfds); | |
| 84 FD_SET(fd, &wfds); | |
| 85 tv.tv_sec = 0; | |
| 86 tv.tv_usec = 100 * 1000; | |
| 87 ret = select(fd_max + 1, NULL, &wfds, NULL, &tv); | |
| 88 if (ret > 0 && FD_ISSET(fd, &wfds)) | |
| 89 break; | |
| 90 } | |
| 885 | 91 |
| 180 | 92 /* test error */ |
| 93 optlen = sizeof(ret); | |
| 94 getsockopt (fd, SOL_SOCKET, SO_ERROR, &ret, &optlen); | |
| 95 if (ret != 0) | |
| 96 goto fail; | |
| 97 } | |
| 0 | 98 s->fd = fd; |
| 99 return 0; | |
| 100 | |
| 101 fail: | |
| 482 | 102 ret = AVERROR_IO; |
| 180 | 103 fail1: |
| 0 | 104 if (fd >= 0) |
| 1670 | 105 closesocket(fd); |
| 0 | 106 av_free(s); |
| 180 | 107 return ret; |
| 0 | 108 } |
| 109 | |
| 65 | 110 static int tcp_read(URLContext *h, uint8_t *buf, int size) |
| 0 | 111 { |
| 112 TCPContext *s = h->priv_data; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
113 int len, fd_max, ret; |
| 180 | 114 fd_set rfds; |
| 115 struct timeval tv; | |
| 0 | 116 |
|
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
|
117 for (;;) { |
| 180 | 118 if (url_interrupt_cb()) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
119 return AVERROR(EINTR); |
| 180 | 120 fd_max = s->fd; |
| 121 FD_ZERO(&rfds); | |
| 122 FD_SET(s->fd, &rfds); | |
| 123 tv.tv_sec = 0; | |
| 124 tv.tv_usec = 100 * 1000; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
125 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
|
126 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
|
127 len = recv(s->fd, buf, size, 0); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
128 if (len < 0) { |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
129 if (ff_neterrno() != FF_NETERROR(EINTR) && |
|
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
130 ff_neterrno() != FF_NETERROR(EAGAIN)) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
131 return AVERROR(errno); |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
132 } else return len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
133 } else if (ret < 0) { |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
134 return -1; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
135 } |
| 0 | 136 } |
| 137 } | |
| 138 | |
| 65 | 139 static int tcp_write(URLContext *h, uint8_t *buf, int size) |
| 0 | 140 { |
| 141 TCPContext *s = h->priv_data; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
142 int ret, size1, fd_max, len; |
| 180 | 143 fd_set wfds; |
| 144 struct timeval tv; | |
| 0 | 145 |
| 146 size1 = size; | |
| 147 while (size > 0) { | |
| 180 | 148 if (url_interrupt_cb()) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
149 return AVERROR(EINTR); |
| 180 | 150 fd_max = s->fd; |
| 151 FD_ZERO(&wfds); | |
| 152 FD_SET(s->fd, &wfds); | |
| 153 tv.tv_sec = 0; | |
| 154 tv.tv_usec = 100 * 1000; | |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
155 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
|
156 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
|
157 len = send(s->fd, buf, size, 0); |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
158 if (len < 0) { |
|
2056
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
159 if (ff_neterrno() != FF_NETERROR(EINTR) && |
|
eeea52739ff3
use ff_neterrno() and FF_NETERROR() for networking error handling
alex
parents:
2050
diff
changeset
|
160 ff_neterrno() != FF_NETERROR(EAGAIN)) |
|
1787
eb16c64144ee
This fixes error handling for BeOS, removing the need for some ifdefs.
mmu_man
parents:
1754
diff
changeset
|
161 return AVERROR(errno); |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
162 continue; |
|
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
163 } |
|
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
164 size -= len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
165 buf += len; |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
166 } else if (ret < 0) { |
|
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
167 return -1; |
|
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
168 } |
| 0 | 169 } |
| 170 return size1 - size; | |
| 171 } | |
| 172 | |
| 173 static int tcp_close(URLContext *h) | |
| 174 { | |
| 175 TCPContext *s = h->priv_data; | |
| 176 closesocket(s->fd); | |
| 177 av_free(s); | |
| 178 return 0; | |
| 179 } | |
| 180 | |
| 181 URLProtocol tcp_protocol = { | |
| 182 "tcp", | |
| 183 tcp_open, | |
| 184 tcp_read, | |
| 185 tcp_write, | |
| 186 NULL, /* seek */ | |
| 187 tcp_close, | |
| 188 }; |
