Add CVS 1.12.11.
[dragonfly.git] / contrib / cvs-1.12.11 / lib / xgetcwd.c
1 /* xgetcwd.c -- return current directory with unlimited length
2
3    Copyright (C) 1992, 1996, 2000, 2001, 2003, 2004 Free Software
4    Foundation, Inc.
5
6    This program is free software; you can redistribute it and/or modify
7    it under the terms of the GNU General Public License as published by
8    the Free Software Foundation; either version 2, or (at your option)
9    any later version.
10
11    This program is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14    GNU General Public License for more details.
15
16    You should have received a copy of the GNU General Public License
17    along with this program; if not, write to the Free Software Foundation,
18    Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
19
20 /* Written by David MacKenzie <djm@gnu.ai.mit.edu>.  */
21
22 #if HAVE_CONFIG_H
23 # include <config.h>
24 #endif
25
26 #include <limits.h>
27 #include <stdio.h>
28 #include <errno.h>
29 #include <sys/types.h>
30 #include <stdlib.h>
31
32 #if HAVE_UNISTD_H
33 # include <unistd.h>
34 #endif
35
36 #if HAVE_GETCWD
37 char *getcwd ();
38 #else
39 # include "pathmax.h"
40 # define INITIAL_BUFFER_SIZE (PATH_MAX + 1)
41 char *getwd ();
42 # define getcwd(Buf, Max) getwd (Buf)
43 #endif
44
45 #include "xalloc.h"
46 #include "xgetcwd.h"
47
48 /* Return the current directory, newly allocated, assuming it fits
49    within PATH_MAX bytes -- this is a common system-imposed limit
50    on how getcwd works.
51    Upon an out-of-memory error, call xalloc_die.
52    Upon any other type of error, return NULL.  */
53
54 char *
55 xgetcwd (void)
56 {
57 #if HAVE_GETCWD_NULL
58   char *cwd = getcwd (NULL, 0);
59   if (! cwd && errno == ENOMEM)
60     xalloc_die ();
61   return cwd;
62 #else
63
64   int saved_errno;
65
66   /* The initial buffer size for the working directory.  A power of 2
67      detects arithmetic overflow earlier, but is not required.  */
68 # ifndef INITIAL_BUFFER_SIZE
69 #  define INITIAL_BUFFER_SIZE 128
70 # endif
71
72   size_t buf_size = INITIAL_BUFFER_SIZE;
73
74   while (1)
75     {
76       char *buf = xmalloc (buf_size);
77       char *cwd = getcwd (buf, buf_size);
78       if (cwd)
79         return cwd;
80       saved_errno = errno;
81       free (buf);
82       if (saved_errno != ERANGE)
83         break;
84
85 #ifdef PATH_MAX
86       if (PATH_MAX / 2 < buf_size)
87         {
88           if (PATH_MAX <= buf_size)
89             break;
90           buf_size = PATH_MAX;
91           continue;
92         }
93 #endif
94
95       buf_size *= 2;
96       if (buf_size == 0)
97         xalloc_die ();
98     }
99
100   errno = saved_errno;
101   return NULL;
102 #endif
103 }