提交 94938920 编写于 作者: R Rich Felker

refactor all stat functions in terms of fstatat

equivalent logic for fstat+O_PATH fallback and direct use of
stat/lstat syscalls where appropriate is kept, now in the fstatat
function. this change both improves functionality (now, fstatat forms
equivalent to fstat/lstat/stat will work even on kernels too old to
have the at functions) and localizes direct interfacing with the
kernel stat structure to one file.
上级 62a73d96
#define _BSD_SOURCE
#include <sys/stat.h> #include <sys/stat.h>
#include <errno.h> #include <errno.h>
#include <fcntl.h> #include <fcntl.h>
...@@ -5,17 +6,7 @@ ...@@ -5,17 +6,7 @@
int fstat(int fd, struct stat *st) int fstat(int fd, struct stat *st)
{ {
int ret = __syscall(SYS_fstat, fd, st); return fstatat(fd, "", st, AT_EMPTY_PATH);
if (ret != -EBADF || __syscall(SYS_fcntl, fd, F_GETFD) < 0)
return __syscall_ret(ret);
char buf[15+3*sizeof(int)];
__procfdname(buf, fd);
#ifdef SYS_stat
return syscall(SYS_stat, buf, st);
#else
return syscall(SYS_fstatat, AT_FDCWD, buf, st, 0);
#endif
} }
weak_alias(fstat, fstat64); weak_alias(fstat, fstat64);
#define _BSD_SOURCE
#include <sys/stat.h> #include <sys/stat.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
#include "syscall.h" #include "syscall.h"
int fstatat(int fd, const char *restrict path, struct stat *restrict buf, int flag) int fstatat(int fd, const char *restrict path, struct stat *restrict st, int flag)
{ {
return syscall(SYS_fstatat, fd, path, buf, flag); int ret;
if (flag==AT_EMPTY_PATH && fd>=0 && !*path) {
ret = __syscall(SYS_fstat, fd, st);
if (ret==-EBADF && __syscall(SYS_fcntl, fd, F_GETFD)>=0) {
ret = __syscall(SYS_fstatat, fd, path, st, flag);
if (ret==-EINVAL) {
char buf[15+3*sizeof(int)];
__procfdname(buf, fd);
#ifdef SYS_stat
ret = __syscall(SYS_stat, buf, st);
#else
ret = __syscall(SYS_fstatat, AT_FDCWD, buf, st, 0);
#endif
}
}
}
#ifdef SYS_lstat
else if ((fd == AT_FDCWD || *path=='/') && flag==AT_SYMLINK_NOFOLLOW)
ret = __syscall(SYS_lstat, path, st);
#endif
#ifdef SYS_stat
else if ((fd == AT_FDCWD || *path=='/') && !flag)
ret = __syscall(SYS_stat, path, st);
#endif
else ret = __syscall(SYS_fstatat, fd, path, st, flag);
return __syscall_ret(ret);
} }
weak_alias(fstatat, fstatat64); weak_alias(fstatat, fstatat64);
...@@ -4,11 +4,7 @@ ...@@ -4,11 +4,7 @@
int lstat(const char *restrict path, struct stat *restrict buf) int lstat(const char *restrict path, struct stat *restrict buf)
{ {
#ifdef SYS_lstat return fstatat(AT_FDCWD, path, buf, AT_SYMLINK_NOFOLLOW);
return syscall(SYS_lstat, path, buf);
#else
return syscall(SYS_fstatat, AT_FDCWD, path, buf, AT_SYMLINK_NOFOLLOW);
#endif
} }
weak_alias(lstat, lstat64); weak_alias(lstat, lstat64);
...@@ -4,11 +4,7 @@ ...@@ -4,11 +4,7 @@
int stat(const char *restrict path, struct stat *restrict buf) int stat(const char *restrict path, struct stat *restrict buf)
{ {
#ifdef SYS_stat return fstatat(AT_FDCWD, path, buf, 0);
return syscall(SYS_stat, path, buf);
#else
return syscall(SYS_fstatat, AT_FDCWD, path, buf, 0);
#endif
} }
weak_alias(stat, stat64); weak_alias(stat, stat64);
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册