diff options
author | ache <ache@FreeBSD.org> | 2001-08-15 02:07:47 +0000 |
---|---|---|
committer | ache <ache@FreeBSD.org> | 2001-08-15 02:07:47 +0000 |
commit | 9d7273169f482f1f74606c9f559ecc986f668088 (patch) | |
tree | c651de630103d0b82ee727fa499a87af0697e6f6 /lib | |
parent | 679dd2c9f81d6f490f3db2613cb62f20c7cf93ab (diff) | |
download | FreeBSD-src-9d7273169f482f1f74606c9f559ecc986f668088.zip FreeBSD-src-9d7273169f482f1f74606c9f559ecc986f668088.tar.gz |
1) Disallow negative seek as POSIX require for fseek{o} (but not for lseek):
"[EINVAL] ... The resulting file-position indicator would be set to a
negative value."
Moreover, in real life negative seek in stdio cause EOF indicator cleared
and not set again forever even if EOF returned.
2) Catch few possible off_t overflows.
Reviewed by: arch discussion
Diffstat (limited to 'lib')
-rw-r--r-- | lib/libc/stdio/fseek.3 | 16 | ||||
-rw-r--r-- | lib/libc/stdio/fseek.c | 23 | ||||
-rw-r--r-- | lib/libc/stdio/stdio.c | 19 |
3 files changed, 45 insertions, 13 deletions
diff --git a/lib/libc/stdio/fseek.3 b/lib/libc/stdio/fseek.3 index 9a58668..b37aa17 100644 --- a/lib/libc/stdio/fseek.3 +++ b/lib/libc/stdio/fseek.3 @@ -180,18 +180,12 @@ is not a seekable stream. .It Bq Er EINVAL The .Fa whence -argument to -.Fn fseek -was not -.Dv SEEK_SET , -.Dv SEEK_END , -or -.Dv SEEK_CUR . +argument is invalid. +The resulting file-position +indicator would be set to a negative value. .It Bq Er EOVERFLOW -For -.Fn ftell , -the resulting file offset would be a value which -cannot be represented correctly in an object of type long. +The resulting file offset would be a value which +cannot be represented correctly in an object of type requested. .El .Pp The functions diff --git a/lib/libc/stdio/fseek.c b/lib/libc/stdio/fseek.c index b434350..d311f7a 100644 --- a/lib/libc/stdio/fseek.c +++ b/lib/libc/stdio/fseek.c @@ -132,12 +132,26 @@ _fseeko(fp, offset, whence) } else if (fp->_flags & __SWR && fp->_p != NULL) curoff += fp->_p - fp->_bf._base; + if (offset > 0 && offset + (off_t)curoff < 0) { + errno = EOVERFLOW; + return (EOF); + } offset += curoff; + /* Disallow negative seeks per POSIX */ + if (offset < 0) { + errno = EINVAL; + return (EOF); + } whence = SEEK_SET; havepos = 1; break; case SEEK_SET: + /* Disallow negative seeks per POSIX */ + if (offset < 0) { + errno = EINVAL; + return (EOF); + } case SEEK_END: curoff = 0; /* XXX just to keep gcc quiet */ havepos = 0; @@ -180,7 +194,16 @@ _fseeko(fp, offset, whence) else { if (_fstat(fp->_file, &st)) goto dumb; + if (offset > 0 && st.st_size + offset < 0) { + errno = EOVERFLOW; + return (EOF); + } target = st.st_size + offset; + /* Disallow negative seeks per POSIX */ + if ((off_t)target < 0) { + errno = EINVAL; + return (EOF); + } } if (!havepos) { diff --git a/lib/libc/stdio/stdio.c b/lib/libc/stdio/stdio.c index 57be5bb..58fafcb 100644 --- a/lib/libc/stdio/stdio.c +++ b/lib/libc/stdio/stdio.c @@ -43,6 +43,7 @@ static const char rcsid[] = #endif /* LIBC_SCCS and not lint */ #include "namespace.h" +#include <errno.h> #include <fcntl.h> #include <unistd.h> #include <stdio.h> @@ -94,10 +95,24 @@ __sseek(cookie, offset, whence) register FILE *fp = cookie; register off_t ret; + /* + * Disallow negative seeks per POSIX. + * It is needed here to help upper level caller + * (fseek) in the cases it can't detect. + */ + if (whence == SEEK_SET && (off_t)offset < 0) { + errno = EINVAL; + return (-1); + } ret = lseek(fp->_file, (off_t)offset, whence); - if (ret == -1) + if (ret < 0) { + if (ret != -1) { + /* Resulting seek is negative! */ + ret = -1; + errno = EINVAL; + } fp->_flags &= ~__SOFF; - else { + } else { fp->_flags |= __SOFF; fp->_offset = ret; } |