summaryrefslogtreecommitdiff
path: root/sysdeps/unix
diff options
context:
space:
mode:
authorUlrich Drepper <drepper@redhat.com>1999-08-20 18:07:55 +0000
committerUlrich Drepper <drepper@redhat.com>1999-08-20 18:07:55 +0000
commit7c9b5e4d47ed18af796e94e799688a77d3632452 (patch)
tree31211ea6bef504737d540a8dbaebc0c8b6e71e91 /sysdeps/unix
parent60c74efa35e648891c3af8fb5ed89b064338879a (diff)
Allocate buffer large enough not only if size == 0, but also for size < 0.
Diffstat (limited to 'sysdeps/unix')
-rw-r--r--sysdeps/unix/sysv/linux/getcwd.c51
1 files changed, 41 insertions, 10 deletions
diff --git a/sysdeps/unix/sysv/linux/getcwd.c b/sysdeps/unix/sysv/linux/getcwd.c
index fe28e2d894..223cd310b8 100644
--- a/sysdeps/unix/sysv/linux/getcwd.c
+++ b/sysdeps/unix/sysv/linux/getcwd.c
@@ -18,6 +18,7 @@
write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Boston, MA 02111-1307, USA. */
+#include <assert.h>
#include <errno.h>
#include <limits.h>
#include <stdlib.h>
@@ -26,7 +27,19 @@
#include <sysdep.h>
#include <sys/syscall.h>
+#include "kernel-features.h"
+
+#if __ASSUME_GETCWD_SYSCALL > 0
+/* Kernel 2.1.92 introduced a third way to get the current working
+ directory: a syscall. We've got to be careful that even when
+ compiling under 2.1.92+ the libc still runs under older kernels. */
+extern int __syscall_getcwd (char *buf, unsigned long size);
+# define no_syscall_getcwd 0
+# define have_new_dcache 1
+/* This is a trick since we don't define generic_getcwd. */
+# define generic_getcwd getcwd
+#else
/* The "proc" filesystem provides an easy method to retrieve the value.
For each process, the corresponding directory contains a symbolic link
named `cwd'. Reading the content of this link immediate gives us the
@@ -34,16 +47,17 @@
the proc filesystem mounted. Use the POSIX implementation in this case. */
static char *generic_getcwd (char *buf, size_t size) internal_function;
-#ifdef __NR_getcwd
+# if __NR_getcwd
/* Kernel 2.1.92 introduced a third way to get the current working
directory: a syscall. We've got to be careful that even when
compiling under 2.1.92+ the libc still runs under older kernels. */
extern int __syscall_getcwd (char *buf, unsigned long size);
static int no_syscall_getcwd;
static int have_new_dcache;
-#else
-# define no_syscall_getcwd 1
+# else
+# define no_syscall_getcwd 1
static int have_new_dcache = 1;
+# endif
#endif
char *
@@ -58,7 +72,7 @@ __getcwd (char *buf, size_t size)
if (no_syscall_getcwd && !have_new_dcache)
return generic_getcwd (buf, size);
- if (size == 0)
+ if (size <= 0)
{
if (buf != NULL)
{
@@ -80,7 +94,7 @@ __getcwd (char *buf, size_t size)
save_errno = errno;
-#ifdef __NR_getcwd
+#if defined __NR_getcwd || __LINUX_GETCWD_SYSCALL > 0
if (!no_syscall_getcwd)
{
int retval;
@@ -98,6 +112,16 @@ __getcwd (char *buf, size_t size)
return buf;
}
+# if __ASSUME_GETCWD_SYSCALL
+ /* It should never happen that the `getcwd' syscall failed because
+ the buffer is too small if we allocated the buffer outself. */
+ assert (errno != ERANGE || buf != NULL);
+
+ if (buf == NULL)
+ free (path);
+
+ return NULL;
+# else
if (errno == ENOSYS)
{
no_syscall_getcwd = 1;
@@ -111,6 +135,7 @@ __getcwd (char *buf, size_t size)
}
__set_errno (save_errno);
+# endif
}
#endif
@@ -136,21 +161,25 @@ __getcwd (char *buf, size_t size)
}
return buf;
}
+#ifndef have_new_dcache
else
have_new_dcache = 0;
+#endif
}
+#if __ASSUME_GETCWD_SYSCALL == 0
/* Set to have_new_dcache only if error indicates that proc doesn't
exist. */
if (errno != EACCES && errno != ENAMETOOLONG)
have_new_dcache = 0;
+#endif
/* Something went wrong. Restore the error number and use the generic
version. */
__set_errno (save_errno);
/* Don't put restrictions on the length of the path unless the user does. */
- if (size == 0)
+ if (size <= 0)
{
free (path);
path = NULL;
@@ -158,14 +187,16 @@ __getcwd (char *buf, size_t size)
result = generic_getcwd (path, size);
- if (result == NULL && buf == NULL && size != 0)
+ if (result == NULL && buf == NULL && size > 0)
free (path);
return result;
}
weak_alias (__getcwd, getcwd)
+#if __ASSUME_GETCWD_SYSCALL == 0
/* Get the code for the generic version. */
-#define GETCWD_RETURN_TYPE static char * internal_function
-#define __getcwd generic_getcwd
-#include <sysdeps/posix/getcwd.c>
+# define GETCWD_RETURN_TYPE static char * internal_function
+# define __getcwd generic_getcwd
+# include <sysdeps/posix/getcwd.c>
+#endif