New module 'memmem', from Simon Josefsson.
[gnulib.git] / 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 <stdio.h>
27 #include <errno.h>
28 #include <sys/types.h>
29 #include <stdlib.h>
30
31 #if HAVE_UNISTD_H
32 # include <unistd.h>
33 #endif
34
35 #if HAVE_GETCWD
36 char *getcwd ();
37 #else
38 # include "pathmax.h"
39 # define INITIAL_BUFFER_SIZE (PATH_MAX + 1)
40 char *getwd ();
41 # define getcwd(Buf, Max) getwd (Buf)
42 #endif
43
44 #include "xalloc.h"
45 #include "xgetcwd.h"
46
47 /* Return the current directory, newly allocated, assuming it fits
48    within PATH_MAX bytes -- this is a common system-imposed limit
49    on how getcwd works.
50    Upon an out-of-memory error, call xalloc_die.
51    Upon any other type of error, return NULL.  */
52
53 char *
54 xgetcwd (void)
55 {
56 #if HAVE_GETCWD_NULL
57   char *cwd = getcwd (NULL, 0);
58   if (! cwd && errno == ENOMEM)
59     xalloc_die ();
60   return cwd;
61 #else
62
63   /* The initial buffer size for the working directory.  A power of 2
64      detects arithmetic overflow earlier, but is not required.  */
65 # ifndef INITIAL_BUFFER_SIZE
66 #  define INITIAL_BUFFER_SIZE 128
67 # endif
68
69   size_t buf_size = INITIAL_BUFFER_SIZE;
70
71   while (1)
72     {
73       char *buf = xmalloc (buf_size);
74       char *cwd = getcwd (buf, buf_size);
75       int saved_errno;
76       if (cwd)
77         return cwd;
78       saved_errno = errno;
79       free (buf);
80       if (saved_errno != ERANGE)
81         return NULL;
82       buf_size *= 2;
83       if (buf_size == 0)
84         xalloc_die ();
85     }
86 #endif
87 }