diff options
author | Thibaut Mattern <tmattern@users.sourceforge.net> | 2003-06-05 19:08:06 +0000 |
---|---|---|
committer | Thibaut Mattern <tmattern@users.sourceforge.net> | 2003-06-05 19:08:06 +0000 |
commit | 807497efc1296d927cb0fc3b9665f03718fea359 (patch) | |
tree | 2582acf6c9dfc5e6125c37fc7335c9b5bd82d176 | |
parent | 78985adfc5ff2f5e99aa8bbc9a1fff089bb9fc57 (diff) | |
download | xine-lib-807497efc1296d927cb0fc3b9665f03718fea359.tar.gz xine-lib-807497efc1296d927cb0fc3b9665f03718fea359.tar.bz2 |
io_helper functions.
see http://article.gmane.org/gmane.comp.video.xine.devel/4178
CVS patchset: 5005
CVS date: 2003/06/05 19:08:06
-rw-r--r-- | src/xine-engine/io_helper.c | 214 | ||||
-rw-r--r-- | src/xine-engine/io_helper.h | 110 |
2 files changed, 324 insertions, 0 deletions
diff --git a/src/xine-engine/io_helper.c b/src/xine-engine/io_helper.c new file mode 100644 index 000000000..a6d1b5fec --- /dev/null +++ b/src/xine-engine/io_helper.c @@ -0,0 +1,214 @@ +/* + * Copyright (C) 2000-2003 the xine project, + * + * This file is part of xine, a free video player. + * + * xine is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * xine is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA + * + * abortable i/o helper functions + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <stdio.h> +#include <unistd.h> +#include <fcntl.h> +#include <sys/time.h> +#include <sys/types.h> +#include <sys/socket.h> +#include <netinet/in.h> +#include <netdb.h> +#include <errno.h> +#include <string.h> +#include <assert.h> +#include "io_helper.h" + +/* private constants */ +#define XIO_FILE_READ 0 +#define XIO_FILE_WRITE 1 +#define XIO_TCP_READ 2 +#define XIO_TCP_WRITE 3 +#define XIO_POLLING_INTERVAL 50000 /* usec */ + + +int xio_tcp_connect(xine_stream_t *stream, const char *host, int port) { + + struct hostent *h; + int i, s; + + h = gethostbyname(host); + if (h == NULL) { + xine_message(stream, XINE_MSG_UNKNOWN_HOST, "unable to resolve", host, NULL); + return -1; + } + + s = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP); + if (s == -1) { + xine_message(stream, XINE_MSG_CONNECTION_REFUSED, "failed to create socket", strerror(errno), NULL); + return -1; + } + + if (fcntl (s, F_SETFL, fcntl (s, F_GETFL) | O_NONBLOCK) == -1) { + xine_message(stream, XINE_MSG_CONNECTION_REFUSED, "can't put socket in non-blocking mode", strerror(errno), NULL); + return -1; + } + + for (i = 0; h->h_addr_list[i]; i++) { + struct in_addr ia; + struct sockaddr_in sin; + + memcpy (&ia, h->h_addr_list[i], 4); + sin.sin_family = AF_INET; + sin.sin_addr = ia; + sin.sin_port = htons(port); + +#ifndef WIN32 + if (connect(s, (struct sockaddr *)&sin, sizeof(sin))==-1 && errno != EINPROGRESS) { +#else + if (connect(s, (struct sockaddr *)&sin, sizeof(sin))==-1 && WSAGetLastError() != WSAEINPROGRESS) { +#endif /* WIN32 */ + xine_message(stream, XINE_MSG_CONNECTION_REFUSED, strerror(errno), NULL); + close(s); + continue; + } + return s; + } + return -1; +} + + +int xio_select (xine_stream_t *stream, int fd, int state, int timeout_msec) { + + fd_set fdset; + fd_set *rset, *wset; + struct timeval select_timeout; + int timeout_usec, total_time_usec; + int ret; + + timeout_usec = 1000 * timeout_msec; + total_time_usec = 0; + + while (total_time_usec < timeout_usec) { + + FD_ZERO (&fdset); + FD_SET (fd, &fdset); + + select_timeout.tv_sec = 0; + select_timeout.tv_usec = XIO_POLLING_INTERVAL; + + rset = (state & XIO_READ_READY) ? &fdset : NULL; + wset = (state & XIO_WRITE_READY) ? &fdset : NULL; + ret = select (fd + 1, rset, wset, NULL, &select_timeout); + if (ret == -1) { + /* select error */ + return XIO_ERROR; + } else if (ret == 1) { + /* fd is ready */ + return XIO_READY; + } + + /* select timeout + * aborts current read if action pending. otherwise xine + * cannot be stopped when no more data is available. + */ + if (stream->demux_action_pending) + return XIO_ABORTED; + + total_time_usec += XIO_POLLING_INTERVAL; + } + return XIO_TIMEOUT; +} + + +off_t xio_rw_abort(xine_stream_t *stream, int fd, int cmd, char *buf, off_t todo) { + + off_t ret = -1; + off_t total = 0; + int sret; + int state = 0; + + if ((cmd == XIO_TCP_READ) || (cmd == XIO_FILE_READ)) { + state = XIO_READ_READY; + } else { + state = XIO_WRITE_READY; + } + + while (total < todo) { + + do { + sret = xio_select(stream, fd, state, 500); /* 500 ms */ + } while (sret == XIO_TIMEOUT); + + if (sret != XIO_READY) + return -1; + + switch (cmd) { + case XIO_FILE_READ: + ret = read(fd, &buf[total], todo - total); + break; + case XIO_FILE_WRITE: + ret = write(fd, &buf[total], todo - total); + break; + case XIO_TCP_READ: + ret = recv(fd, &buf[total], todo - total, 0); + break; + case XIO_TCP_WRITE: + ret = send(fd, &buf[total], todo - total, 0); + break; + default: + assert(1); + } + /* check EOF */ + if (!ret) + break; + + /* check errors */ + if (ret < 0) { + + /* non-blocking mode */ +#ifndef WIN32 + if (errno == EAGAIN) + continue; + perror("io_helper: I/O error"); +#else + if (WSAGetLastError() == WSAEWOULDBLOCK) + continue; + printf("io_helper: WSAGetLastError() = %d\n", WSAGetLastError()); +#endif + + return ret; + } + total += ret; + } + return total; +} + +off_t xio_tcp_read (xine_stream_t *stream, int s, char *buf, off_t todo) { + return xio_rw_abort (stream, s, XIO_TCP_READ, buf, todo); +} + +off_t xio_tcp_write (xine_stream_t *stream, int s, char *buf, off_t todo) { + return xio_rw_abort (stream, s, XIO_TCP_WRITE, buf, todo); +} + +off_t xio_file_read (xine_stream_t *stream, int s, char *buf, off_t todo) { + return xio_rw_abort (stream, s, XIO_FILE_READ, buf, todo); +} + +off_t xio_file_write (xine_stream_t *stream, int s, char *buf, off_t todo) { + return xio_rw_abort (stream, s, XIO_FILE_WRITE, buf, todo); +} diff --git a/src/xine-engine/io_helper.h b/src/xine-engine/io_helper.h new file mode 100644 index 000000000..45624eab2 --- /dev/null +++ b/src/xine-engine/io_helper.h @@ -0,0 +1,110 @@ +/* + * Copyright (C) 2000-2003 the xine project, + * + * This file is part of xine, a free video player. + * + * xine is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * xine is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA + * + * abortable i/o helper functions + */ + +#ifndef IO_HELPER_H +#define IO_HELPER_H + +#include "xine_internal.h" + + +/* select states */ +#define XIO_READ_READY 1 +#define XIO_WRITE_READY 2 + +/* xine select return codes */ +#define XIO_READY 0 +#define XIO_ERROR 1 +#define XIO_ABORTED 2 +#define XIO_TIMEOUT 3 + + +/* + * Waits for a file descriptor/socket to change status. + * + * network input plugins should use this function in order to + * not freeze the engine. + * + * params : + * fd file/socket descriptor + * state XIO_READ_READY, XIO_WRITE_READY + * *abort an other thread can abort this function by setting *abort + * timeout_sec timeout in seconds + * + * return value : + * XIO_READY the file descriptor is ready for cmd + * XIO_ERROR an i/o error occured + * XIO_ABORTED command aborted by an other thread + * XIO_TIMEOUT the file descriptor is not ready after timeout_msec milliseconds + */ +int xio_select (xine_stream_t *stream, int fd, int state, int timeout_msec); + + +/* + * open a tcp connection + * + * returns a socket descriptor or -1 if an error occured + */ +int xio_tcp_connect(xine_stream_t *stream, const char *host, int port); + +/* + * read from tcp socket checking demux_action_pending + * + * network input plugins should use this function in order to + * not freeze the engine. + * + * aborts with zero if no data is available and *abort is set + */ +off_t xio_tcp_read (xine_stream_t *stream, int s, char *buf, off_t todo); + + +/* + * write to a tcp socket checking demux_action_pending + * + * network input plugins should use this function in order to + * not freeze the engine. + * + * aborts with zero if no data is available and *abort is set + */ +off_t xio_tcp_write (xine_stream_t *stream, int s, char *buf, off_t todo); + +/* + * read from a file descriptor checking demux_action_pending + * + * the fifo input plugin should use this function in order to + * not freeze the engine. + * + * aborts with zero if no data is available and *abort is set + */ +off_t xio_file_read (xine_stream_t *stream, int fd, char *buf, off_t todo); + + +/* + * write to a tcp socket checking demux_action_pending + * + * the fifo input plugin should use this function in order to + * not freeze the engine. + * + * aborts with zero if *abort is set + */ +off_t xio_file_write (xine_stream_t *stream, int fd, char *buf, off_t todo); + +#endif |