Absolute File Name: | /home/opencoverage/opencoverage/guest-scripts/coreutils/src/src/rmdir.c |
Source code | Switch to Preprocessed file |
Line | Source | Count | ||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
1 | /* rmdir -- remove directories | - | ||||||||||||
2 | - | |||||||||||||
3 | Copyright (C) 1990-2018 Free Software Foundation, Inc. | - | ||||||||||||
4 | - | |||||||||||||
5 | This program is free software: you can redistribute it and/or modify | - | ||||||||||||
6 | it under the terms of the GNU General Public License as published by | - | ||||||||||||
7 | the Free Software Foundation, either version 3 of the License, or | - | ||||||||||||
8 | (at your option) any later version. | - | ||||||||||||
9 | - | |||||||||||||
10 | This program is distributed in the hope that it will be useful, | - | ||||||||||||
11 | but WITHOUT ANY WARRANTY; without even the implied warranty of | - | ||||||||||||
12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | - | ||||||||||||
13 | GNU General Public License for more details. | - | ||||||||||||
14 | - | |||||||||||||
15 | You should have received a copy of the GNU General Public License | - | ||||||||||||
16 | along with this program. If not, see <https://www.gnu.org/licenses/>. */ | - | ||||||||||||
17 | - | |||||||||||||
18 | /* Options: | - | ||||||||||||
19 | -p, --parent Remove any parent dirs that are explicitly mentioned | - | ||||||||||||
20 | in an argument, if they become empty after the | - | ||||||||||||
21 | argument file is removed. | - | ||||||||||||
22 | - | |||||||||||||
23 | David MacKenzie <djm@ai.mit.edu> */ | - | ||||||||||||
24 | - | |||||||||||||
25 | #include <config.h> | - | ||||||||||||
26 | #include <stdio.h> | - | ||||||||||||
27 | #include <getopt.h> | - | ||||||||||||
28 | #include <sys/types.h> | - | ||||||||||||
29 | - | |||||||||||||
30 | #include "system.h" | - | ||||||||||||
31 | #include "error.h" | - | ||||||||||||
32 | #include "prog-fprintf.h" | - | ||||||||||||
33 | - | |||||||||||||
34 | /* The official name of this program (e.g., no 'g' prefix). */ | - | ||||||||||||
35 | #define PROGRAM_NAME "rmdir" | - | ||||||||||||
36 | - | |||||||||||||
37 | #define AUTHORS proper_name ("David MacKenzie") | - | ||||||||||||
38 | - | |||||||||||||
39 | /* If true, remove empty parent directories. */ | - | ||||||||||||
40 | static bool remove_empty_parents; | - | ||||||||||||
41 | - | |||||||||||||
42 | /* If true, don't treat failure to remove a nonempty directory | - | ||||||||||||
43 | as an error. */ | - | ||||||||||||
44 | static bool ignore_fail_on_non_empty; | - | ||||||||||||
45 | - | |||||||||||||
46 | /* If true, output a diagnostic for every directory processed. */ | - | ||||||||||||
47 | static bool verbose; | - | ||||||||||||
48 | - | |||||||||||||
49 | /* For long options that have no equivalent short option, use a | - | ||||||||||||
50 | non-character as a pseudo short option, starting with CHAR_MAX + 1. */ | - | ||||||||||||
51 | enum | - | ||||||||||||
52 | { | - | ||||||||||||
53 | IGNORE_FAIL_ON_NON_EMPTY_OPTION = CHAR_MAX + 1 | - | ||||||||||||
54 | }; | - | ||||||||||||
55 | - | |||||||||||||
56 | static struct option const longopts[] = | - | ||||||||||||
57 | { | - | ||||||||||||
58 | /* Don't name this '--force' because it's not close enough in meaning | - | ||||||||||||
59 | to e.g. rm's -f option. */ | - | ||||||||||||
60 | {"ignore-fail-on-non-empty", no_argument, NULL, | - | ||||||||||||
61 | IGNORE_FAIL_ON_NON_EMPTY_OPTION}, | - | ||||||||||||
62 | - | |||||||||||||
63 | {"path", no_argument, NULL, 'p'}, /* Deprecated. */ | - | ||||||||||||
64 | {"parents", no_argument, NULL, 'p'}, | - | ||||||||||||
65 | {"verbose", no_argument, NULL, 'v'}, | - | ||||||||||||
66 | {GETOPT_HELP_OPTION_DECL}, | - | ||||||||||||
67 | {GETOPT_VERSION_OPTION_DECL}, | - | ||||||||||||
68 | {NULL, 0, NULL, 0} | - | ||||||||||||
69 | }; | - | ||||||||||||
70 | - | |||||||||||||
71 | /* Return true if ERROR_NUMBER is one of the values associated | - | ||||||||||||
72 | with a failed rmdir due to non-empty target directory. */ | - | ||||||||||||
73 | static bool | - | ||||||||||||
74 | errno_rmdir_non_empty (int error_number) | - | ||||||||||||
75 | { | - | ||||||||||||
76 | return error_number == ENOTEMPTY || error_number == EEXIST; executed 1 time by 1 test: return error_number == 39 || error_number == 17 ; Executed by:
| 1 | ||||||||||||
77 | } | - | ||||||||||||
78 | - | |||||||||||||
79 | /* Return true if when rmdir fails with errno == ERROR_NUMBER | - | ||||||||||||
80 | the directory may be empty. */ | - | ||||||||||||
81 | static bool | - | ||||||||||||
82 | errno_may_be_empty (int error_number) | - | ||||||||||||
83 | { | - | ||||||||||||
84 | switch (error_number) | - | ||||||||||||
85 | { | - | ||||||||||||
86 | case EACCES: never executed: case 13 : | 0 | ||||||||||||
87 | case EPERM: never executed: case 1 : | 0 | ||||||||||||
88 | case EROFS: never executed: case 30 : | 0 | ||||||||||||
89 | case EEXIST: never executed: case 17 : | 0 | ||||||||||||
90 | case EBUSY: never executed: case 16 : | 0 | ||||||||||||
91 | return true; never executed: return 1 ; | 0 | ||||||||||||
92 | default: never executed: default: | 0 | ||||||||||||
93 | return false; never executed: return 0 ; | 0 | ||||||||||||
94 | } | - | ||||||||||||
95 | } | - | ||||||||||||
96 | - | |||||||||||||
97 | /* Return true if an rmdir failure with errno == error_number | - | ||||||||||||
98 | for DIR is ignorable. */ | - | ||||||||||||
99 | static bool | - | ||||||||||||
100 | ignorable_failure (int error_number, char const *dir) | - | ||||||||||||
101 | { | - | ||||||||||||
102 | return (ignore_fail_on_non_empty executed 3 times by 1 test: return (ignore_fail_on_non_empty && (errno_rmdir_non_empty (error_number) || (errno_may_be_empty (error_number) && is_empty_dir ( -100 , dir)))); Executed by:
| 3 | ||||||||||||
103 | && (errno_rmdir_non_empty (error_number) executed 3 times by 1 test: return (ignore_fail_on_non_empty && (errno_rmdir_non_empty (error_number) || (errno_may_be_empty (error_number) && is_empty_dir ( -100 , dir)))); Executed by:
| 3 | ||||||||||||
104 | || (errno_may_be_empty (error_number) executed 3 times by 1 test: return (ignore_fail_on_non_empty && (errno_rmdir_non_empty (error_number) || (errno_may_be_empty (error_number) && is_empty_dir ( -100 , dir)))); Executed by:
| 3 | ||||||||||||
105 | && is_empty_dir (AT_FDCWD, dir)))); executed 3 times by 1 test: return (ignore_fail_on_non_empty && (errno_rmdir_non_empty (error_number) || (errno_may_be_empty (error_number) && is_empty_dir ( -100 , dir)))); Executed by:
| 3 | ||||||||||||
106 | } | - | ||||||||||||
107 | - | |||||||||||||
108 | /* Remove any empty parent directories of DIR. | - | ||||||||||||
109 | If DIR contains slash characters, at least one of them | - | ||||||||||||
110 | (beginning with the rightmost) is replaced with a NUL byte. | - | ||||||||||||
111 | Return true if successful. */ | - | ||||||||||||
112 | - | |||||||||||||
113 | static bool | - | ||||||||||||
114 | remove_parents (char *dir) | - | ||||||||||||
115 | { | - | ||||||||||||
116 | char *slash; | - | ||||||||||||
117 | bool ok = true; | - | ||||||||||||
118 | - | |||||||||||||
119 | strip_trailing_slashes (dir); | - | ||||||||||||
120 | while (1) | - | ||||||||||||
121 | { | - | ||||||||||||
122 | slash = strrchr (dir, '/'); | - | ||||||||||||
123 | if (slash == NULL)
| 1-3 | ||||||||||||
124 | break; executed 1 time by 1 test: break; Executed by:
| 1 | ||||||||||||
125 | /* Remove any characters after the slash, skipping any extra | - | ||||||||||||
126 | slashes in a row. */ | - | ||||||||||||
127 | while (slash > dir && *slash == '/')
| 0-6 | ||||||||||||
128 | --slash; executed 3 times by 1 test: --slash; Executed by:
| 3 | ||||||||||||
129 | slash[1] = 0; | - | ||||||||||||
130 | - | |||||||||||||
131 | /* Give a diagnostic for each attempted removal if --verbose. */ | - | ||||||||||||
132 | if (verbose)
| 0-3 | ||||||||||||
133 | prog_fprintf (stdout, _("removing directory, %s"), quoteaf (dir)); never executed: prog_fprintf ( stdout , dcgettext (((void *)0), "removing directory, %s" , 5) , quotearg_style (shell_escape_always_quoting_style, dir)); | 0 | ||||||||||||
134 | - | |||||||||||||
135 | ok = (rmdir (dir) == 0); | - | ||||||||||||
136 | - | |||||||||||||
137 | if (!ok)
| 1-2 | ||||||||||||
138 | { | - | ||||||||||||
139 | /* Stop quietly if --ignore-fail-on-non-empty. */ | - | ||||||||||||
140 | if (ignorable_failure (errno, dir))
| 1 | ||||||||||||
141 | { | - | ||||||||||||
142 | ok = true; | - | ||||||||||||
143 | } executed 1 time by 1 test: end of block Executed by:
| 1 | ||||||||||||
144 | else | - | ||||||||||||
145 | { | - | ||||||||||||
146 | /* Barring race conditions, DIR is expected to be a directory. */ | - | ||||||||||||
147 | error (0, errno, _("failed to remove directory %s"), | - | ||||||||||||
148 | quoteaf (dir)); | - | ||||||||||||
149 | } executed 1 time by 1 test: end of block Executed by:
| 1 | ||||||||||||
150 | break; executed 2 times by 1 test: break; Executed by:
| 2 | ||||||||||||
151 | } | - | ||||||||||||
152 | } executed 1 time by 1 test: end of block Executed by:
| 1 | ||||||||||||
153 | return ok; executed 3 times by 1 test: return ok; Executed by:
| 3 | ||||||||||||
154 | } | - | ||||||||||||
155 | - | |||||||||||||
156 | void | - | ||||||||||||
157 | usage (int status) | - | ||||||||||||
158 | { | - | ||||||||||||
159 | if (status != EXIT_SUCCESS)
| 3-8 | ||||||||||||
160 | emit_try_help (); executed 3 times by 1 test: end of block Executed by:
| 3 | ||||||||||||
161 | else | - | ||||||||||||
162 | { | - | ||||||||||||
163 | printf (_("Usage: %s [OPTION]... DIRECTORY...\n"), program_name); | - | ||||||||||||
164 | fputs (_("\ | - | ||||||||||||
165 | Remove the DIRECTORY(ies), if they are empty.\n\ | - | ||||||||||||
166 | \n\ | - | ||||||||||||
167 | --ignore-fail-on-non-empty\n\ | - | ||||||||||||
168 | ignore each failure that is solely because a directory\n\ | - | ||||||||||||
169 | is non-empty\n\ | - | ||||||||||||
170 | "), stdout); | - | ||||||||||||
171 | fputs (_("\ | - | ||||||||||||
172 | -p, --parents remove DIRECTORY and its ancestors; e.g., 'rmdir -p a/b/c' is\ | - | ||||||||||||
173 | \n\ | - | ||||||||||||
174 | similar to 'rmdir a/b/c a/b a'\n\ | - | ||||||||||||
175 | -v, --verbose output a diagnostic for every directory processed\n\ | - | ||||||||||||
176 | "), stdout); | - | ||||||||||||
177 | fputs (HELP_OPTION_DESCRIPTION, stdout); | - | ||||||||||||
178 | fputs (VERSION_OPTION_DESCRIPTION, stdout); | - | ||||||||||||
179 | emit_ancillary_info (PROGRAM_NAME); | - | ||||||||||||
180 | } executed 8 times by 1 test: end of block Executed by:
| 8 | ||||||||||||
181 | exit (status); executed 11 times by 1 test: exit (status); Executed by:
| 11 | ||||||||||||
182 | } | - | ||||||||||||
183 | - | |||||||||||||
184 | int | - | ||||||||||||
185 | main (int argc, char **argv) | - | ||||||||||||
186 | { | - | ||||||||||||
187 | bool ok = true; | - | ||||||||||||
188 | int optc; | - | ||||||||||||
189 | - | |||||||||||||
190 | initialize_main (&argc, &argv); | - | ||||||||||||
191 | set_program_name (argv[0]); | - | ||||||||||||
192 | setlocale (LC_ALL, ""); | - | ||||||||||||
193 | bindtextdomain (PACKAGE, LOCALEDIR); | - | ||||||||||||
194 | textdomain (PACKAGE); | - | ||||||||||||
195 | - | |||||||||||||
196 | atexit (close_stdout); | - | ||||||||||||
197 | - | |||||||||||||
198 | remove_empty_parents = false; | - | ||||||||||||
199 | - | |||||||||||||
200 | while ((optc = getopt_long (argc, argv, "pv", longopts, NULL)) != -1)
| 16-26 | ||||||||||||
201 | { | - | ||||||||||||
202 | switch (optc) | - | ||||||||||||
203 | { | - | ||||||||||||
204 | case 'p': executed 5 times by 1 test: case 'p': Executed by:
| 5 | ||||||||||||
205 | remove_empty_parents = true; | - | ||||||||||||
206 | break; executed 5 times by 1 test: break; Executed by:
| 5 | ||||||||||||
207 | case IGNORE_FAIL_ON_NON_EMPTY_OPTION: executed 2 times by 1 test: case IGNORE_FAIL_ON_NON_EMPTY_OPTION: Executed by:
| 2 | ||||||||||||
208 | ignore_fail_on_non_empty = true; | - | ||||||||||||
209 | break; executed 2 times by 1 test: break; Executed by:
| 2 | ||||||||||||
210 | case 'v': executed 2 times by 1 test: case 'v': Executed by:
| 2 | ||||||||||||
211 | verbose = true; | - | ||||||||||||
212 | break; executed 2 times by 1 test: break; Executed by:
| 2 | ||||||||||||
213 | case_GETOPT_HELP_CHAR; never executed: break; executed 8 times by 1 test: case GETOPT_HELP_CHAR: Executed by:
| 0-8 | ||||||||||||
214 | case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS); executed 6 times by 1 test: exit ( 0 ); Executed by:
never executed: break; executed 6 times by 1 test: case GETOPT_VERSION_CHAR: Executed by:
| 0-6 | ||||||||||||
215 | default: executed 3 times by 1 test: default: Executed by:
| 3 | ||||||||||||
216 | usage (EXIT_FAILURE); | - | ||||||||||||
217 | } never executed: end of block | 0 | ||||||||||||
218 | } | - | ||||||||||||
219 | - | |||||||||||||
220 | if (optind == argc)
| 0-16 | ||||||||||||
221 | { | - | ||||||||||||
222 | error (0, 0, _("missing operand")); | - | ||||||||||||
223 | usage (EXIT_FAILURE); | - | ||||||||||||
224 | } never executed: end of block | 0 | ||||||||||||
225 | - | |||||||||||||
226 | for (; optind < argc; ++optind)
| 16-17 | ||||||||||||
227 | { | - | ||||||||||||
228 | char *dir = argv[optind]; | - | ||||||||||||
229 | - | |||||||||||||
230 | /* Give a diagnostic for each attempted removal if --verbose. */ | - | ||||||||||||
231 | if (verbose)
| 0-17 | ||||||||||||
232 | prog_fprintf (stdout, _("removing directory, %s"), quoteaf (dir)); never executed: prog_fprintf ( stdout , dcgettext (((void *)0), "removing directory, %s" , 5) , quotearg_style (shell_escape_always_quoting_style, dir)); | 0 | ||||||||||||
233 | - | |||||||||||||
234 | if (rmdir (dir) != 0)
| 1-16 | ||||||||||||
235 | { | - | ||||||||||||
236 | if (ignorable_failure (errno, dir))
| 0-1 | ||||||||||||
237 | continue; never executed: continue; | 0 | ||||||||||||
238 | - | |||||||||||||
239 | /* Here, the diagnostic is less precise, since we have no idea | - | ||||||||||||
240 | whether DIR is a directory. */ | - | ||||||||||||
241 | error (0, errno, _("failed to remove %s"), quoteaf (dir)); | - | ||||||||||||
242 | ok = false; | - | ||||||||||||
243 | } executed 1 time by 1 test: end of block Executed by:
| 1 | ||||||||||||
244 | else if (remove_empty_parents)
| 3-13 | ||||||||||||
245 | { | - | ||||||||||||
246 | ok &= remove_parents (dir); | - | ||||||||||||
247 | } executed 3 times by 1 test: end of block Executed by:
| 3 | ||||||||||||
248 | } executed 17 times by 1 test: end of block Executed by:
| 17 | ||||||||||||
249 | - | |||||||||||||
250 | return ok ? EXIT_SUCCESS : EXIT_FAILURE; executed 16 times by 1 test: return ok ? 0 : 1 ; Executed by:
| 16 | ||||||||||||
251 | } | - | ||||||||||||
Source code | Switch to Preprocessed file |