Sync from coreutils.
[gnulib.git] / lib / xgethostname.c
1 /* xgethostname.c -- return current hostname with unlimited length
2
3    Copyright (C) 1992, 1996, 2000, 2001, 2003, 2004, 2005 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
19
20 /* written by Jim Meyering */
21
22 #ifdef HAVE_CONFIG_H
23 # include <config.h>
24 #endif
25
26 /* Specification.  */
27 #include "xgethostname.h"
28
29 #include <stdlib.h>
30 #include <errno.h>
31 #include <unistd.h>
32
33 #include "xalloc.h"
34
35 #ifndef ENAMETOOLONG
36 # define ENAMETOOLONG 0
37 #endif
38
39 #ifndef INITIAL_HOSTNAME_LENGTH
40 # define INITIAL_HOSTNAME_LENGTH 34
41 #endif
42
43 /* Return the current hostname in malloc'd storage.
44    If malloc fails, exit.
45    Upon any other failure, return NULL and set errno.  */
46 char *
47 xgethostname (void)
48 {
49   char *hostname = NULL;
50   size_t size = INITIAL_HOSTNAME_LENGTH;
51
52   while (1)
53     {
54       /* Use SIZE_1 here rather than SIZE to work around the bug in
55          SunOS 5.5's gethostname whereby it NUL-terminates HOSTNAME
56          even when the name is as long as the supplied buffer.  */
57       size_t size_1;
58
59       hostname = x2realloc (hostname, &size);
60       size_1 = size - 1;
61       hostname[size_1 - 1] = '\0';
62       errno = 0;
63
64       if (gethostname (hostname, size_1) == 0)
65         {
66           if (! hostname[size_1 - 1])
67             break;
68         }
69       else if (errno != 0 && errno != ENAMETOOLONG && errno != EINVAL
70                /* OSX/Darwin does this when the buffer is not large enough */
71                && errno != ENOMEM)
72         {
73           int saved_errno = errno;
74           free (hostname);
75           errno = saved_errno;
76           return NULL;
77         }
78     }
79
80   return hostname;
81 }