From ba24ef86e7e3a8b9b3b03108c097142e503e1fd6 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Thu, 7 Oct 2021 22:23:35 +0200 Subject: [PATCH 1/3] dirent-util: get rid of stat_mode_to_dirent_type() Apparently glibc already has a helper for this. (Not in the man pages for Linux, but FreeBSD does document these cryptic helpers, and its exported by glibc. That should be good enough for us.) --- src/basic/dirent-util.c | 14 +------------- src/basic/dirent-util.h | 2 -- src/basic/recurse-dir.c | 2 +- 3 files changed, 2 insertions(+), 16 deletions(-) diff --git a/src/basic/dirent-util.c b/src/basic/dirent-util.c index a70871b33d..366cc077f3 100644 --- a/src/basic/dirent-util.c +++ b/src/basic/dirent-util.c @@ -7,18 +7,6 @@ #include "path-util.h" #include "string-util.h" -int stat_mode_to_dirent_type(mode_t mode) { - return - S_ISREG(mode) ? DT_REG : - S_ISDIR(mode) ? DT_DIR : - S_ISLNK(mode) ? DT_LNK : - S_ISFIFO(mode) ? DT_FIFO : - S_ISSOCK(mode) ? DT_SOCK : - S_ISCHR(mode) ? DT_CHR : - S_ISBLK(mode) ? DT_BLK : - DT_UNKNOWN; -} - static int dirent_ensure_type(DIR *d, struct dirent *de) { struct stat st; @@ -36,7 +24,7 @@ static int dirent_ensure_type(DIR *d, struct dirent *de) { if (fstatat(dirfd(d), de->d_name, &st, AT_SYMLINK_NOFOLLOW) < 0) return -errno; - de->d_type = stat_mode_to_dirent_type(st.st_mode); + de->d_type = IFTODT(st.st_mode); return 0; } diff --git a/src/basic/dirent-util.h b/src/basic/dirent-util.h index 5bbd54df5a..c7956e7c1b 100644 --- a/src/basic/dirent-util.h +++ b/src/basic/dirent-util.h @@ -8,8 +8,6 @@ #include "macro.h" #include "path-util.h" -int stat_mode_to_dirent_type(mode_t mode); - bool dirent_is_file(const struct dirent *de) _pure_; bool dirent_is_file_with_suffix(const struct dirent *de, const char *suffix) _pure_; diff --git a/src/basic/recurse-dir.c b/src/basic/recurse-dir.c index fe34ffbfd4..2aabbccadb 100644 --- a/src/basic/recurse-dir.c +++ b/src/basic/recurse-dir.c @@ -290,7 +290,7 @@ int recurse_dir( /* Copy over the data we acquired through statx() if we acquired any */ if (sx.stx_mask & STATX_TYPE) { assert(!!subdir == !!S_ISDIR(sx.stx_mode)); - de[i]->d_type = stat_mode_to_dirent_type(sx.stx_mode); + de[i]->d_type = IFTODT(sx.stx_mode); } if (sx.stx_mask & STATX_INO) From 3214129369a0daa6558565a93aa17eb8ada69bc3 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Thu, 7 Oct 2021 22:55:20 +0200 Subject: [PATCH 2/3] dirent-util: use statx() in readdir_ensure_type() Let's ask exactly for the one field we actually want to know, i.e. STATX_TYPE. (While we are at it, also copy over the inode number, if we have it, simply to report the most recent info we have) (Also, see AT_NO_AUTOMOUNT, so that we don't trigger automounts here. After all, if we want to know the inode type of a dirent here, then there's not need to trigger the automount, the inode type is not going to change by that.) --- src/basic/dirent-util.c | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/src/basic/dirent-util.c b/src/basic/dirent-util.c index 366cc077f3..ee2aff0f4d 100644 --- a/src/basic/dirent-util.c +++ b/src/basic/dirent-util.c @@ -5,10 +5,12 @@ #include "dirent-util.h" #include "path-util.h" +#include "stat-util.h" #include "string-util.h" static int dirent_ensure_type(DIR *d, struct dirent *de) { - struct stat st; + STRUCT_STATX_DEFINE(sx); + int r; assert(d); assert(de); @@ -21,10 +23,17 @@ static int dirent_ensure_type(DIR *d, struct dirent *de) { return 0; } - if (fstatat(dirfd(d), de->d_name, &st, AT_SYMLINK_NOFOLLOW) < 0) - return -errno; + /* Let's ask only for the type, nothing else. */ + r = statx_fallback(dirfd(d), de->d_name, AT_SYMLINK_NOFOLLOW|AT_NO_AUTOMOUNT, STATX_TYPE, &sx); + if (r < 0) + return r; - de->d_type = IFTODT(st.st_mode); + assert(FLAGS_SET(sx.stx_mask, STATX_TYPE)); + de->d_type = IFTODT(sx.stx_mode); + + /* If the inode is passed too, update the field, i.e. report most recent data */ + if (FLAGS_SET(sx.stx_mask, STATX_INO)) + de->d_ino = sx.stx_ino; return 0; } From 49a0931f625127a7cdbb02b6a9119119a2f7e1a7 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Thu, 7 Oct 2021 22:58:43 +0200 Subject: [PATCH 3/3] dirent-util: tweak readdir_ensure_type() a bit MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit So far we ignored if readdir_ensure_type() failed, the .d_type would then still possibly report DT_UNKNOWN, possibly confusing the caller. Let's make this safer: if we get an error on readdir_ensure_type() then report it — except if it is ENOENT which indicates the dirent vanished by now, which is not a problem and we should just skip to the next entry. --- src/basic/dirent-util.c | 44 ++++++++++++++++++++++++++++------------- 1 file changed, 30 insertions(+), 14 deletions(-) diff --git a/src/basic/dirent-util.c b/src/basic/dirent-util.c index ee2aff0f4d..d578e6836d 100644 --- a/src/basic/dirent-util.c +++ b/src/basic/dirent-util.c @@ -66,24 +66,40 @@ bool dirent_is_file_with_suffix(const struct dirent *de, const char *suffix) { } struct dirent *readdir_ensure_type(DIR *d) { - struct dirent *de; + int r; assert(d); - errno = 0; - de = readdir(d); - if (de) - (void) dirent_ensure_type(d, de); - return de; -} - -struct dirent *readdir_no_dot(DIR *dirp) { - struct dirent *d; + /* Like readdir(), but fills in .d_type if it is DT_UNKNOWN */ for (;;) { - d = readdir_ensure_type(dirp); - if (d && dot_or_dot_dot(d->d_name)) - continue; - return d; + struct dirent *de; + + errno = 0; + de = readdir(d); + if (!de) + return NULL; + + r = dirent_ensure_type(d, de); + if (r >= 0) + return de; + if (r != -ENOENT) { + errno = -r; /* We want to be compatible with readdir(), hence propagate error via errno here */ + return NULL; + } + + /* Vanished by now? Then skip immedately to next */ + } +} + +struct dirent *readdir_no_dot(DIR *d) { + assert(d); + + for (;;) { + struct dirent *de; + + de = readdir_ensure_type(d); + if (!de || !dot_or_dot_dot(de->d_name)) + return de; } }