diff options
author | Tavian Barnes <tavianator@tavianator.com> | 2017-08-12 18:12:13 -0400 |
---|---|---|
committer | Tavian Barnes <tavianator@tavianator.com> | 2017-08-12 18:12:13 -0400 |
commit | acd7f7ed437793e7c67ecd869cfac32a87c1ec52 (patch) | |
tree | abfb1561c90aca00362fb0f99b8a8da4a70fe08a /util.c | |
parent | 01a754bc5572103f9a49242d756dc04b0e86bb6e (diff) | |
download | bfs-acd7f7ed437793e7c67ecd869cfac32a87c1ec52.tar.xz |
Unify broken symlink handling
Rather than open-code the fallback logic for broken symlinks everywhere
it's needed, introduce a new xfstatat() utility function that performs
the fallback automatically.
Using xfstatat() consistently fixes a few bugs, including cases where
broken symlinks are given as arguments to predicates like -samefile.
Diffstat (limited to 'util.c')
-rw-r--r-- | util.c | 11 |
1 files changed, 11 insertions, 0 deletions
@@ -226,6 +226,17 @@ const char *xbasename(const char *path) { return i; } +int xfstatat(int fd, const char *path, struct stat *buf, int *flags) { + int ret = fstatat(fd, path, buf, *flags); + + if (ret != 0 && !(*flags & AT_SYMLINK_NOFOLLOW) && (errno == ENOENT || errno == ENOTDIR)) { + *flags |= AT_SYMLINK_NOFOLLOW; + ret = fstatat(fd, path, buf, *flags); + } + + return ret; +} + enum bftw_typeflag mode_to_typeflag(mode_t mode) { switch (mode & S_IFMT) { #ifdef S_IFBLK |