summaryrefslogtreecommitdiffstats
path: root/lib/libc/stdio/fopen.c
diff options
context:
space:
mode:
authorjhb <jhb@FreeBSD.org>2008-02-27 19:02:02 +0000
committerjhb <jhb@FreeBSD.org>2008-02-27 19:02:02 +0000
commit4c65fa8afdeb144b1a2be9476db19538f56c60ba (patch)
tree636eaa9d47e5f42f8f59cf335161c68af6b7300a /lib/libc/stdio/fopen.c
parent76461e66bc08ee42c8b9bf6eb8b1a6c92b3b9d82 (diff)
downloadFreeBSD-src-4c65fa8afdeb144b1a2be9476db19538f56c60ba.zip
FreeBSD-src-4c65fa8afdeb144b1a2be9476db19538f56c60ba.tar.gz
File descriptors are an int, but our stdio FILE object uses a short to hold
them. Thus, any fd whose value is greater than SHRT_MAX is handled incorrectly (the short value is sign-extended when converted to an int). An unpleasant side effect is that if fopen() opens a file and gets a backing fd that is greater than SHRT_MAX, fclose() will fail and the file descriptor will be leaked. Better handle this by fixing fopen(), fdopen(), and freopen() to fail attempts to use a fd greater than SHRT_MAX with EMFILE. At some point in the future we should look at expanding the file descriptor in FILE to an int, but that is a bit complicated due to ABI issues. MFC after: 1 week Discussed on: arch Reviewed by: wollman
Diffstat (limited to 'lib/libc/stdio/fopen.c')
-rw-r--r--lib/libc/stdio/fopen.c13
1 files changed, 13 insertions, 0 deletions
diff --git a/lib/libc/stdio/fopen.c b/lib/libc/stdio/fopen.c
index 9cedcda..a6c0028 100644
--- a/lib/libc/stdio/fopen.c
+++ b/lib/libc/stdio/fopen.c
@@ -40,6 +40,7 @@ __FBSDID("$FreeBSD$");
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
+#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include "un-namespace.h"
@@ -63,6 +64,18 @@ fopen(file, mode)
fp->_flags = 0; /* release */
return (NULL);
}
+ /*
+ * File descriptors are a full int, but _file is only a short.
+ * If we get a valid file descriptor that is greater than
+ * SHRT_MAX, then the fd will get sign-extended into an
+ * invalid file descriptor. Handle this case by failing the
+ * open.
+ */
+ if (f > SHRT_MAX) {
+ _close(f);
+ errno = EMFILE;
+ return (NULL);
+ }
fp->_file = f;
fp->_flags = flags;
fp->_cookie = fp;
OpenPOWER on IntegriCloud