X-Git-Url: http://erislabs.net/gitweb/?a=blobdiff_plain;f=lib%2Fsafe-read.c;h=25a4e2f39b24179bb1cc54c64f1f9c7126426cc0;hb=387c96fc38b1985d677a25881ab99a449a00af22;hp=79d85417d867c2a6cdcb92ee17f94869fa337676;hpb=849efe7683e163ede9240b27d675977c71c76ce8;p=gnulib.git diff --git a/lib/safe-read.c b/lib/safe-read.c index 79d85417d..25a4e2f39 100644 --- a/lib/safe-read.c +++ b/lib/safe-read.c @@ -1,10 +1,12 @@ -/* An interface to read() that retries after interrupts. - Copyright (C) 1993, 1994, 1998, 2002 Free Software Foundation, Inc. +/* An interface to read and write that retries after interrupts. - This program is free software; you can redistribute it and/or modify + Copyright (C) 1993-1994, 1998, 2002-2006, 2009-2012 Free Software + Foundation, Inc. + + This program 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, or (at your option) - any later version. + the Free Software Foundation; either version 3 of the License, or + (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of @@ -12,72 +14,64 @@ 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. */ + along with this program. If not, see . */ -#if HAVE_CONFIG_H -# include -#endif +#include /* Specification. */ -#include "safe-read.h" +#ifdef SAFE_WRITE +# include "safe-write.h" +#else +# include "safe-read.h" +#endif /* Get ssize_t. */ #include -#if HAVE_UNISTD_H -# include -#endif +#include #include -#ifndef errno -extern int errno; + +#ifdef EINTR +# define IS_EINTR(x) ((x) == EINTR) +#else +# define IS_EINTR(x) 0 #endif #include -/* We don't pass an nbytes count > SSIZE_MAX to read() - POSIX says the - effect would be implementation-defined. Also we don't pass an nbytes - count > INT_MAX but <= SSIZE_MAX to read() - this triggers a bug in - Tru64 5.1. */ -#define MAX_BYTES_TO_READ INT_MAX +#ifdef SAFE_WRITE +# define safe_rw safe_write +# define rw write +#else +# define safe_rw safe_read +# define rw read +# undef const +# define const /* empty */ +#endif -/* Read up to COUNT bytes at BUF from descriptor FD, retrying if interrupted. - Return the actual number of bytes read, zero for EOF, or (size_t) -1 - for an error. */ +/* Read(write) up to COUNT bytes at BUF from(to) descriptor FD, retrying if + interrupted. Return the actual number of bytes read(written), zero for EOF, + or SAFE_READ_ERROR(SAFE_WRITE_ERROR) upon error. */ size_t -safe_read (int fd, void *buf, size_t count) +safe_rw (int fd, void const *buf, size_t count) { - size_t total_read = 0; + /* Work around a bug in Tru64 5.1. Attempting to read more than + INT_MAX bytes fails with errno == EINVAL. See + . + When decreasing COUNT, keep it block-aligned. */ + enum { BUGGY_READ_MAXIMUM = INT_MAX & ~8191 }; - if (count > 0) + for (;;) { - char *ptr = (char *) buf; - do - { - size_t nbytes_to_read = count; - ssize_t result; - - /* Limit the number of bytes to read in one round, to avoid running - into unspecified behaviour. But keep the file pointer block - aligned when doing so. */ - if (nbytes_to_read > MAX_BYTES_TO_READ) - nbytes_to_read = MAX_BYTES_TO_READ & ~8191; + ssize_t result = rw (fd, buf, count); - result = read (fd, ptr, nbytes_to_read); - if (result < 0) - { -#ifdef EINTR - if (errno == EINTR) - continue; -#endif - return result; - } - total_read += result; - ptr += result; - count -= result; - } - while (count > 0); + if (0 <= result) + return result; + else if (IS_EINTR (errno)) + continue; + else if (errno == EINVAL && BUGGY_READ_MAXIMUM < count) + count = BUGGY_READ_MAXIMUM; + else + return result; } - - return total_read; }