| 1 | /* Open a directory relative to another directory.
|
|---|
| 2 |
|
|---|
| 3 | Copyright 2006-2021 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 | Written by Jim Meyering and Paul Eggert. */
|
|---|
| 19 |
|
|---|
| 20 | #include <config.h>
|
|---|
| 21 |
|
|---|
| 22 | #include <opendirat.h>
|
|---|
| 23 |
|
|---|
| 24 | #include <errno.h>
|
|---|
| 25 | #include <fcntl--.h>
|
|---|
| 26 | #include <unistd.h>
|
|---|
| 27 |
|
|---|
| 28 | /* Relative to DIR_FD, open the directory DIR, passing EXTRA_FLAGS to
|
|---|
| 29 | the underlying openat call. On success, store into *PNEW_FD the
|
|---|
| 30 | underlying file descriptor of the newly opened directory and return
|
|---|
| 31 | the directory stream. On failure, return NULL and set errno.
|
|---|
| 32 |
|
|---|
| 33 | On success, *PNEW_FD is at least 3, so this is a "safer" function. */
|
|---|
| 34 |
|
|---|
| 35 | DIR *
|
|---|
| 36 | opendirat (int dir_fd, char const *dir, int extra_flags, int *pnew_fd)
|
|---|
| 37 | {
|
|---|
| 38 | int open_flags = (O_RDONLY | O_CLOEXEC | O_DIRECTORY | O_NOCTTY
|
|---|
| 39 | | O_NONBLOCK | extra_flags);
|
|---|
| 40 | int new_fd = openat (dir_fd, dir, open_flags);
|
|---|
| 41 |
|
|---|
| 42 | if (new_fd < 0)
|
|---|
| 43 | return NULL;
|
|---|
| 44 | DIR *dirp = fdopendir (new_fd);
|
|---|
| 45 | if (dirp)
|
|---|
| 46 | *pnew_fd = new_fd;
|
|---|
| 47 | else
|
|---|
| 48 | {
|
|---|
| 49 | int fdopendir_errno = errno;
|
|---|
| 50 | close (new_fd);
|
|---|
| 51 | errno = fdopendir_errno;
|
|---|
| 52 | }
|
|---|
| 53 | return dirp;
|
|---|
| 54 | }
|
|---|