Annotation of embedaddon/libiconv/srclib/dirname-lgpl.c, revision 1.1.1.1
1.1 misho 1: /* dirname.c -- return all but the last element in a file name
2:
3: Copyright (C) 1990, 1998, 2000-2001, 2003-2006, 2009-2019 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 3 of the License, or
9: (at your option) 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, see <https://www.gnu.org/licenses/>. */
18:
19: #include <config.h>
20:
21: #include "dirname.h"
22:
23: #include <stdlib.h>
24: #include <string.h>
25:
26: /* Return the length of the prefix of FILE that will be used by
27: dir_name. If FILE is in the working directory, this returns zero
28: even though 'dir_name (FILE)' will return ".". Works properly even
29: if there are trailing slashes (by effectively ignoring them). */
30:
31: size_t
32: dir_len (char const *file)
33: {
34: size_t prefix_length = FILE_SYSTEM_PREFIX_LEN (file);
35: size_t length;
36:
37: /* Advance prefix_length beyond important leading slashes. */
38: prefix_length += (prefix_length != 0
39: ? (FILE_SYSTEM_DRIVE_PREFIX_CAN_BE_RELATIVE
40: && ISSLASH (file[prefix_length]))
41: : (ISSLASH (file[0])
42: ? ((DOUBLE_SLASH_IS_DISTINCT_ROOT
43: && ISSLASH (file[1]) && ! ISSLASH (file[2])
44: ? 2 : 1))
45: : 0));
46:
47: /* Strip the basename and any redundant slashes before it. */
48: for (length = last_component (file) - file;
49: prefix_length < length; length--)
50: if (! ISSLASH (file[length - 1]))
51: break;
52: return length;
53: }
54:
55:
56: /* In general, we can't use the builtin 'dirname' function if available,
57: since it has different meanings in different environments.
58: In some environments the builtin 'dirname' modifies its argument.
59:
60: Return the leading directories part of FILE, allocated with malloc.
61: Works properly even if there are trailing slashes (by effectively
62: ignoring them). Return NULL on failure.
63:
64: If lstat (FILE) would succeed, then { chdir (dir_name (FILE));
65: lstat (base_name (FILE)); } will access the same file. Likewise,
66: if the sequence { chdir (dir_name (FILE));
67: rename (base_name (FILE), "foo"); } succeeds, you have renamed FILE
68: to "foo" in the same directory FILE was in. */
69:
70: char *
71: mdir_name (char const *file)
72: {
73: size_t length = dir_len (file);
74: bool append_dot = (length == 0
75: || (FILE_SYSTEM_DRIVE_PREFIX_CAN_BE_RELATIVE
76: && length == FILE_SYSTEM_PREFIX_LEN (file)
77: && file[2] != '\0' && ! ISSLASH (file[2])));
78: char *dir = malloc (length + append_dot + 1);
79: if (!dir)
80: return NULL;
81: memcpy (dir, file, length);
82: if (append_dot)
83: dir[length++] = '.';
84: dir[length] = '\0';
85: return dir;
86: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>