4 * Copyright IBM, Corp. 2017
7 * Greg Kurz <groug@kaod.org>
9 * This work is licensed under the terms of the GNU GPL, version 2 or later.
10 * See the COPYING file in the top-level directory.
13 #ifndef QEMU_9P_UTIL_H
14 #define QEMU_9P_UTIL_H
17 #define O_PATH_9P_UTIL O_PATH
19 #define O_PATH_9P_UTIL 0
22 static inline void close_preserve_errno(int fd
)
29 static inline int openat_dir(int dirfd
, const char *name
)
31 return openat(dirfd
, name
,
32 O_DIRECTORY
| O_RDONLY
| O_NOFOLLOW
| O_PATH_9P_UTIL
);
35 static inline int openat_file(int dirfd
, const char *name
, int flags
,
41 fd
= openat(dirfd
, name
, flags
| O_NOFOLLOW
| O_NOCTTY
| O_NONBLOCK
,
44 if (errno
== EPERM
&& (flags
& O_NOATIME
)) {
46 * The client passed O_NOATIME but we lack permissions to honor it.
47 * Rather than failing the open, fall back without O_NOATIME. This
48 * doesn't break the semantics on the client side, as the Linux
49 * open(2) man page notes that O_NOATIME "may not be effective on
50 * all filesystems". In particular, NFS and other network
51 * filesystems ignore it entirely.
60 /* O_NONBLOCK was only needed to open the file. Let's drop it. We don't
61 * do that with O_PATH since fcntl(F_SETFL) isn't supported, and openat()
64 if (!(flags
& O_PATH_9P_UTIL
)) {
65 ret
= fcntl(fd
, F_SETFL
, flags
);
72 ssize_t
fgetxattrat_nofollow(int dirfd
, const char *path
, const char *name
,
73 void *value
, size_t size
);
74 int fsetxattrat_nofollow(int dirfd
, const char *path
, const char *name
,
75 void *value
, size_t size
, int flags
);
76 ssize_t
flistxattrat_nofollow(int dirfd
, const char *filename
,
77 char *list
, size_t size
);
78 ssize_t
fremovexattrat_nofollow(int dirfd
, const char *filename
,