annotate lib/areadlink.c @ 11224:5aa57cee93aa

avoid gcc 3.4.3 bug on long double NaN on Irix 6.5 * tests/nan.h (NaNl): Rewrite as function on Irix, to avoid compilation bug by using runtime conversion. * m4/isfinite.m4 (gl_ISFINITE): Likewise. * m4/isnanl.m4 (gl_FUNC_ISNANL): Likewise. * modules/ceill-tests (Files): Use nan.h. * modules/floorl-tests (Files): Likewise. * modules/frexpl-tests (Files): Likewise. * modules/isnanl-tests (Files): Likewise. * modules/ldexpl-tests (Files): Likewise. * modules/roundl-tests (Files): Likewise. * modules/truncl-tests (Files): Likewise. * tests/test-ceill.c (main): Use a working NaN. * tests/test-floorl.c (main): Likewise. * tests/test-frexpl.c (main): Likewise. * tests/test-isnan.c (test_long_double): Likewise. * tests/test-isnanl.h (main): Likewise. * tests/test-ldexpl.h (main): Likewise. * tests/test-roundl.h (main): Likewise. * tests/test-truncl.h (main): Likewise. See http://lists.gnu.org/archive/html/bug-gnulib/2009-02/msg00190.html. Signed-off-by: Eric Blake <ebb9@byu.net>
author Eric Blake <ebb9@byu.net>
date Thu, 26 Feb 2009 20:18:42 -0700
parents bbbbbf4cd1c5
children 1fd105be1362
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
9217
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
1 /* areadlink.c -- readlink wrapper to return the link name in malloc'd storage
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
2 Unlike xreadlink and xreadlink_with_size, don't ever call exit.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
3
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
4 Copyright (C) 2001, 2003-2007 Free Software Foundation, Inc.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
5
9309
bbbbbf4cd1c5 Change copyright notice from GPLv2+ to GPLv3+.
Bruno Haible <bruno@clisp.org>
parents: 9217
diff changeset
6 This program is free software: you can redistribute it and/or modify
9217
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
7 it under the terms of the GNU General Public License as published by
9309
bbbbbf4cd1c5 Change copyright notice from GPLv2+ to GPLv3+.
Bruno Haible <bruno@clisp.org>
parents: 9217
diff changeset
8 the Free Software Foundation; either version 3 of the License, or
bbbbbf4cd1c5 Change copyright notice from GPLv2+ to GPLv3+.
Bruno Haible <bruno@clisp.org>
parents: 9217
diff changeset
9 (at your option) any later version.
9217
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
10
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
11 This program is distributed in the hope that it will be useful,
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
14 GNU General Public License for more details.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
15
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
16 You should have received a copy of the GNU General Public License
9309
bbbbbf4cd1c5 Change copyright notice from GPLv2+ to GPLv3+.
Bruno Haible <bruno@clisp.org>
parents: 9217
diff changeset
17 along with this program. If not, see <http://www.gnu.org/licenses/>. */
9217
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
18
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
19 /* Written by Jim Meyering <jim@meyering.net>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
20 and Bruno Haible <bruno@clisp.org>. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
21
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
22 #include <config.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
23
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
24 /* Specification. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
25 #include "areadlink.h"
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
26
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
27 #include <string.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
28 #include <errno.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
29 #include <limits.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
30 #include <sys/types.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
31 #include <stdlib.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
32 #include <unistd.h>
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
33
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
34 #ifndef SIZE_MAX
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
35 # define SIZE_MAX ((size_t) -1)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
36 #endif
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
37 #ifndef SSIZE_MAX
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
38 # define SSIZE_MAX ((ssize_t) (SIZE_MAX / 2))
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
39 #endif
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
40
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
41 /* Call readlink to get the symbolic link value of FILENAME.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
42 Return a pointer to that NUL-terminated string in malloc'd storage.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
43 If readlink fails, return NULL and set errno.
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
44 If realloc fails, or if the link value is longer than SIZE_MAX :-),
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
45 return NULL and set errno to ENOMEM. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
46
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
47 char *
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
48 areadlink (char const *filename)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
49 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
50 /* The initial buffer size for the link value. A power of 2
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
51 detects arithmetic overflow earlier, but is not required. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
52 #define INITIAL_BUF_SIZE 1024
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
53
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
54 /* Allocate the initial buffer on the stack. This way, in the common
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
55 case of a symlink of small size, we get away with a single small malloc()
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
56 instead of a big malloc() followed by a shrinking realloc(). */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
57 char initial_buf[INITIAL_BUF_SIZE];
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
58
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
59 char *buffer = initial_buf;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
60 size_t buf_size = sizeof (initial_buf);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
61
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
62 while (1)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
63 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
64 /* Attempt to read the link into the current buffer. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
65 ssize_t link_length = readlink (filename, buffer, buf_size);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
66
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
67 /* On AIX 5L v5.3 and HP-UX 11i v2 04/09, readlink returns -1
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
68 with errno == ERANGE if the buffer is too small. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
69 if (link_length < 0 && errno != ERANGE)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
70 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
71 if (buffer != initial_buf)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
72 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
73 int saved_errno = errno;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
74 free (buffer);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
75 errno = saved_errno;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
76 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
77 return NULL;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
78 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
79
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
80 if ((size_t) link_length < buf_size)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
81 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
82 buffer[link_length++] = '\0';
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
83
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
84 /* Return it in a chunk of memory as small as possible. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
85 if (buffer == initial_buf)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
86 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
87 buffer = (char *) malloc (link_length);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
88 if (buffer == NULL)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
89 /* errno is ENOMEM. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
90 return NULL;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
91 memcpy (buffer, initial_buf, link_length);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
92 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
93 else
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
94 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
95 /* Shrink buffer before returning it. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
96 if ((size_t) link_length < buf_size)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
97 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
98 char *smaller_buffer = (char *) realloc (buffer, link_length);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
99
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
100 if (smaller_buffer != NULL)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
101 buffer = smaller_buffer;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
102 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
103 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
104 return buffer;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
105 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
106
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
107 if (buffer != initial_buf)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
108 free (buffer);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
109 buf_size *= 2;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
110 if (SSIZE_MAX < buf_size || (SIZE_MAX / 2 < SSIZE_MAX && buf_size == 0))
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
111 {
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
112 errno = ENOMEM;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
113 return NULL;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
114 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
115 buffer = (char *) malloc (buf_size);
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
116 if (buffer == NULL)
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
117 /* errno is ENOMEM. */
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
118 return NULL;
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
119 }
71cb1dd07dcd New module 'areadlink'. File based on lib/xreadlink.c.
Bruno Haible <bruno@clisp.org>
parents:
diff changeset
120 }