diff options
author | des <des@FreeBSD.org> | 2000-01-07 12:58:40 +0000 |
---|---|---|
committer | des <des@FreeBSD.org> | 2000-01-07 12:58:40 +0000 |
commit | c701cfdd7c04109f912bfb68fd7a585ecbd44e1d (patch) | |
tree | 818f61f93dc1f5c6c7436a1c8f6e20e3b7c09201 /lib/libfetch/common.c | |
parent | 3acbd8ffda546db2a916c987d7912527644c316f (diff) | |
download | FreeBSD-src-c701cfdd7c04109f912bfb68fd7a585ecbd44e1d.zip FreeBSD-src-c701cfdd7c04109f912bfb68fd7a585ecbd44e1d.tar.gz |
More old uncommitted patches: implement timeouts at the protocol level.
Currently only supported for ftp connections.
Diffstat (limited to 'lib/libfetch/common.c')
-rw-r--r-- | lib/libfetch/common.c | 82 |
1 files changed, 82 insertions, 0 deletions
diff --git a/lib/libfetch/common.c b/lib/libfetch/common.c index 8b3b891..ed5072c 100644 --- a/lib/libfetch/common.c +++ b/lib/libfetch/common.c @@ -30,6 +30,7 @@ #include <sys/param.h> #include <sys/socket.h> +#include <sys/time.h> #include <netinet/in.h> #include <com_err.h> @@ -242,6 +243,87 @@ _fetch_connect(char *host, int port, int verbose) } +/* + * Read a line of text from a socket w/ timeout + */ +#define MIN_BUF_SIZE 1024 + +int +_fetch_getln(int fd, char **buf, size_t *size, size_t *len) +{ + struct timeval now, timeout, wait; + fd_set readfds; + int r; + char c; + + if (*buf == NULL) { + if ((*buf = malloc(MIN_BUF_SIZE)) == NULL) { + errno = ENOMEM; + return -1; + } + *size = MIN_BUF_SIZE; + } + + **buf = '\0'; + *len = 0; + + if (fetchTimeout) { + gettimeofday(&timeout, NULL); + timeout.tv_sec += fetchTimeout; + FD_ZERO(&readfds); + } + + do { + if (fetchTimeout) { + FD_SET(fd, &readfds); + gettimeofday(&now, NULL); + wait.tv_sec = timeout.tv_sec - now.tv_sec; + wait.tv_usec = timeout.tv_usec - now.tv_usec; + if (wait.tv_usec < 0) { + wait.tv_usec += 1000000; + wait.tv_sec--; + } + if (wait.tv_sec < 0) { + errno = ETIMEDOUT; + return -1; + } + r = select(fd+1, &readfds, NULL, NULL, &wait); + if (r == -1) { + if (errno == EINTR) + continue; + /* EBADF or EINVAL: shouldn't happen */ + return -1; + } + if (!FD_ISSET(fd, &readfds)) + continue; + } + r = read(fd, &c, 1); + if (r == 0) + break; + if (r == -1) { + if (errno == EINTR) + continue; + /* any other error is bad news */ + return -1; + } + (*buf)[*len] = c; + *len += 1; + if (*len == *size) { + char *tmp; + + if ((tmp = realloc(*buf, *size * 2 + 1)) == NULL) { + errno = ENOMEM; + return -1; + } + *buf = tmp; + *size = *size * 2 + 1; + } + } while (c != '\n'); + + return 0; +} + + /*** Directory-related utility functions *************************************/ int |