Absolute File Name: | /home/opencoverage/opencoverage/guest-scripts/coreutils/src/gnulib/lib/canonicalize.c |
Source code | Switch to Preprocessed file |
Line | Source | Count | ||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
1 | /* Return the canonical absolute name of a given file. | - | ||||||||||||||||||
2 | Copyright (C) 1996-2018 Free Software Foundation, Inc. | - | ||||||||||||||||||
3 | - | |||||||||||||||||||
4 | This program is free software: you can redistribute it and/or modify | - | ||||||||||||||||||
5 | it under the terms of the GNU General Public License as published by | - | ||||||||||||||||||
6 | the Free Software Foundation; either version 3 of the License, or | - | ||||||||||||||||||
7 | (at your option) any later version. | - | ||||||||||||||||||
8 | - | |||||||||||||||||||
9 | This program is distributed in the hope that it will be useful, | - | ||||||||||||||||||
10 | but WITHOUT ANY WARRANTY; without even the implied warranty of | - | ||||||||||||||||||
11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | - | ||||||||||||||||||
12 | GNU General Public License for more details. | - | ||||||||||||||||||
13 | - | |||||||||||||||||||
14 | You should have received a copy of the GNU General Public License | - | ||||||||||||||||||
15 | along with this program. If not, see <https://www.gnu.org/licenses/>. */ | - | ||||||||||||||||||
16 | - | |||||||||||||||||||
17 | #include <config.h> | - | ||||||||||||||||||
18 | - | |||||||||||||||||||
19 | #include "canonicalize.h" | - | ||||||||||||||||||
20 | - | |||||||||||||||||||
21 | #include <errno.h> | - | ||||||||||||||||||
22 | #include <stdlib.h> | - | ||||||||||||||||||
23 | #include <string.h> | - | ||||||||||||||||||
24 | #include <sys/stat.h> | - | ||||||||||||||||||
25 | #include <unistd.h> | - | ||||||||||||||||||
26 | - | |||||||||||||||||||
27 | #include "areadlink.h" | - | ||||||||||||||||||
28 | #include "file-set.h" | - | ||||||||||||||||||
29 | #include "hash-triple.h" | - | ||||||||||||||||||
30 | #include "pathmax.h" | - | ||||||||||||||||||
31 | #include "xalloc.h" | - | ||||||||||||||||||
32 | #include "xgetcwd.h" | - | ||||||||||||||||||
33 | #include "dosname.h" | - | ||||||||||||||||||
34 | - | |||||||||||||||||||
35 | #define MULTIPLE_BITS_SET(i) (((i) & ((i) - 1)) != 0) | - | ||||||||||||||||||
36 | - | |||||||||||||||||||
37 | /* In this file, we cannot handle file names longer than PATH_MAX. | - | ||||||||||||||||||
38 | On systems with no file name length limit, use a fallback. */ | - | ||||||||||||||||||
39 | #ifndef PATH_MAX | - | ||||||||||||||||||
40 | # define PATH_MAX 8192 | - | ||||||||||||||||||
41 | #endif | - | ||||||||||||||||||
42 | - | |||||||||||||||||||
43 | #ifndef DOUBLE_SLASH_IS_DISTINCT_ROOT | - | ||||||||||||||||||
44 | # define DOUBLE_SLASH_IS_DISTINCT_ROOT 0 | - | ||||||||||||||||||
45 | #endif | - | ||||||||||||||||||
46 | - | |||||||||||||||||||
47 | #if ISSLASH ('\\') | - | ||||||||||||||||||
48 | # define SLASHES "/\\" | - | ||||||||||||||||||
49 | #else | - | ||||||||||||||||||
50 | # define SLASHES "/" | - | ||||||||||||||||||
51 | #endif | - | ||||||||||||||||||
52 | - | |||||||||||||||||||
53 | #if !((HAVE_CANONICALIZE_FILE_NAME && FUNC_REALPATH_WORKS) \ | - | ||||||||||||||||||
54 | || GNULIB_CANONICALIZE_LGPL) | - | ||||||||||||||||||
55 | /* Return the canonical absolute name of file NAME. A canonical name | - | ||||||||||||||||||
56 | does not contain any ".", ".." components nor any repeated file name | - | ||||||||||||||||||
57 | separators ('/') or symlinks. All components must exist. | - | ||||||||||||||||||
58 | The result is malloc'd. */ | - | ||||||||||||||||||
59 | - | |||||||||||||||||||
60 | char * | - | ||||||||||||||||||
61 | canonicalize_file_name (const char *name) | - | ||||||||||||||||||
62 | { | - | ||||||||||||||||||
63 | return canonicalize_filename_mode (name, CAN_EXISTING); | - | ||||||||||||||||||
64 | } | - | ||||||||||||||||||
65 | #endif /* !HAVE_CANONICALIZE_FILE_NAME */ | - | ||||||||||||||||||
66 | - | |||||||||||||||||||
67 | /* Return true if we've already seen the triple, <FILENAME, dev, ino>. | - | ||||||||||||||||||
68 | If *HT is not initialized, initialize it. */ | - | ||||||||||||||||||
69 | static bool | - | ||||||||||||||||||
70 | seen_triple (Hash_table **ht, char const *filename, struct stat const *st) | - | ||||||||||||||||||
71 | { | - | ||||||||||||||||||
72 | if (*ht == NULL)
| 45-130 | ||||||||||||||||||
73 | { | - | ||||||||||||||||||
74 | size_t initial_capacity = 7; | - | ||||||||||||||||||
75 | *ht = hash_initialize (initial_capacity, | - | ||||||||||||||||||
76 | NULL, | - | ||||||||||||||||||
77 | triple_hash, | - | ||||||||||||||||||
78 | triple_compare_ino_str, | - | ||||||||||||||||||
79 | triple_free); | - | ||||||||||||||||||
80 | if (*ht == NULL)
| 0-130 | ||||||||||||||||||
81 | xalloc_die (); never executed: xalloc_die (); | 0 | ||||||||||||||||||
82 | } executed 130 times by 3 tests: end of block Executed by:
| 130 | ||||||||||||||||||
83 | - | |||||||||||||||||||
84 | if (seen_file (*ht, filename, st))
| 10-165 | ||||||||||||||||||
85 | return true; executed 10 times by 1 test: return 1 ; Executed by:
| 10 | ||||||||||||||||||
86 | - | |||||||||||||||||||
87 | record_file (*ht, filename, st); | - | ||||||||||||||||||
88 | return false; executed 165 times by 3 tests: return 0 ; Executed by:
| 165 | ||||||||||||||||||
89 | } | - | ||||||||||||||||||
90 | - | |||||||||||||||||||
91 | /* Return the canonical absolute name of file NAME, while treating | - | ||||||||||||||||||
92 | missing elements according to CAN_MODE. A canonical name | - | ||||||||||||||||||
93 | does not contain any ".", ".." components nor any repeated file name | - | ||||||||||||||||||
94 | separators ('/') or, depending on other CAN_MODE flags, symlinks. | - | ||||||||||||||||||
95 | Whether components must exist or not depends on canonicalize mode. | - | ||||||||||||||||||
96 | The result is malloc'd. */ | - | ||||||||||||||||||
97 | - | |||||||||||||||||||
98 | char * | - | ||||||||||||||||||
99 | canonicalize_filename_mode (const char *name, canonicalize_mode_t can_mode) | - | ||||||||||||||||||
100 | { | - | ||||||||||||||||||
101 | char *rname, *dest, *extra_buf = NULL; | - | ||||||||||||||||||
102 | char const *start; | - | ||||||||||||||||||
103 | char const *end; | - | ||||||||||||||||||
104 | char const *rname_limit; | - | ||||||||||||||||||
105 | size_t extra_len = 0; | - | ||||||||||||||||||
106 | Hash_table *ht = NULL; | - | ||||||||||||||||||
107 | int saved_errno; | - | ||||||||||||||||||
108 | int can_flags = can_mode & ~CAN_MODE_MASK; | - | ||||||||||||||||||
109 | bool logical = can_flags & CAN_NOLINKS; | - | ||||||||||||||||||
110 | size_t prefix_len; | - | ||||||||||||||||||
111 | - | |||||||||||||||||||
112 | can_mode &= CAN_MODE_MASK; | - | ||||||||||||||||||
113 | - | |||||||||||||||||||
114 | if (MULTIPLE_BITS_SET (can_mode))
| 0-273 | ||||||||||||||||||
115 | { | - | ||||||||||||||||||
116 | errno = EINVAL; | - | ||||||||||||||||||
117 | return NULL; never executed: return ((void *)0) ; | 0 | ||||||||||||||||||
118 | } | - | ||||||||||||||||||
119 | - | |||||||||||||||||||
120 | if (name == NULL)
| 0-273 | ||||||||||||||||||
121 | { | - | ||||||||||||||||||
122 | errno = EINVAL; | - | ||||||||||||||||||
123 | return NULL; never executed: return ((void *)0) ; | 0 | ||||||||||||||||||
124 | } | - | ||||||||||||||||||
125 | - | |||||||||||||||||||
126 | if (name[0] == '\0')
| 3-270 | ||||||||||||||||||
127 | { | - | ||||||||||||||||||
128 | errno = ENOENT; | - | ||||||||||||||||||
129 | return NULL; executed 3 times by 2 tests: return ((void *)0) ; Executed by:
| 3 | ||||||||||||||||||
130 | } | - | ||||||||||||||||||
131 | - | |||||||||||||||||||
132 | /* This is always zero for Posix hosts, but can be 2 for MS-Windows | - | ||||||||||||||||||
133 | and MS-DOS X:/foo/bar file names. */ | - | ||||||||||||||||||
134 | prefix_len = FILE_SYSTEM_PREFIX_LEN (name); | - | ||||||||||||||||||
135 | - | |||||||||||||||||||
136 | if (!IS_ABSOLUTE_FILE_NAME (name))
| 0-135 | ||||||||||||||||||
137 | { | - | ||||||||||||||||||
138 | rname = xgetcwd (); | - | ||||||||||||||||||
139 | if (!rname)
| 3-132 | ||||||||||||||||||
140 | return NULL; executed 3 times by 1 test: return ((void *)0) ; Executed by:
| 3 | ||||||||||||||||||
141 | dest = strchr (rname, '\0');
| 0-132 | ||||||||||||||||||
142 | if (dest - rname < PATH_MAX)
| 0-132 | ||||||||||||||||||
143 | { | - | ||||||||||||||||||
144 | char *p = xrealloc (rname, PATH_MAX); | - | ||||||||||||||||||
145 | dest = p + (dest - rname); | - | ||||||||||||||||||
146 | rname = p; | - | ||||||||||||||||||
147 | rname_limit = rname + PATH_MAX; | - | ||||||||||||||||||
148 | } executed 132 times by 3 tests: end of block Executed by:
| 132 | ||||||||||||||||||
149 | else | - | ||||||||||||||||||
150 | { | - | ||||||||||||||||||
151 | rname_limit = dest; | - | ||||||||||||||||||
152 | } never executed: end of block | 0 | ||||||||||||||||||
153 | start = name; | - | ||||||||||||||||||
154 | prefix_len = FILE_SYSTEM_PREFIX_LEN (rname); | - | ||||||||||||||||||
155 | } executed 132 times by 3 tests: end of block Executed by:
| 132 | ||||||||||||||||||
156 | else | - | ||||||||||||||||||
157 | { | - | ||||||||||||||||||
158 | rname = xmalloc (PATH_MAX); | - | ||||||||||||||||||
159 | rname_limit = rname + PATH_MAX; | - | ||||||||||||||||||
160 | dest = rname; | - | ||||||||||||||||||
161 | if (prefix_len)
| 0-135 | ||||||||||||||||||
162 | { | - | ||||||||||||||||||
163 | memcpy (rname, name, prefix_len); | - | ||||||||||||||||||
164 | dest += prefix_len; | - | ||||||||||||||||||
165 | } never executed: end of block | 0 | ||||||||||||||||||
166 | *dest++ = '/'; | - | ||||||||||||||||||
167 | if (DOUBLE_SLASH_IS_DISTINCT_ROOT) dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
168 | { dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
169 | if (ISSLASH (name[1]) && !ISSLASH (name[2]) && !prefix_len) dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
170 | *dest++ = '/'; dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
171 | *dest = '\0'; dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
172 | } dead code: { if (((name[1]) == '/') && !((name[2]) == '/') && !prefix_len) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
173 | start = name + prefix_len; | - | ||||||||||||||||||
174 | } executed 135 times by 2 tests: end of block Executed by:
| 135 | ||||||||||||||||||
175 | - | |||||||||||||||||||
176 | for ( ; *start; start = end)
| 124-1351 | ||||||||||||||||||
177 | { | - | ||||||||||||||||||
178 | /* Skip sequence of multiple file name separators. */ | - | ||||||||||||||||||
179 | while (ISSLASH (*start))
| 1121-1351 | ||||||||||||||||||
180 | ++start; executed 1121 times by 3 tests: ++start; Executed by:
| 1121 | ||||||||||||||||||
181 | - | |||||||||||||||||||
182 | /* Find end of component. */ | - | ||||||||||||||||||
183 | for (end = start; *end && !ISSLASH (*end); ++end)
| 271-9435 | ||||||||||||||||||
184 | /* Nothing. */; executed 8355 times by 3 tests: ; Executed by:
| 8355 | ||||||||||||||||||
185 | - | |||||||||||||||||||
186 | if (end - start == 0)
| 81-1270 | ||||||||||||||||||
187 | break; executed 81 times by 2 tests: break; Executed by:
| 81 | ||||||||||||||||||
188 | else if (end - start == 1 && start[0] == '.')
| 94-1036 | ||||||||||||||||||
189 | /* nothing */; executed 94 times by 3 tests: ; Executed by:
| 94 | ||||||||||||||||||
190 | else if (end - start == 2 && start[0] == '.' && start[1] == '.')
| 0-1154 | ||||||||||||||||||
191 | { | - | ||||||||||||||||||
192 | /* Back up to previous component, ignore if at root already. */ | - | ||||||||||||||||||
193 | if (dest > rname + prefix_len + 1)
| 9-13 | ||||||||||||||||||
194 | for (--dest; dest > rname && !ISSLASH (dest[-1]); --dest)
| 0-33 | ||||||||||||||||||
195 | continue; executed 20 times by 3 tests: continue; Executed by:
| 20 | ||||||||||||||||||
196 | if (DOUBLE_SLASH_IS_DISTINCT_ROOT && dest == rname + 1 dead code: dest == rname + 1 | - | ||||||||||||||||||
197 | && !prefix_len && ISSLASH (*dest) && !ISSLASH (dest[1])) dead code: !prefix_len dead code: ((*dest) == '/') dead code: !((dest[1]) == '/') dead code: dest++; | - | ||||||||||||||||||
198 | dest++; dead code: dest++; | - | ||||||||||||||||||
199 | } executed 22 times by 3 tests: end of block Executed by:
| 22 | ||||||||||||||||||
200 | else | - | ||||||||||||||||||
201 | { | - | ||||||||||||||||||
202 | struct stat st; | - | ||||||||||||||||||
203 | - | |||||||||||||||||||
204 | if (!ISSLASH (dest[-1]))
| 265-889 | ||||||||||||||||||
205 | *dest++ = '/'; executed 889 times by 3 tests: *dest++ = '/'; Executed by:
| 889 | ||||||||||||||||||
206 | - | |||||||||||||||||||
207 | if (dest + (end - start) >= rname_limit)
| 0-1154 | ||||||||||||||||||
208 | { | - | ||||||||||||||||||
209 | ptrdiff_t dest_offset = dest - rname; | - | ||||||||||||||||||
210 | size_t new_size = rname_limit - rname; | - | ||||||||||||||||||
211 | - | |||||||||||||||||||
212 | if (end - start + 1 > PATH_MAX)
| 0 | ||||||||||||||||||
213 | new_size += end - start + 1; never executed: new_size += end - start + 1; | 0 | ||||||||||||||||||
214 | else | - | ||||||||||||||||||
215 | new_size += PATH_MAX; never executed: new_size += 4096 ; | 0 | ||||||||||||||||||
216 | rname = xrealloc (rname, new_size); | - | ||||||||||||||||||
217 | rname_limit = rname + new_size; | - | ||||||||||||||||||
218 | - | |||||||||||||||||||
219 | dest = rname + dest_offset; | - | ||||||||||||||||||
220 | } never executed: end of block | 0 | ||||||||||||||||||
221 | - | |||||||||||||||||||
222 | dest = memcpy (dest, start, end - start); | - | ||||||||||||||||||
223 | dest += end - start; | - | ||||||||||||||||||
224 | *dest = '\0'; | - | ||||||||||||||||||
225 | - | |||||||||||||||||||
226 | if (logical && (can_mode == CAN_MISSING))
| 2-1131 | ||||||||||||||||||
227 | { | - | ||||||||||||||||||
228 | /* Avoid the stat in this case as it's inconsequential. | - | ||||||||||||||||||
229 | i.e. we're neither resolving symlinks or testing | - | ||||||||||||||||||
230 | component existence. */ | - | ||||||||||||||||||
231 | st.st_mode = 0; | - | ||||||||||||||||||
232 | } executed 21 times by 1 test: end of block Executed by:
| 21 | ||||||||||||||||||
233 | else if ((logical ? stat (rname, &st) : lstat (rname, &st)) != 0)
| 2-1131 | ||||||||||||||||||
234 | { | - | ||||||||||||||||||
235 | /* FIXME: If errno == EOVERFLOW here, the entry exists. */ | - | ||||||||||||||||||
236 | saved_errno = errno; | - | ||||||||||||||||||
237 | if (can_mode == CAN_EXISTING)
| 18-115 | ||||||||||||||||||
238 | goto error; executed 18 times by 1 test: goto error; Executed by:
| 18 | ||||||||||||||||||
239 | if (can_mode == CAN_ALL_BUT_LAST)
| 36-79 | ||||||||||||||||||
240 | { | - | ||||||||||||||||||
241 | if (end[strspn (end, SLASHES)] || saved_errno != ENOENT)
| 0-20 | ||||||||||||||||||
242 | goto error; executed 16 times by 1 test: goto error; Executed by:
| 16 | ||||||||||||||||||
243 | continue; executed 20 times by 1 test: continue; Executed by:
| 20 | ||||||||||||||||||
244 | } | - | ||||||||||||||||||
245 | st.st_mode = 0; | - | ||||||||||||||||||
246 | } executed 79 times by 3 tests: end of block Executed by:
| 79 | ||||||||||||||||||
247 | - | |||||||||||||||||||
248 | if (S_ISLNK (st.st_mode))
| 175-925 | ||||||||||||||||||
249 | { | - | ||||||||||||||||||
250 | char *buf; | - | ||||||||||||||||||
251 | size_t n, len; | - | ||||||||||||||||||
252 | - | |||||||||||||||||||
253 | /* Detect loops. We cannot use the cycle-check module here, | - | ||||||||||||||||||
254 | since it's actually possible to encounter the same symlink | - | ||||||||||||||||||
255 | more than once in a given traversal. However, encountering | - | ||||||||||||||||||
256 | the same symlink,NAME pair twice does indicate a loop. */ | - | ||||||||||||||||||
257 | if (seen_triple (&ht, name, &st))
| 10-165 | ||||||||||||||||||
258 | { | - | ||||||||||||||||||
259 | if (can_mode == CAN_MISSING)
| 0-10 | ||||||||||||||||||
260 | continue; never executed: continue; | 0 | ||||||||||||||||||
261 | saved_errno = ELOOP; | - | ||||||||||||||||||
262 | goto error; executed 10 times by 1 test: goto error; Executed by:
| 10 | ||||||||||||||||||
263 | } | - | ||||||||||||||||||
264 | - | |||||||||||||||||||
265 | buf = areadlink_with_size (rname, st.st_size); | - | ||||||||||||||||||
266 | if (!buf)
| 0-165 | ||||||||||||||||||
267 | { | - | ||||||||||||||||||
268 | if (can_mode == CAN_MISSING && errno != ENOMEM)
| 0 | ||||||||||||||||||
269 | continue; never executed: continue; | 0 | ||||||||||||||||||
270 | saved_errno = errno; | - | ||||||||||||||||||
271 | goto error; never executed: goto error; | 0 | ||||||||||||||||||
272 | } | - | ||||||||||||||||||
273 | - | |||||||||||||||||||
274 | n = strlen (buf); | - | ||||||||||||||||||
275 | len = strlen (end); | - | ||||||||||||||||||
276 | - | |||||||||||||||||||
277 | if (!extra_len)
| 35-130 | ||||||||||||||||||
278 | { | - | ||||||||||||||||||
279 | extra_len = | - | ||||||||||||||||||
280 | ((n + len + 1) > PATH_MAX) ? (n + len + 1) : PATH_MAX;
| 0-130 | ||||||||||||||||||
281 | extra_buf = xmalloc (extra_len); | - | ||||||||||||||||||
282 | } executed 130 times by 3 tests: end of block Executed by:
| 130 | ||||||||||||||||||
283 | else if ((n + len + 1) > extra_len)
| 0-35 | ||||||||||||||||||
284 | { | - | ||||||||||||||||||
285 | extra_len = n + len + 1; | - | ||||||||||||||||||
286 | extra_buf = xrealloc (extra_buf, extra_len); | - | ||||||||||||||||||
287 | } never executed: end of block | 0 | ||||||||||||||||||
288 | - | |||||||||||||||||||
289 | /* Careful here, end may be a pointer into extra_buf... */ | - | ||||||||||||||||||
290 | memmove (&extra_buf[n], end, len + 1); | - | ||||||||||||||||||
291 | name = end = memcpy (extra_buf, buf, n); | - | ||||||||||||||||||
292 | - | |||||||||||||||||||
293 | if (IS_ABSOLUTE_FILE_NAME (buf))
| 0-147 | ||||||||||||||||||
294 | { | - | ||||||||||||||||||
295 | size_t pfxlen = FILE_SYSTEM_PREFIX_LEN (buf); | - | ||||||||||||||||||
296 | - | |||||||||||||||||||
297 | if (pfxlen)
| 0-18 | ||||||||||||||||||
298 | memcpy (rname, buf, pfxlen); never executed: memcpy (rname, buf, pfxlen); | 0 | ||||||||||||||||||
299 | dest = rname + pfxlen; | - | ||||||||||||||||||
300 | *dest++ = '/'; /* It's an absolute symlink */ | - | ||||||||||||||||||
301 | if (DOUBLE_SLASH_IS_DISTINCT_ROOT) dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
302 | { dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
303 | if (ISSLASH (buf[1]) && !ISSLASH (buf[2]) && !pfxlen) dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
304 | *dest++ = '/'; dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
305 | *dest = '\0'; dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
306 | } dead code: { if (((buf[1]) == '/') && !((buf[2]) == '/') && !pfxlen) *dest++ = '/'; *dest = '\0'; } | - | ||||||||||||||||||
307 | /* Install the new prefix to be in effect hereafter. */ | - | ||||||||||||||||||
308 | prefix_len = pfxlen; | - | ||||||||||||||||||
309 | } executed 18 times by 2 tests: end of block Executed by:
| 18 | ||||||||||||||||||
310 | else | - | ||||||||||||||||||
311 | { | - | ||||||||||||||||||
312 | /* Back up to previous component, ignore if at root | - | ||||||||||||||||||
313 | already: */ | - | ||||||||||||||||||
314 | if (dest > rname + prefix_len + 1)
| 0-147 | ||||||||||||||||||
315 | for (--dest; dest > rname && !ISSLASH (dest[-1]); --dest)
| 0-608 | ||||||||||||||||||
316 | continue; executed 461 times by 3 tests: continue; Executed by:
| 461 | ||||||||||||||||||
317 | if (DOUBLE_SLASH_IS_DISTINCT_ROOT && dest == rname + 1 dead code: dest == rname + 1 | - | ||||||||||||||||||
318 | && ISSLASH (*dest) && !ISSLASH (dest[1]) && !prefix_len) dead code: ((*dest) == '/') dead code: !((dest[1]) == '/') dead code: !prefix_len dead code: dest++; | - | ||||||||||||||||||
319 | dest++; dead code: dest++; | - | ||||||||||||||||||
320 | } executed 147 times by 3 tests: end of block Executed by:
| 147 | ||||||||||||||||||
321 | - | |||||||||||||||||||
322 | free (buf); | - | ||||||||||||||||||
323 | } executed 165 times by 3 tests: end of block Executed by:
| 165 | ||||||||||||||||||
324 | else | - | ||||||||||||||||||
325 | { | - | ||||||||||||||||||
326 | if (!S_ISDIR (st.st_mode) && *end && (can_mode != CAN_MISSING))
| 18-776 | ||||||||||||||||||
327 | { | - | ||||||||||||||||||
328 | saved_errno = ENOTDIR; | - | ||||||||||||||||||
329 | goto error; executed 18 times by 1 test: goto error; Executed by:
| 18 | ||||||||||||||||||
330 | } | - | ||||||||||||||||||
331 | } executed 907 times by 3 tests: end of block Executed by:
| 907 | ||||||||||||||||||
332 | } | - | ||||||||||||||||||
333 | } | - | ||||||||||||||||||
334 | if (dest > rname + prefix_len + 1 && ISSLASH (dest[-1]))
| 2-170 | ||||||||||||||||||
335 | --dest; executed 2 times by 1 test: --dest; Executed by:
| 2 | ||||||||||||||||||
336 | if (DOUBLE_SLASH_IS_DISTINCT_ROOT && dest == rname + 1 && !prefix_len dead code: dest == rname + 1 dead code: !prefix_len | - | ||||||||||||||||||
337 | && ISSLASH (*dest) && !ISSLASH (dest[1])) dead code: ((*dest) == '/') dead code: !((dest[1]) == '/') dead code: dest++; | - | ||||||||||||||||||
338 | dest++; dead code: dest++; | - | ||||||||||||||||||
339 | *dest = '\0'; | - | ||||||||||||||||||
340 | if (rname_limit != dest + 1)
| 0-205 | ||||||||||||||||||
341 | rname = xrealloc (rname, dest - rname + 1); executed 205 times by 3 tests: rname = xrealloc (rname, dest - rname + 1); Executed by:
| 205 | ||||||||||||||||||
342 | - | |||||||||||||||||||
343 | free (extra_buf); | - | ||||||||||||||||||
344 | if (ht)
| 84-121 | ||||||||||||||||||
345 | hash_free (ht); executed 84 times by 3 tests: hash_free (ht); Executed by:
| 84 | ||||||||||||||||||
346 | return rname; executed 205 times by 3 tests: return rname; Executed by:
| 205 | ||||||||||||||||||
347 | - | |||||||||||||||||||
348 | error: | - | ||||||||||||||||||
349 | free (extra_buf); | - | ||||||||||||||||||
350 | free (rname); | - | ||||||||||||||||||
351 | if (ht)
| 16-46 | ||||||||||||||||||
352 | hash_free (ht); executed 46 times by 1 test: hash_free (ht); Executed by:
| 46 | ||||||||||||||||||
353 | errno = saved_errno; | - | ||||||||||||||||||
354 | return NULL; executed 62 times by 1 test: return ((void *)0) ; Executed by:
| 62 | ||||||||||||||||||
355 | } | - | ||||||||||||||||||
Source code | Switch to Preprocessed file |