mirror of
https://sourceware.org/git/glibc.git
synced 2024-12-16 16:10:06 +00:00
Linux: Emulate fchmodat with AT_SYMLINK_NOFOLLOW using O_PATH [BZ #14578]
/proc/self/fd files are special and chmod on O_PATH descriptors in that directory operates on the symbolic link itself (like lchmod).
This commit is contained in:
parent
6b89c385d8
commit
752dd17443
@ -18,24 +18,61 @@
|
|||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <stddef.h>
|
#include <not-cancel.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <sys/stat.h>
|
||||||
#include <unistd.h>
|
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <alloca.h>
|
|
||||||
#include <sysdep.h>
|
#include <sysdep.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
int
|
int
|
||||||
fchmodat (int fd, const char *file, mode_t mode, int flag)
|
fchmodat (int fd, const char *file, mode_t mode, int flag)
|
||||||
{
|
{
|
||||||
if (flag & ~AT_SYMLINK_NOFOLLOW)
|
if (flag == 0)
|
||||||
|
return INLINE_SYSCALL (fchmodat, 3, fd, file, mode);
|
||||||
|
else if (flag != AT_SYMLINK_NOFOLLOW)
|
||||||
return INLINE_SYSCALL_ERROR_RETURN_VALUE (EINVAL);
|
return INLINE_SYSCALL_ERROR_RETURN_VALUE (EINVAL);
|
||||||
#ifndef __NR_lchmod /* Linux so far has no lchmod syscall. */
|
else
|
||||||
if (flag & AT_SYMLINK_NOFOLLOW)
|
{
|
||||||
return INLINE_SYSCALL_ERROR_RETURN_VALUE (ENOTSUP);
|
/* The kernel system call does not have a mode argument.
|
||||||
#endif
|
However, we can create an O_PATH descriptor and change that
|
||||||
|
via /proc (which does not resolve symbolic links). */
|
||||||
|
|
||||||
return INLINE_SYSCALL (fchmodat, 3, fd, file, mode);
|
int pathfd = __openat_nocancel (fd, file,
|
||||||
|
O_PATH | O_NOFOLLOW | O_CLOEXEC);
|
||||||
|
if (pathfd < 0)
|
||||||
|
/* This may report errors such as ENFILE and EMFILE. The
|
||||||
|
caller can treat them as temporary if necessary. */
|
||||||
|
return pathfd;
|
||||||
|
|
||||||
|
char buf[32];
|
||||||
|
if (__snprintf (buf, sizeof (buf), "/proc/self/fd/%d", pathfd) < 0)
|
||||||
|
{
|
||||||
|
/* This also may report strange error codes to the caller
|
||||||
|
(although snprintf really should not fail). */
|
||||||
|
__close_nocancel (pathfd);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This operates directly on the symbolic link if it is one.
|
||||||
|
/proc/self/fd files look like symbolic links, but they are
|
||||||
|
not. (fchmod and fchmodat do not work on O_PATH descriptors,
|
||||||
|
similar to fstat before Linux 3.6.) */
|
||||||
|
int ret = __chmod (buf, mode);
|
||||||
|
if (ret != 0)
|
||||||
|
{
|
||||||
|
if (errno == ENOENT)
|
||||||
|
/* /proc has not been mounted. Without /proc, there is no
|
||||||
|
way to upgrade the O_PATH descriptor to a full
|
||||||
|
descriptor. It is also not possible to re-open the
|
||||||
|
file without O_PATH because the file name may refer to
|
||||||
|
another file, and opening that without O_PATH may have
|
||||||
|
side effects (such as blocking, device rewinding, or
|
||||||
|
releasing POSIX locks). */
|
||||||
|
__set_errno (EOPNOTSUPP);
|
||||||
|
}
|
||||||
|
__close_nocancel (pathfd);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
libc_hidden_def (fchmodat)
|
libc_hidden_def (fchmodat)
|
||||||
|
Loading…
Reference in New Issue
Block a user