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
24 #include "os_support.h"
27 typedef struct TCPContext
{
31 /* return non zero if error */
32 static int tcp_open(URLContext
*h
, const char *uri
, int flags
)
34 struct sockaddr_in dest_addr
;
35 char hostname
[1024], *q
;
42 char proto
[1024],path
[1024],tmp
[1024]; // PETR: protocol and path strings
44 url_split(proto
, sizeof(proto
), NULL
, 0, hostname
, sizeof(hostname
),
45 &port
, path
, sizeof(path
), uri
); // PETR: use url_split
46 if (strcmp(proto
,"tcp")) goto fail
; // PETR: check protocol
47 if ((q
= strchr(hostname
,'@'))) { strcpy(tmp
,q
+1); strcpy(hostname
,tmp
); } // PETR: take only the part after '@' for tcp protocol
49 s
= av_malloc(sizeof(TCPContext
));
51 return AVERROR(ENOMEM
);
55 if (port
<= 0 || port
>= 65536)
58 if(!ff_network_init())
61 dest_addr
.sin_family
= AF_INET
;
62 dest_addr
.sin_port
= htons(port
);
63 if (resolve_host(&dest_addr
.sin_addr
, hostname
) < 0)
66 fd
= socket(AF_INET
, SOCK_STREAM
, 0);
69 ff_socket_nonblock(fd
, 1);
72 ret
= connect(fd
, (struct sockaddr
*)&dest_addr
,
75 if (ff_neterrno() == FF_NETERROR(EINTR
))
77 if (ff_neterrno() != FF_NETERROR(EINPROGRESS
) &&
78 ff_neterrno() != FF_NETERROR(EAGAIN
))
81 /* wait until we are connected or until abort */
83 if (url_interrupt_cb()) {
91 tv
.tv_usec
= 100 * 1000;
92 ret
= select(fd_max
+ 1, NULL
, &wfds
, NULL
, &tv
);
93 if (ret
> 0 && FD_ISSET(fd
, &wfds
))
99 getsockopt (fd
, SOL_SOCKET
, SO_ERROR
, &ret
, &optlen
);
115 static int tcp_read(URLContext
*h
, uint8_t *buf
, int size
)
117 TCPContext
*s
= h
->priv_data
;
118 int len
, fd_max
, ret
;
123 if (url_interrupt_cb())
124 return AVERROR(EINTR
);
127 FD_SET(s
->fd
, &rfds
);
129 tv
.tv_usec
= 100 * 1000;
130 ret
= select(fd_max
+ 1, &rfds
, NULL
, NULL
, &tv
);
131 if (ret
> 0 && FD_ISSET(s
->fd
, &rfds
)) {
132 len
= recv(s
->fd
, buf
, size
, 0);
134 if (ff_neterrno() != FF_NETERROR(EINTR
) &&
135 ff_neterrno() != FF_NETERROR(EAGAIN
))
136 return AVERROR(errno
);
138 } else if (ret
< 0) {
144 static int tcp_write(URLContext
*h
, uint8_t *buf
, int size
)
146 TCPContext
*s
= h
->priv_data
;
147 int ret
, size1
, fd_max
, len
;
153 if (url_interrupt_cb())
154 return AVERROR(EINTR
);
157 FD_SET(s
->fd
, &wfds
);
159 tv
.tv_usec
= 100 * 1000;
160 ret
= select(fd_max
+ 1, NULL
, &wfds
, NULL
, &tv
);
161 if (ret
> 0 && FD_ISSET(s
->fd
, &wfds
)) {
162 len
= send(s
->fd
, buf
, size
, 0);
164 if (ff_neterrno() != FF_NETERROR(EINTR
) &&
165 ff_neterrno() != FF_NETERROR(EAGAIN
))
166 return AVERROR(errno
);
171 } else if (ret
< 0) {
178 static int tcp_close(URLContext
*h
)
180 TCPContext
*s
= h
->priv_data
;
187 URLProtocol tcp_protocol
= {