summaryrefslogtreecommitdiff
path: root/lib/xgetcwd.c
diff options
context:
space:
mode:
authorJim Meyering <jim@meyering.net>2000-06-21 20:16:03 +0000
committerJim Meyering <jim@meyering.net>2000-06-21 20:16:03 +0000
commit978bf5f7509a9e48078886fd62d7a258dab12efc (patch)
tree30df2d081e82d504e0fad23c3f30843d0fce332a /lib/xgetcwd.c
parent0c3c6ef55e5cbce18cd1bd37c5f6fc64fa6d289d (diff)
downloadcoreutils-978bf5f7509a9e48078886fd62d7a258dab12efc.tar.xz
(xgetcwd): If the required pathname length is smaller
than 1024, return a memory chunk of least possible size, instead of size PATH_MAX + 2. In the loop, increment the size proportionally. Use free/xmalloc instead of xrealloc to avoid copying for very long paths.
Diffstat (limited to 'lib/xgetcwd.c')
-rw-r--r--lib/xgetcwd.c48
1 files changed, 27 insertions, 21 deletions
diff --git a/lib/xgetcwd.c b/lib/xgetcwd.c
index 7ab220468..896be5d6a 100644
--- a/lib/xgetcwd.c
+++ b/lib/xgetcwd.c
@@ -1,5 +1,5 @@
/* xgetcwd.c -- return current directory with unlimited length
- Copyright (C) 1992, 1996 Free Software Foundation, Inc.
+ Copyright (C) 1992, 1996, 2000 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
@@ -37,12 +37,9 @@ char *getwd ();
# define getcwd(Buf, Max) getwd (Buf)
#endif
-/* Amount to increase buffer size by in each try. */
-#define PATH_INCR 32
-
-char *xmalloc ();
-char *xrealloc ();
-void free ();
+extern void *xmalloc ();
+extern char *xstrdup ();
+extern void free ();
/* Return the current directory, newly allocated, arbitrarily long.
Return NULL and set errno on error. */
@@ -50,30 +47,39 @@ void free ();
char *
xgetcwd ()
{
- char *cwd;
char *ret;
unsigned path_max;
+ char buf[1024];
errno = 0;
+ ret = getcwd (buf, sizeof (buf));
+ if (ret != NULL)
+ return xstrdup (buf);
+ if (errno != ERANGE)
+ return NULL;
+
path_max = (unsigned) PATH_MAX;
path_max += 2; /* The getcwd docs say to do this. */
- cwd = xmalloc (path_max);
-
- errno = 0;
- while ((ret = getcwd (cwd, path_max)) == NULL && errno == ERANGE)
+ for (;;)
{
- path_max += PATH_INCR;
- cwd = xrealloc (cwd, path_max);
+ char *cwd = (char *) xmalloc (path_max);
+
errno = 0;
- }
+ ret = getcwd (cwd, path_max);
+ if (ret != NULL)
+ return ret;
+ if (errno != ERANGE)
+ {
+ int save_errno = errno;
+ free (cwd);
+ errno = save_errno;
+ return NULL;
+ }
- if (ret == NULL)
- {
- int save_errno = errno;
free (cwd);
- errno = save_errno;
- return NULL;
+
+ path_max += path_max / 16;
+ path_max += 32;
}
- return cwd;
}