Mercurial > libavformat.hg
annotate tcp.c @ 1627:e4f1922ac981 libavformat
removed unused av_frac_set
author | bcoudurier |
---|---|
date | Mon, 08 Jan 2007 14:44:13 +0000 |
parents | 0899bfe4105c |
children | 92afee454599 |
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> | |
23 #include <sys/types.h> | |
24 #include <sys/socket.h> | |
25 #include <netinet/in.h> | |
923
76c5dfe0b805
os2 thread, network and freetype2 support by (Paul Smedley | paulat a t smedleydot d o t info)
michael
parents:
896
diff
changeset
|
26 #if defined(__BEOS__) || defined(__INNOTEK_LIBC__) |
182 | 27 typedef int socklen_t; |
28 #endif | |
0 | 29 #ifndef __BEOS__ |
30 # include <arpa/inet.h> | |
31 #else | |
32 # include "barpainet.h" | |
33 #endif | |
34 #include <netdb.h> | |
180 | 35 #include <sys/time.h> |
36 #include <fcntl.h> | |
0 | 37 |
38 typedef struct TCPContext { | |
39 int fd; | |
40 } TCPContext; | |
41 | |
42 /* resolve host with also IP address parsing */ | |
43 int resolve_host(struct in_addr *sin_addr, const char *hostname) | |
44 { | |
45 struct hostent *hp; | |
46 | |
47 if ((inet_aton(hostname, sin_addr)) == 0) { | |
48 hp = gethostbyname(hostname); | |
49 if (!hp) | |
50 return -1; | |
51 memcpy (sin_addr, hp->h_addr, sizeof(struct in_addr)); | |
52 } | |
53 return 0; | |
54 } | |
55 | |
56 /* return non zero if error */ | |
57 static int tcp_open(URLContext *h, const char *uri, int flags) | |
58 { | |
59 struct sockaddr_in dest_addr; | |
60 char hostname[1024], *q; | |
61 int port, fd = -1; | |
683
095009fc2f35
kill warnings patch by (Mns Rullgrd <mru inprovide com>)
michael
parents:
511
diff
changeset
|
62 TCPContext *s = NULL; |
180 | 63 fd_set wfds; |
64 int fd_max, ret; | |
65 struct timeval tv; | |
66 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
|
67 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
|
68 |
056991ab9f10
HTTP Authentication Patch by (Petr Doubek <doubek at vision dot ee dot ethz dot ch>)
michael
parents:
482
diff
changeset
|
69 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
|
70 &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
|
71 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
|
72 if ((q = strchr(hostname,'@'))) { strcpy(tmp,q+1); strcpy(hostname,tmp); } // PETR: take only the part after '@' for tcp protocol |
885 | 73 |
0 | 74 s = av_malloc(sizeof(TCPContext)); |
75 if (!s) | |
76 return -ENOMEM; | |
77 h->priv_data = s; | |
885 | 78 |
0 | 79 if (port <= 0 || port >= 65536) |
80 goto fail; | |
885 | 81 |
0 | 82 dest_addr.sin_family = AF_INET; |
83 dest_addr.sin_port = htons(port); | |
84 if (resolve_host(&dest_addr.sin_addr, hostname) < 0) | |
85 goto fail; | |
86 | |
87 fd = socket(PF_INET, SOCK_STREAM, 0); | |
88 if (fd < 0) | |
89 goto fail; | |
180 | 90 fcntl(fd, F_SETFL, O_NONBLOCK); |
885 | 91 |
180 | 92 redo: |
885 | 93 ret = connect(fd, (struct sockaddr *)&dest_addr, |
180 | 94 sizeof(dest_addr)); |
95 if (ret < 0) { | |
96 if (errno == EINTR) | |
97 goto redo; | |
98 if (errno != EINPROGRESS) | |
99 goto fail; | |
0 | 100 |
180 | 101 /* wait until we are connected or until abort */ |
102 for(;;) { | |
103 if (url_interrupt_cb()) { | |
104 ret = -EINTR; | |
105 goto fail1; | |
106 } | |
107 fd_max = fd; | |
108 FD_ZERO(&wfds); | |
109 FD_SET(fd, &wfds); | |
110 tv.tv_sec = 0; | |
111 tv.tv_usec = 100 * 1000; | |
112 ret = select(fd_max + 1, NULL, &wfds, NULL, &tv); | |
113 if (ret > 0 && FD_ISSET(fd, &wfds)) | |
114 break; | |
115 } | |
885 | 116 |
180 | 117 /* test error */ |
118 optlen = sizeof(ret); | |
119 getsockopt (fd, SOL_SOCKET, SO_ERROR, &ret, &optlen); | |
120 if (ret != 0) | |
121 goto fail; | |
122 } | |
0 | 123 s->fd = fd; |
124 return 0; | |
125 | |
126 fail: | |
482 | 127 ret = AVERROR_IO; |
180 | 128 fail1: |
0 | 129 if (fd >= 0) |
130 close(fd); | |
131 av_free(s); | |
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()) |
144 return -EINTR; | |
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)) { |
180 | 152 #ifdef __BEOS__ |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
153 len = recv(s->fd, buf, size, 0); |
0 | 154 #else |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
155 len = read(s->fd, buf, size); |
0 | 156 #endif |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
157 if (len < 0) { |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
158 if (errno != EINTR && errno != EAGAIN) |
0 | 159 #ifdef __BEOS__ |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
160 return errno; |
0 | 161 #else |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
162 return -errno; |
0 | 163 #endif |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
164 } else return len; |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
165 } else if (ret < 0) { |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
166 return -1; |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
167 } |
0 | 168 } |
169 } | |
170 | |
65 | 171 static int tcp_write(URLContext *h, uint8_t *buf, int size) |
0 | 172 { |
173 TCPContext *s = h->priv_data; | |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
174 int ret, size1, fd_max, len; |
180 | 175 fd_set wfds; |
176 struct timeval tv; | |
0 | 177 |
178 size1 = size; | |
179 while (size > 0) { | |
180 | 180 if (url_interrupt_cb()) |
181 return -EINTR; | |
182 fd_max = s->fd; | |
183 FD_ZERO(&wfds); | |
184 FD_SET(s->fd, &wfds); | |
185 tv.tv_sec = 0; | |
186 tv.tv_usec = 100 * 1000; | |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
187 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
|
188 if (ret > 0 && FD_ISSET(s->fd, &wfds)) { |
180 | 189 #ifdef __BEOS__ |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
190 len = send(s->fd, buf, size, 0); |
0 | 191 #else |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
192 len = write(s->fd, buf, size); |
0 | 193 #endif |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
194 if (len < 0) { |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
195 if (errno != EINTR && errno != EAGAIN) { |
0 | 196 #ifdef __BEOS__ |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
197 return errno; |
0 | 198 #else |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
199 return -errno; |
0 | 200 #endif |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
201 } |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
202 continue; |
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
203 } |
388
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
204 size -= len; |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
205 buf += len; |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
206 } else if (ret < 0) { |
9af30d452a0a
tcp select() check and enables pressing 'q' when reading/(writing) from
michael
parents:
385
diff
changeset
|
207 return -1; |
261
5f27f90ed496
Fix a very nasty problem with extra bytes appearing in TCP data streams.
philipjsg
parents:
229
diff
changeset
|
208 } |
0 | 209 } |
210 return size1 - size; | |
211 } | |
212 | |
213 static int tcp_close(URLContext *h) | |
214 { | |
215 TCPContext *s = h->priv_data; | |
216 #ifdef CONFIG_BEOS_NETSERVER | |
217 closesocket(s->fd); | |
218 #else | |
219 close(s->fd); | |
220 #endif | |
221 av_free(s); | |
222 return 0; | |
223 } | |
224 | |
225 URLProtocol tcp_protocol = { | |
226 "tcp", | |
227 tcp_open, | |
228 tcp_read, | |
229 tcp_write, | |
230 NULL, /* seek */ | |
231 tcp_close, | |
232 }; |