3 * Copyright (c) 2002 Fabrice Bellard.
5 * This file is part of FFmpeg.
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * FFmpeg is distributed in the hope that it will be useful,
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.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
23 #include <sys/types.h>
24 #include <sys/socket.h>
25 #include <netinet/in.h>
26 #include <arpa/inet.h>
31 typedef struct TCPContext
{
35 /* resolve host with also IP address parsing */
36 int resolve_host(struct in_addr
*sin_addr
, const char *hostname
)
40 if ((inet_aton(hostname
, sin_addr
)) == 0) {
41 hp
= gethostbyname(hostname
);
44 memcpy (sin_addr
, hp
->h_addr
, sizeof(struct in_addr
));
49 /* return non zero if error */
50 static int tcp_open(URLContext
*h
, const char *uri
, int flags
)
52 struct sockaddr_in dest_addr
;
53 char hostname
[1024], *q
;
60 char proto
[1024],path
[1024],tmp
[1024]; // PETR: protocol and path strings
62 url_split(proto
, sizeof(proto
), NULL
, 0, hostname
, sizeof(hostname
),
63 &port
, path
, sizeof(path
), uri
); // PETR: use url_split
64 if (strcmp(proto
,"tcp")) goto fail
; // PETR: check protocol
65 if ((q
= strchr(hostname
,'@'))) { strcpy(tmp
,q
+1); strcpy(hostname
,tmp
); } // PETR: take only the part after '@' for tcp protocol
67 s
= av_malloc(sizeof(TCPContext
));
72 if (port
<= 0 || port
>= 65536)
75 dest_addr
.sin_family
= AF_INET
;
76 dest_addr
.sin_port
= htons(port
);
77 if (resolve_host(&dest_addr
.sin_addr
, hostname
) < 0)
80 fd
= socket(PF_INET
, SOCK_STREAM
, 0);
83 fcntl(fd
, F_SETFL
, O_NONBLOCK
);
86 ret
= connect(fd
, (struct sockaddr
*)&dest_addr
,
91 if (errno
!= EINPROGRESS
)
94 /* wait until we are connected or until abort */
96 if (url_interrupt_cb()) {
104 tv
.tv_usec
= 100 * 1000;
105 ret
= select(fd_max
+ 1, NULL
, &wfds
, NULL
, &tv
);
106 if (ret
> 0 && FD_ISSET(fd
, &wfds
))
111 optlen
= sizeof(ret
);
112 getsockopt (fd
, SOL_SOCKET
, SO_ERROR
, &ret
, &optlen
);
128 static int tcp_read(URLContext
*h
, uint8_t *buf
, int size
)
130 TCPContext
*s
= h
->priv_data
;
131 int len
, fd_max
, ret
;
136 if (url_interrupt_cb())
140 FD_SET(s
->fd
, &rfds
);
142 tv
.tv_usec
= 100 * 1000;
143 ret
= select(fd_max
+ 1, &rfds
, NULL
, NULL
, &tv
);
144 if (ret
> 0 && FD_ISSET(s
->fd
, &rfds
)) {
145 len
= recv(s
->fd
, buf
, size
, 0);
147 if (errno
!= EINTR
&& errno
!= EAGAIN
)
154 } else if (ret
< 0) {
160 static int tcp_write(URLContext
*h
, uint8_t *buf
, int size
)
162 TCPContext
*s
= h
->priv_data
;
163 int ret
, size1
, fd_max
, len
;
169 if (url_interrupt_cb())
173 FD_SET(s
->fd
, &wfds
);
175 tv
.tv_usec
= 100 * 1000;
176 ret
= select(fd_max
+ 1, NULL
, &wfds
, NULL
, &tv
);
177 if (ret
> 0 && FD_ISSET(s
->fd
, &wfds
)) {
178 len
= send(s
->fd
, buf
, size
, 0);
180 if (errno
!= EINTR
&& errno
!= EAGAIN
) {
191 } else if (ret
< 0) {
198 static int tcp_close(URLContext
*h
)
200 TCPContext
*s
= h
->priv_data
;
206 URLProtocol tcp_protocol
= {