select: Fix "warning: no previous prototype for function".
[gnulib.git] / lib / savewd.c
blobecc70fbcc19a164ded447b8baddd503c6f3bf834
1 /* Save and restore the working directory, possibly using a child process.
3 Copyright (C) 2006-2007, 2009-2020 Free Software Foundation, Inc.
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.
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.
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/>. */
18 /* Written by Paul Eggert. */
20 #include <config.h>
22 #define SAVEWD_INLINE _GL_EXTERN_INLINE
24 #include "savewd.h"
26 #include <errno.h>
27 #include <fcntl.h>
28 #include <signal.h>
29 #include <stdbool.h>
30 #include <stdlib.h>
31 #include <sys/types.h>
32 #include <sys/wait.h>
33 #include <unistd.h>
35 #include "assure.h"
36 #include "attribute.h"
37 #include "fcntl-safer.h"
38 #include "filename.h"
40 /* Save the working directory into *WD, if it hasn't been saved
41 already. Return true if a child has been forked to do the real
42 work. */
43 static bool
44 savewd_save (struct savewd *wd)
46 switch (wd->state)
48 case INITIAL_STATE:
49 /* Save the working directory, or prepare to fall back if possible. */
51 int fd = open_safer (".", O_SEARCH);
52 if (0 <= fd)
54 wd->state = FD_STATE;
55 wd->val.fd = fd;
56 break;
58 if (errno != EACCES && errno != ESTALE)
60 wd->state = ERROR_STATE;
61 wd->val.errnum = errno;
62 break;
65 wd->state = FORKING_STATE;
66 wd->val.child = -1;
67 FALLTHROUGH;
68 case FORKING_STATE:
69 if (wd->val.child < 0)
71 /* "Save" the initial working directory by forking a new
72 subprocess that will attempt all the work from the chdir
73 until the next savewd_restore. */
74 wd->val.child = fork ();
75 if (wd->val.child != 0)
77 if (0 < wd->val.child)
78 return true;
79 wd->state = ERROR_STATE;
80 wd->val.errnum = errno;
83 break;
85 case FD_STATE:
86 case FD_POST_CHDIR_STATE:
87 case ERROR_STATE:
88 case FINAL_STATE:
89 break;
91 default:
92 assure (false);
95 return false;
98 int
99 savewd_chdir (struct savewd *wd, char const *dir, int options,
100 int open_result[2])
102 int fd = -1;
103 int result = 0;
105 /* Open the directory if requested, or if avoiding a race condition
106 is requested and possible. */
107 if (open_result
108 || (options & (HAVE_WORKING_O_NOFOLLOW ? SAVEWD_CHDIR_NOFOLLOW : 0)))
110 fd = open (dir,
111 (O_SEARCH | O_DIRECTORY | O_NOCTTY | O_NONBLOCK
112 | (options & SAVEWD_CHDIR_NOFOLLOW ? O_NOFOLLOW : 0)));
114 if (open_result)
116 open_result[0] = fd;
117 open_result[1] = errno;
120 if (fd < 0 && errno != EACCES)
121 result = -1;
124 if (result == 0 && ! (0 <= fd && options & SAVEWD_CHDIR_SKIP_READABLE))
126 if (savewd_save (wd))
128 open_result = NULL;
129 result = -2;
131 else
133 result = (fd < 0 ? chdir (dir) : fchdir (fd));
135 if (result == 0)
136 switch (wd->state)
138 case FD_STATE:
139 wd->state = FD_POST_CHDIR_STATE;
140 break;
142 case ERROR_STATE:
143 case FD_POST_CHDIR_STATE:
144 case FINAL_STATE:
145 break;
147 case FORKING_STATE:
148 assure (wd->val.child == 0);
149 break;
151 default:
152 assure (false);
157 if (0 <= fd && ! open_result)
159 int e = errno;
160 close (fd);
161 errno = e;
164 return result;
168 savewd_restore (struct savewd *wd, int status)
170 switch (wd->state)
172 case INITIAL_STATE:
173 case FD_STATE:
174 /* The working directory is the desired directory, so there's no
175 work to do. */
176 break;
178 case FD_POST_CHDIR_STATE:
179 /* Restore the working directory using fchdir. */
180 if (fchdir (wd->val.fd) == 0)
182 wd->state = FD_STATE;
183 break;
185 else
187 int chdir_errno = errno;
188 close (wd->val.fd);
189 wd->state = ERROR_STATE;
190 wd->val.errnum = chdir_errno;
192 FALLTHROUGH;
193 case ERROR_STATE:
194 /* Report an error if asked to restore the working directory. */
195 errno = wd->val.errnum;
196 return -1;
198 case FORKING_STATE:
199 /* "Restore" the working directory by waiting for the subprocess
200 to finish. */
202 pid_t child = wd->val.child;
203 if (child == 0)
204 _exit (status);
205 if (0 < child)
207 int child_status;
208 while (waitpid (child, &child_status, 0) < 0)
209 assure (errno == EINTR);
210 wd->val.child = -1;
211 if (! WIFEXITED (child_status))
212 raise (WTERMSIG (child_status));
213 return WEXITSTATUS (child_status);
216 break;
218 default:
219 assure (false);
222 return 0;
225 void
226 savewd_finish (struct savewd *wd)
228 switch (wd->state)
230 case INITIAL_STATE:
231 case ERROR_STATE:
232 break;
234 case FD_STATE:
235 case FD_POST_CHDIR_STATE:
236 close (wd->val.fd);
237 break;
239 case FORKING_STATE:
240 assure (wd->val.child < 0);
241 break;
243 default:
244 assure (false);
247 wd->state = FINAL_STATE;
250 /* Return true if the actual work is currently being done by a
251 subprocess.
253 A true return means that the caller and the subprocess should
254 resynchronize later with savewd_restore, using only their own
255 memory to decide when to resynchronize; they should not consult the
256 file system to decide, because that might lead to race conditions.
257 This is why savewd_chdir is broken out into another function;
258 savewd_chdir's callers _can_ inspect the file system to decide
259 whether to call savewd_chdir. */
260 static bool
261 savewd_delegating (struct savewd const *wd)
263 return wd->state == FORKING_STATE && 0 < wd->val.child;
267 savewd_process_files (int n_files, char **file,
268 int (*act) (char *, struct savewd *, void *),
269 void *options)
271 int i = 0;
272 int last_relative;
273 int exit_status = EXIT_SUCCESS;
274 struct savewd wd;
275 savewd_init (&wd);
277 for (last_relative = n_files - 1; 0 <= last_relative; last_relative--)
278 if (! IS_ABSOLUTE_FILE_NAME (file[last_relative]))
279 break;
281 for (; i < last_relative; i++)
283 if (! savewd_delegating (&wd))
285 int s = act (file[i], &wd, options);
286 if (exit_status < s)
287 exit_status = s;
290 if (! IS_ABSOLUTE_FILE_NAME (file[i + 1]))
292 int r = savewd_restore (&wd, exit_status);
293 if (exit_status < r)
294 exit_status = r;
298 savewd_finish (&wd);
300 for (; i < n_files; i++)
302 int s = act (file[i], &wd, options);
303 if (exit_status < s)
304 exit_status = s;
307 return exit_status;