Commit b6ff1070 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'fsnotify_for_v5.7-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/jack/linux-fs

Pull fsnotify updates from Jan Kara:
 "This implements the fanotify FAN_DIR_MODIFY event.

  This event reports the name in a directory under which a change
  happened and together with the directory filehandle and fstatat()
  allows reliable and efficient implementation of directory
  synchronization"

* tag 'fsnotify_for_v5.7-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/jack/linux-fs:
  fanotify: Fix the checks in fanotify_fsid_equal
  fanotify: report name info for FAN_DIR_MODIFY event
  fanotify: record name info for FAN_DIR_MODIFY event
  fanotify: Drop fanotify_event_has_fid()
  fanotify: prepare to report both parent and child fid's
  fanotify: send FAN_DIR_MODIFY event flavor with dir inode and name
  fanotify: divorce fanotify_path_event and fanotify_fid_event
  fanotify: Store fanotify handles differently
  fanotify: Simplify create_fd()
  fanotify: fix merging marks masks with FAN_ONDIR
  fanotify: merge duplicate events on parent and child
  fsnotify: replace inode pointer with an object id
  fsnotify: simplify arguments passing to fsnotify_parent()
  fsnotify: use helpers to access data by data_type
  fsnotify: funnel all dirent events through fsnotify_name()
  fsnotify: factor helpers fsnotify_dentry() and fsnotify_file()
  fsnotify: tidy up FS_ and FAN_ constants
parents 74e934ba 6def1a1d
This diff is collapsed.
...@@ -5,7 +5,8 @@ ...@@ -5,7 +5,8 @@
#include <linux/exportfs.h> #include <linux/exportfs.h>
extern struct kmem_cache *fanotify_mark_cache; extern struct kmem_cache *fanotify_mark_cache;
extern struct kmem_cache *fanotify_event_cachep; extern struct kmem_cache *fanotify_fid_event_cachep;
extern struct kmem_cache *fanotify_path_event_cachep;
extern struct kmem_cache *fanotify_perm_event_cachep; extern struct kmem_cache *fanotify_perm_event_cachep;
/* Possible states of the permission event */ /* Possible states of the permission event */
...@@ -18,94 +19,140 @@ enum { ...@@ -18,94 +19,140 @@ enum {
/* /*
* 3 dwords are sufficient for most local fs (64bit ino, 32bit generation). * 3 dwords are sufficient for most local fs (64bit ino, 32bit generation).
* For 32bit arch, fid increases the size of fanotify_event by 12 bytes and * fh buf should be dword aligned. On 64bit arch, the ext_buf pointer is
* fh_* fields increase the size of fanotify_event by another 4 bytes. * stored in either the first or last 2 dwords.
* For 64bit arch, fid increases the size of fanotify_fid by 8 bytes and
* fh_* fields are packed in a hole after mask.
*/ */
#if BITS_PER_LONG == 32
#define FANOTIFY_INLINE_FH_LEN (3 << 2) #define FANOTIFY_INLINE_FH_LEN (3 << 2)
#else
#define FANOTIFY_INLINE_FH_LEN (4 << 2)
#endif
struct fanotify_fid { struct fanotify_fh {
__kernel_fsid_t fsid; unsigned char buf[FANOTIFY_INLINE_FH_LEN];
union { u8 type;
unsigned char fh[FANOTIFY_INLINE_FH_LEN]; u8 len;
unsigned char *ext_fh; } __aligned(4);
};
}; static inline bool fanotify_fh_has_ext_buf(struct fanotify_fh *fh)
{
return fh->len > FANOTIFY_INLINE_FH_LEN;
}
static inline char **fanotify_fh_ext_buf_ptr(struct fanotify_fh *fh)
{
BUILD_BUG_ON(__alignof__(char *) - 4 + sizeof(char *) >
FANOTIFY_INLINE_FH_LEN);
return (char **)ALIGN((unsigned long)(fh->buf), __alignof__(char *));
}
static inline void *fanotify_fid_fh(struct fanotify_fid *fid, static inline void *fanotify_fh_ext_buf(struct fanotify_fh *fh)
unsigned int fh_len)
{ {
return fh_len <= FANOTIFY_INLINE_FH_LEN ? fid->fh : fid->ext_fh; return *fanotify_fh_ext_buf_ptr(fh);
} }
static inline bool fanotify_fid_equal(struct fanotify_fid *fid1, static inline void *fanotify_fh_buf(struct fanotify_fh *fh)
struct fanotify_fid *fid2,
unsigned int fh_len)
{ {
return fid1->fsid.val[0] == fid2->fsid.val[0] && return fanotify_fh_has_ext_buf(fh) ? fanotify_fh_ext_buf(fh) : fh->buf;
fid1->fsid.val[1] == fid2->fsid.val[1] &&
!memcmp(fanotify_fid_fh(fid1, fh_len),
fanotify_fid_fh(fid2, fh_len), fh_len);
} }
/* /*
* Structure for normal fanotify events. It gets allocated in * Common structure for fanotify events. Concrete structs are allocated in
* fanotify_handle_event() and freed when the information is retrieved by * fanotify_handle_event() and freed when the information is retrieved by
* userspace * userspace. The type of event determines how it was allocated, how it will
* be freed and which concrete struct it may be cast to.
*/ */
enum fanotify_event_type {
FANOTIFY_EVENT_TYPE_FID, /* fixed length */
FANOTIFY_EVENT_TYPE_FID_NAME, /* variable length */
FANOTIFY_EVENT_TYPE_PATH,
FANOTIFY_EVENT_TYPE_PATH_PERM,
};
struct fanotify_event { struct fanotify_event {
struct fsnotify_event fse; struct fsnotify_event fse;
u32 mask; u32 mask;
/* enum fanotify_event_type type;
* Those fields are outside fanotify_fid to pack fanotify_event nicely
* on 64bit arch and to use fh_type as an indication of whether path
* or fid are used in the union:
* FILEID_ROOT (0) for path, > 0 for fid, FILEID_INVALID for neither.
*/
u8 fh_type;
u8 fh_len;
u16 pad;
union {
/*
* We hold ref to this path so it may be dereferenced at any
* point during this object's lifetime
*/
struct path path;
/*
* With FAN_REPORT_FID, we do not hold any reference on the
* victim object. Instead we store its NFS file handle and its
* filesystem's fsid as a unique identifier.
*/
struct fanotify_fid fid;
};
struct pid *pid; struct pid *pid;
}; };
static inline bool fanotify_event_has_path(struct fanotify_event *event) struct fanotify_fid_event {
struct fanotify_event fae;
__kernel_fsid_t fsid;
struct fanotify_fh object_fh;
};
static inline struct fanotify_fid_event *
FANOTIFY_FE(struct fanotify_event *event)
{ {
return event->fh_type == FILEID_ROOT; return container_of(event, struct fanotify_fid_event, fae);
} }
static inline bool fanotify_event_has_fid(struct fanotify_event *event) struct fanotify_name_event {
struct fanotify_event fae;
__kernel_fsid_t fsid;
struct fanotify_fh dir_fh;
u8 name_len;
char name[0];
};
static inline struct fanotify_name_event *
FANOTIFY_NE(struct fanotify_event *event)
{ {
return event->fh_type != FILEID_ROOT && return container_of(event, struct fanotify_name_event, fae);
event->fh_type != FILEID_INVALID;
} }
static inline bool fanotify_event_has_ext_fh(struct fanotify_event *event) static inline __kernel_fsid_t *fanotify_event_fsid(struct fanotify_event *event)
{ {
return fanotify_event_has_fid(event) && if (event->type == FANOTIFY_EVENT_TYPE_FID)
event->fh_len > FANOTIFY_INLINE_FH_LEN; return &FANOTIFY_FE(event)->fsid;
else if (event->type == FANOTIFY_EVENT_TYPE_FID_NAME)
return &FANOTIFY_NE(event)->fsid;
else
return NULL;
} }
static inline void *fanotify_event_fh(struct fanotify_event *event) static inline struct fanotify_fh *fanotify_event_object_fh(
struct fanotify_event *event)
{ {
return fanotify_fid_fh(&event->fid, event->fh_len); if (event->type == FANOTIFY_EVENT_TYPE_FID)
return &FANOTIFY_FE(event)->object_fh;
else
return NULL;
}
static inline struct fanotify_fh *fanotify_event_dir_fh(
struct fanotify_event *event)
{
if (event->type == FANOTIFY_EVENT_TYPE_FID_NAME)
return &FANOTIFY_NE(event)->dir_fh;
else
return NULL;
}
static inline int fanotify_event_object_fh_len(struct fanotify_event *event)
{
struct fanotify_fh *fh = fanotify_event_object_fh(event);
return fh ? fh->len : 0;
}
static inline bool fanotify_event_has_name(struct fanotify_event *event)
{
return event->type == FANOTIFY_EVENT_TYPE_FID_NAME;
}
static inline int fanotify_event_name_len(struct fanotify_event *event)
{
return fanotify_event_has_name(event) ?
FANOTIFY_NE(event)->name_len : 0;
}
struct fanotify_path_event {
struct fanotify_event fae;
struct path path;
};
static inline struct fanotify_path_event *
FANOTIFY_PE(struct fanotify_event *event)
{
return container_of(event, struct fanotify_path_event, fae);
} }
/* /*
...@@ -117,15 +164,16 @@ static inline void *fanotify_event_fh(struct fanotify_event *event) ...@@ -117,15 +164,16 @@ static inline void *fanotify_event_fh(struct fanotify_event *event)
*/ */
struct fanotify_perm_event { struct fanotify_perm_event {
struct fanotify_event fae; struct fanotify_event fae;
struct path path;
unsigned short response; /* userspace answer to the event */ unsigned short response; /* userspace answer to the event */
unsigned short state; /* state of the event */ unsigned short state; /* state of the event */
int fd; /* fd we passed to userspace for this event */ int fd; /* fd we passed to userspace for this event */
}; };
static inline struct fanotify_perm_event * static inline struct fanotify_perm_event *
FANOTIFY_PE(struct fsnotify_event *fse) FANOTIFY_PERM(struct fanotify_event *event)
{ {
return container_of(fse, struct fanotify_perm_event, fae.fse); return container_of(event, struct fanotify_perm_event, fae);
} }
static inline bool fanotify_is_perm_event(u32 mask) static inline bool fanotify_is_perm_event(u32 mask)
...@@ -139,7 +187,24 @@ static inline struct fanotify_event *FANOTIFY_E(struct fsnotify_event *fse) ...@@ -139,7 +187,24 @@ static inline struct fanotify_event *FANOTIFY_E(struct fsnotify_event *fse)
return container_of(fse, struct fanotify_event, fse); return container_of(fse, struct fanotify_event, fse);
} }
static inline bool fanotify_event_has_path(struct fanotify_event *event)
{
return event->type == FANOTIFY_EVENT_TYPE_PATH ||
event->type == FANOTIFY_EVENT_TYPE_PATH_PERM;
}
static inline struct path *fanotify_event_path(struct fanotify_event *event)
{
if (event->type == FANOTIFY_EVENT_TYPE_PATH)
return &FANOTIFY_PE(event)->path;
else if (event->type == FANOTIFY_EVENT_TYPE_PATH_PERM)
return &FANOTIFY_PERM(event)->path;
else
return NULL;
}
struct fanotify_event *fanotify_alloc_event(struct fsnotify_group *group, struct fanotify_event *fanotify_alloc_event(struct fsnotify_group *group,
struct inode *inode, u32 mask, struct inode *inode, u32 mask,
const void *data, int data_type, const void *data, int data_type,
const struct qstr *file_name,
__kernel_fsid_t *fsid); __kernel_fsid_t *fsid);
This diff is collapsed.
...@@ -143,15 +143,13 @@ void __fsnotify_update_child_dentry_flags(struct inode *inode) ...@@ -143,15 +143,13 @@ void __fsnotify_update_child_dentry_flags(struct inode *inode)
} }
/* Notify this dentry's parent about a child's events. */ /* Notify this dentry's parent about a child's events. */
int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask) int fsnotify_parent(struct dentry *dentry, __u32 mask, const void *data,
int data_type)
{ {
struct dentry *parent; struct dentry *parent;
struct inode *p_inode; struct inode *p_inode;
int ret = 0; int ret = 0;
if (!dentry)
dentry = path->dentry;
if (!(dentry->d_flags & DCACHE_FSNOTIFY_PARENT_WATCHED)) if (!(dentry->d_flags & DCACHE_FSNOTIFY_PARENT_WATCHED))
return 0; return 0;
...@@ -168,12 +166,7 @@ int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask ...@@ -168,12 +166,7 @@ int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask
mask |= FS_EVENT_ON_CHILD; mask |= FS_EVENT_ON_CHILD;
take_dentry_name_snapshot(&name, dentry); take_dentry_name_snapshot(&name, dentry);
if (path) ret = fsnotify(p_inode, mask, data, data_type, &name.name, 0);
ret = fsnotify(p_inode, mask, path, FSNOTIFY_EVENT_PATH,
&name.name, 0);
else
ret = fsnotify(p_inode, mask, dentry->d_inode, FSNOTIFY_EVENT_INODE,
&name.name, 0);
release_dentry_name_snapshot(&name); release_dentry_name_snapshot(&name);
} }
...@@ -181,7 +174,7 @@ int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask ...@@ -181,7 +174,7 @@ int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask
return ret; return ret;
} }
EXPORT_SYMBOL_GPL(__fsnotify_parent); EXPORT_SYMBOL_GPL(fsnotify_parent);
static int send_to_group(struct inode *to_tell, static int send_to_group(struct inode *to_tell,
__u32 mask, const void *data, __u32 mask, const void *data,
...@@ -318,6 +311,7 @@ static void fsnotify_iter_next(struct fsnotify_iter_info *iter_info) ...@@ -318,6 +311,7 @@ static void fsnotify_iter_next(struct fsnotify_iter_info *iter_info)
int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is, int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is,
const struct qstr *file_name, u32 cookie) const struct qstr *file_name, u32 cookie)
{ {
const struct path *path = fsnotify_data_path(data, data_is);
struct fsnotify_iter_info iter_info = {}; struct fsnotify_iter_info iter_info = {};
struct super_block *sb = to_tell->i_sb; struct super_block *sb = to_tell->i_sb;
struct mount *mnt = NULL; struct mount *mnt = NULL;
...@@ -325,8 +319,8 @@ int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is, ...@@ -325,8 +319,8 @@ int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is,
int ret = 0; int ret = 0;
__u32 test_mask = (mask & ALL_FSNOTIFY_EVENTS); __u32 test_mask = (mask & ALL_FSNOTIFY_EVENTS);
if (data_is == FSNOTIFY_EVENT_PATH) { if (path) {
mnt = real_mount(((const struct path *)data)->mnt); mnt = real_mount(path->mnt);
mnt_or_sb_mask |= mnt->mnt_fsnotify_mask; mnt_or_sb_mask |= mnt->mnt_fsnotify_mask;
} }
/* An event "on child" is not intended for a mount/sb mark */ /* An event "on child" is not intended for a mount/sb mark */
...@@ -389,7 +383,7 @@ static __init int fsnotify_init(void) ...@@ -389,7 +383,7 @@ static __init int fsnotify_init(void)
{ {
int ret; int ret;
BUILD_BUG_ON(HWEIGHT32(ALL_FSNOTIFY_BITS) != 25); BUILD_BUG_ON(HWEIGHT32(ALL_FSNOTIFY_BITS) != 26);
ret = init_srcu_struct(&fsnotify_mark_srcu); ret = init_srcu_struct(&fsnotify_mark_srcu);
if (ret) if (ret)
......
...@@ -39,7 +39,7 @@ static bool event_compare(struct fsnotify_event *old_fsn, ...@@ -39,7 +39,7 @@ static bool event_compare(struct fsnotify_event *old_fsn,
if (old->mask & FS_IN_IGNORED) if (old->mask & FS_IN_IGNORED)
return false; return false;
if ((old->mask == new->mask) && if ((old->mask == new->mask) &&
(old_fsn->inode == new_fsn->inode) && (old_fsn->objectid == new_fsn->objectid) &&
(old->name_len == new->name_len) && (old->name_len == new->name_len) &&
(!old->name_len || !strcmp(old->name, new->name))) (!old->name_len || !strcmp(old->name, new->name)))
return true; return true;
...@@ -61,6 +61,7 @@ int inotify_handle_event(struct fsnotify_group *group, ...@@ -61,6 +61,7 @@ int inotify_handle_event(struct fsnotify_group *group,
const struct qstr *file_name, u32 cookie, const struct qstr *file_name, u32 cookie,
struct fsnotify_iter_info *iter_info) struct fsnotify_iter_info *iter_info)
{ {
const struct path *path = fsnotify_data_path(data, data_type);
struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info); struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info);
struct inotify_inode_mark *i_mark; struct inotify_inode_mark *i_mark;
struct inotify_event_info *event; struct inotify_event_info *event;
...@@ -73,12 +74,9 @@ int inotify_handle_event(struct fsnotify_group *group, ...@@ -73,12 +74,9 @@ int inotify_handle_event(struct fsnotify_group *group,
return 0; return 0;
if ((inode_mark->mask & FS_EXCL_UNLINK) && if ((inode_mark->mask & FS_EXCL_UNLINK) &&
(data_type == FSNOTIFY_EVENT_PATH)) { path && d_unlinked(path->dentry))
const struct path *path = data; return 0;
if (d_unlinked(path->dentry))
return 0;
}
if (file_name) { if (file_name) {
len = file_name->len; len = file_name->len;
alloc_len += len + 1; alloc_len += len + 1;
...@@ -118,7 +116,7 @@ int inotify_handle_event(struct fsnotify_group *group, ...@@ -118,7 +116,7 @@ int inotify_handle_event(struct fsnotify_group *group,
mask &= ~IN_ISDIR; mask &= ~IN_ISDIR;
fsn_event = &event->fse; fsn_event = &event->fse;
fsnotify_init_event(fsn_event, inode); fsnotify_init_event(fsn_event, (unsigned long)inode);
event->mask = mask; event->mask = mask;
event->wd = i_mark->wd; event->wd = i_mark->wd;
event->sync_cookie = cookie; event->sync_cookie = cookie;
......
...@@ -635,7 +635,7 @@ static struct fsnotify_group *inotify_new_group(unsigned int max_events) ...@@ -635,7 +635,7 @@ static struct fsnotify_group *inotify_new_group(unsigned int max_events)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
} }
group->overflow_event = &oevent->fse; group->overflow_event = &oevent->fse;
fsnotify_init_event(group->overflow_event, NULL); fsnotify_init_event(group->overflow_event, 0);
oevent->mask = FS_Q_OVERFLOW; oevent->mask = FS_Q_OVERFLOW;
oevent->wd = -1; oevent->wd = -1;
oevent->sync_cookie = 0; oevent->sync_cookie = 0;
......
...@@ -47,7 +47,8 @@ ...@@ -47,7 +47,8 @@
* Directory entry modification events - reported only to directory * Directory entry modification events - reported only to directory
* where entry is modified and not to a watching parent. * where entry is modified and not to a watching parent.
*/ */
#define FANOTIFY_DIRENT_EVENTS (FAN_MOVE | FAN_CREATE | FAN_DELETE) #define FANOTIFY_DIRENT_EVENTS (FAN_MOVE | FAN_CREATE | FAN_DELETE | \
FAN_DIR_MODIFY)
/* Events that can only be reported with data type FSNOTIFY_EVENT_INODE */ /* Events that can only be reported with data type FSNOTIFY_EVENT_INODE */
#define FANOTIFY_INODE_EVENTS (FANOTIFY_DIRENT_EVENTS | \ #define FANOTIFY_INODE_EVENTS (FANOTIFY_DIRENT_EVENTS | \
......
...@@ -18,39 +18,63 @@ ...@@ -18,39 +18,63 @@
#include <linux/bug.h> #include <linux/bug.h>
/* /*
* Notify this @dir inode about a change in the directory entry @dentry. * Notify this @dir inode about a change in a child directory entry.
* The directory entry may have turned positive or negative or its inode may
* have changed (i.e. renamed over).
* *
* Unlike fsnotify_parent(), the event will be reported regardless of the * Unlike fsnotify_parent(), the event will be reported regardless of the
* FS_EVENT_ON_CHILD mask on the parent inode. * FS_EVENT_ON_CHILD mask on the parent inode.
*/ */
static inline int fsnotify_dirent(struct inode *dir, struct dentry *dentry, static inline void fsnotify_name(struct inode *dir, __u32 mask,
__u32 mask) struct inode *child,
const struct qstr *name, u32 cookie)
{ {
return fsnotify(dir, mask, d_inode(dentry), FSNOTIFY_EVENT_INODE, fsnotify(dir, mask, child, FSNOTIFY_EVENT_INODE, name, cookie);
&dentry->d_name, 0); /*
* Send another flavor of the event without child inode data and
* without the specific event type (e.g. FS_CREATE|FS_IS_DIR).
* The name is relative to the dir inode the event is reported to.
*/
fsnotify(dir, FS_DIR_MODIFY, dir, FSNOTIFY_EVENT_INODE, name, 0);
} }
/* Notify this dentry's parent about a child's events. */ static inline void fsnotify_dirent(struct inode *dir, struct dentry *dentry,
static inline int fsnotify_parent(const struct path *path, __u32 mask)
struct dentry *dentry, __u32 mask)
{ {
if (!dentry) fsnotify_name(dir, mask, d_inode(dentry), &dentry->d_name, 0);
dentry = path->dentry;
return __fsnotify_parent(path, dentry, mask);
} }
/* /*
* Simple wrapper to consolidate calls fsnotify_parent()/fsnotify() when * Simple wrappers to consolidate calls fsnotify_parent()/fsnotify() when
* an event is on a path. * an event is on a file/dentry.
*/ */
static inline int fsnotify_path(struct inode *inode, const struct path *path, static inline void fsnotify_dentry(struct dentry *dentry, __u32 mask)
__u32 mask) {
struct inode *inode = d_inode(dentry);
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
fsnotify_parent(dentry, mask, inode, FSNOTIFY_EVENT_INODE);
fsnotify(inode, mask, inode, FSNOTIFY_EVENT_INODE, NULL, 0);
}
static inline int fsnotify_file(struct file *file, __u32 mask)
{ {
int ret = fsnotify_parent(path, NULL, mask); const struct path *path = &file->f_path;
struct inode *inode = file_inode(file);
int ret;
if (file->f_mode & FMODE_NONOTIFY)
return 0;
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
ret = fsnotify_parent(path->dentry, mask, path, FSNOTIFY_EVENT_PATH);
if (ret) if (ret)
return ret; return ret;
return fsnotify(inode, mask, path, FSNOTIFY_EVENT_PATH, NULL, 0); return fsnotify(inode, mask, path, FSNOTIFY_EVENT_PATH, NULL, 0);
} }
...@@ -58,19 +82,16 @@ static inline int fsnotify_path(struct inode *inode, const struct path *path, ...@@ -58,19 +82,16 @@ static inline int fsnotify_path(struct inode *inode, const struct path *path,
static inline int fsnotify_perm(struct file *file, int mask) static inline int fsnotify_perm(struct file *file, int mask)
{ {
int ret; int ret;
const struct path *path = &file->f_path;
struct inode *inode = file_inode(file);
__u32 fsnotify_mask = 0; __u32 fsnotify_mask = 0;
if (file->f_mode & FMODE_NONOTIFY)
return 0;
if (!(mask & (MAY_READ | MAY_OPEN))) if (!(mask & (MAY_READ | MAY_OPEN)))
return 0; return 0;
if (mask & MAY_OPEN) { if (mask & MAY_OPEN) {
fsnotify_mask = FS_OPEN_PERM; fsnotify_mask = FS_OPEN_PERM;
if (file->f_flags & __FMODE_EXEC) { if (file->f_flags & __FMODE_EXEC) {
ret = fsnotify_path(inode, path, FS_OPEN_EXEC_PERM); ret = fsnotify_file(file, FS_OPEN_EXEC_PERM);
if (ret) if (ret)
return ret; return ret;
...@@ -79,10 +100,7 @@ static inline int fsnotify_perm(struct file *file, int mask) ...@@ -79,10 +100,7 @@ static inline int fsnotify_perm(struct file *file, int mask)
fsnotify_mask = FS_ACCESS_PERM; fsnotify_mask = FS_ACCESS_PERM;
} }
if (S_ISDIR(inode->i_mode)) return fsnotify_file(file, fsnotify_mask);
fsnotify_mask |= FS_ISDIR;
return fsnotify_path(inode, path, fsnotify_mask);
} }
/* /*
...@@ -122,10 +140,8 @@ static inline void fsnotify_move(struct inode *old_dir, struct inode *new_dir, ...@@ -122,10 +140,8 @@ static inline void fsnotify_move(struct inode *old_dir, struct inode *new_dir,
mask |= FS_ISDIR; mask |= FS_ISDIR;
} }
fsnotify(old_dir, old_dir_mask, source, FSNOTIFY_EVENT_INODE, old_name, fsnotify_name(old_dir, old_dir_mask, source, old_name, fs_cookie);
fs_cookie); fsnotify_name(new_dir, new_dir_mask, source, new_name, fs_cookie);
fsnotify(new_dir, new_dir_mask, source, FSNOTIFY_EVENT_INODE, new_name,
fs_cookie);
if (target) if (target)
fsnotify_link_count(target); fsnotify_link_count(target);
...@@ -180,12 +196,13 @@ static inline void fsnotify_create(struct inode *inode, struct dentry *dentry) ...@@ -180,12 +196,13 @@ static inline void fsnotify_create(struct inode *inode, struct dentry *dentry)
* Note: We have to pass also the linked inode ptr as some filesystems leave * Note: We have to pass also the linked inode ptr as some filesystems leave
* new_dentry->d_inode NULL and instantiate inode pointer later * new_dentry->d_inode NULL and instantiate inode pointer later
*/ */
static inline void fsnotify_link(struct inode *dir, struct inode *inode, struct dentry *new_dentry) static inline void fsnotify_link(struct inode *dir, struct inode *inode,
struct dentry *new_dentry)
{ {
fsnotify_link_count(inode); fsnotify_link_count(inode);
audit_inode_child(dir, new_dentry, AUDIT_TYPE_CHILD_CREATE); audit_inode_child(dir, new_dentry, AUDIT_TYPE_CHILD_CREATE);
fsnotify(dir, FS_CREATE, inode, FSNOTIFY_EVENT_INODE, &new_dentry->d_name, 0); fsnotify_name(dir, FS_CREATE, inode, &new_dentry->d_name, 0);
} }
/* /*
...@@ -229,15 +246,7 @@ static inline void fsnotify_rmdir(struct inode *dir, struct dentry *dentry) ...@@ -229,15 +246,7 @@ static inline void fsnotify_rmdir(struct inode *dir, struct dentry *dentry)
*/ */
static inline void fsnotify_access(struct file *file) static inline void fsnotify_access(struct file *file)
{ {
const struct path *path = &file->f_path; fsnotify_file(file, FS_ACCESS);
struct inode *inode = file_inode(file);
__u32 mask = FS_ACCESS;
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
if (!(file->f_mode & FMODE_NONOTIFY))
fsnotify_path(inode, path, mask);
} }
/* /*
...@@ -245,15 +254,7 @@ static inline void fsnotify_access(struct file *file) ...@@ -245,15 +254,7 @@ static inline void fsnotify_access(struct file *file)
*/ */
static inline void fsnotify_modify(struct file *file) static inline void fsnotify_modify(struct file *file)
{ {
const struct path *path = &file->f_path; fsnotify_file(file, FS_MODIFY);
struct inode *inode = file_inode(file);
__u32 mask = FS_MODIFY;
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
if (!(file->f_mode & FMODE_NONOTIFY))
fsnotify_path(inode, path, mask);
} }
/* /*
...@@ -261,16 +262,12 @@ static inline void fsnotify_modify(struct file *file) ...@@ -261,16 +262,12 @@ static inline void fsnotify_modify(struct file *file)
*/ */
static inline void fsnotify_open(struct file *file) static inline void fsnotify_open(struct file *file)
{ {
const struct path *path = &file->f_path;
struct inode *inode = file_inode(file);
__u32 mask = FS_OPEN; __u32 mask = FS_OPEN;
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
if (file->f_flags & __FMODE_EXEC) if (file->f_flags & __FMODE_EXEC)
mask |= FS_OPEN_EXEC; mask |= FS_OPEN_EXEC;
fsnotify_path(inode, path, mask); fsnotify_file(file, mask);
} }
/* /*
...@@ -278,16 +275,10 @@ static inline void fsnotify_open(struct file *file) ...@@ -278,16 +275,10 @@ static inline void fsnotify_open(struct file *file)
*/ */
static inline void fsnotify_close(struct file *file) static inline void fsnotify_close(struct file *file)
{ {
const struct path *path = &file->f_path; __u32 mask = (file->f_mode & FMODE_WRITE) ? FS_CLOSE_WRITE :
struct inode *inode = file_inode(file); FS_CLOSE_NOWRITE;
fmode_t mode = file->f_mode;
__u32 mask = (mode & FMODE_WRITE) ? FS_CLOSE_WRITE : FS_CLOSE_NOWRITE;
if (S_ISDIR(inode->i_mode)) fsnotify_file(file, mask);
mask |= FS_ISDIR;
if (!(file->f_mode & FMODE_NONOTIFY))
fsnotify_path(inode, path, mask);
} }
/* /*
...@@ -295,14 +286,7 @@ static inline void fsnotify_close(struct file *file) ...@@ -295,14 +286,7 @@ static inline void fsnotify_close(struct file *file)
*/ */
static inline void fsnotify_xattr(struct dentry *dentry) static inline void fsnotify_xattr(struct dentry *dentry)
{ {
struct inode *inode = dentry->d_inode; fsnotify_dentry(dentry, FS_ATTRIB);
__u32 mask = FS_ATTRIB;
if (S_ISDIR(inode->i_mode))
mask |= FS_ISDIR;
fsnotify_parent(NULL, dentry, mask);
fsnotify(inode, mask, inode, FSNOTIFY_EVENT_INODE, NULL, 0);
} }
/* /*
...@@ -311,7 +295,6 @@ static inline void fsnotify_xattr(struct dentry *dentry) ...@@ -311,7 +295,6 @@ static inline void fsnotify_xattr(struct dentry *dentry)
*/ */
static inline void fsnotify_change(struct dentry *dentry, unsigned int ia_valid) static inline void fsnotify_change(struct dentry *dentry, unsigned int ia_valid)
{ {
struct inode *inode = dentry->d_inode;
__u32 mask = 0; __u32 mask = 0;
if (ia_valid & ATTR_UID) if (ia_valid & ATTR_UID)
...@@ -332,13 +315,8 @@ static inline void fsnotify_change(struct dentry *dentry, unsigned int ia_valid) ...@@ -332,13 +315,8 @@ static inline void fsnotify_change(struct dentry *dentry, unsigned int ia_valid)
if (ia_valid & ATTR_MODE) if (ia_valid & ATTR_MODE)
mask |= FS_ATTRIB; mask |= FS_ATTRIB;
if (mask) { if (mask)
if (S_ISDIR(inode->i_mode)) fsnotify_dentry(dentry, mask);
mask |= FS_ISDIR;
fsnotify_parent(NULL, dentry, mask);
fsnotify(inode, mask, inode, FSNOTIFY_EVENT_INODE, NULL, 0);
}
} }
#endif /* _LINUX_FS_NOTIFY_H */ #endif /* _LINUX_FS_NOTIFY_H */
...@@ -47,18 +47,18 @@ ...@@ -47,18 +47,18 @@
#define FS_OPEN_PERM 0x00010000 /* open event in an permission hook */ #define FS_OPEN_PERM 0x00010000 /* open event in an permission hook */
#define FS_ACCESS_PERM 0x00020000 /* access event in a permissions hook */ #define FS_ACCESS_PERM 0x00020000 /* access event in a permissions hook */
#define FS_OPEN_EXEC_PERM 0x00040000 /* open/exec event in a permission hook */ #define FS_OPEN_EXEC_PERM 0x00040000 /* open/exec event in a permission hook */
#define FS_DIR_MODIFY 0x00080000 /* Directory entry was modified */
#define FS_EXCL_UNLINK 0x04000000 /* do not send events if object is unlinked */ #define FS_EXCL_UNLINK 0x04000000 /* do not send events if object is unlinked */
#define FS_ISDIR 0x40000000 /* event occurred against dir */
#define FS_IN_ONESHOT 0x80000000 /* only send event once */
#define FS_DN_RENAME 0x10000000 /* file renamed */
#define FS_DN_MULTISHOT 0x20000000 /* dnotify multishot */
/* This inode cares about things that happen to its children. Always set for /* This inode cares about things that happen to its children. Always set for
* dnotify and inotify. */ * dnotify and inotify. */
#define FS_EVENT_ON_CHILD 0x08000000 #define FS_EVENT_ON_CHILD 0x08000000
#define FS_DN_RENAME 0x10000000 /* file renamed */
#define FS_DN_MULTISHOT 0x20000000 /* dnotify multishot */
#define FS_ISDIR 0x40000000 /* event occurred against dir */
#define FS_IN_ONESHOT 0x80000000 /* only send event once */
#define FS_MOVE (FS_MOVED_FROM | FS_MOVED_TO) #define FS_MOVE (FS_MOVED_FROM | FS_MOVED_TO)
/* /*
...@@ -67,7 +67,8 @@ ...@@ -67,7 +67,8 @@
* The watching parent may get an FS_ATTRIB|FS_EVENT_ON_CHILD event * The watching parent may get an FS_ATTRIB|FS_EVENT_ON_CHILD event
* when a directory entry inside a child subdir changes. * when a directory entry inside a child subdir changes.
*/ */
#define ALL_FSNOTIFY_DIRENT_EVENTS (FS_CREATE | FS_DELETE | FS_MOVE) #define ALL_FSNOTIFY_DIRENT_EVENTS (FS_CREATE | FS_DELETE | FS_MOVE | \
FS_DIR_MODIFY)
#define ALL_FSNOTIFY_PERM_EVENTS (FS_OPEN_PERM | FS_ACCESS_PERM | \ #define ALL_FSNOTIFY_PERM_EVENTS (FS_OPEN_PERM | FS_ACCESS_PERM | \
FS_OPEN_EXEC_PERM) FS_OPEN_EXEC_PERM)
...@@ -133,8 +134,7 @@ struct fsnotify_ops { ...@@ -133,8 +134,7 @@ struct fsnotify_ops {
*/ */
struct fsnotify_event { struct fsnotify_event {
struct list_head list; struct list_head list;
/* inode may ONLY be dereferenced during handle_event(). */ unsigned long objectid; /* identifier for queue merges */
struct inode *inode; /* either the inode the event happened to or its parent */
}; };
/* /*
...@@ -213,10 +213,36 @@ struct fsnotify_group { ...@@ -213,10 +213,36 @@ struct fsnotify_group {
}; };
}; };
/* when calling fsnotify tell it if the data is a path or inode */ /* When calling fsnotify tell it if the data is a path or inode */
#define FSNOTIFY_EVENT_NONE 0 enum fsnotify_data_type {
#define FSNOTIFY_EVENT_PATH 1 FSNOTIFY_EVENT_NONE,
#define FSNOTIFY_EVENT_INODE 2 FSNOTIFY_EVENT_PATH,
FSNOTIFY_EVENT_INODE,
};
static inline const struct inode *fsnotify_data_inode(const void *data,
int data_type)
{
switch (data_type) {
case FSNOTIFY_EVENT_INODE:
return data;
case FSNOTIFY_EVENT_PATH:
return d_inode(((const struct path *)data)->dentry);
default:
return NULL;
}
}
static inline const struct path *fsnotify_data_path(const void *data,
int data_type)
{
switch (data_type) {
case FSNOTIFY_EVENT_PATH:
return data;
default:
return NULL;
}
}
enum fsnotify_obj_type { enum fsnotify_obj_type {
FSNOTIFY_OBJ_TYPE_INODE, FSNOTIFY_OBJ_TYPE_INODE,
...@@ -351,9 +377,10 @@ struct fsnotify_mark { ...@@ -351,9 +377,10 @@ struct fsnotify_mark {
/* called from the vfs helpers */ /* called from the vfs helpers */
/* main fsnotify call to send events */ /* main fsnotify call to send events */
extern int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is, extern int fsnotify(struct inode *to_tell, __u32 mask, const void *data,
const struct qstr *name, u32 cookie); int data_type, const struct qstr *name, u32 cookie);
extern int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask); extern int fsnotify_parent(struct dentry *dentry, __u32 mask, const void *data,
int data_type);
extern void __fsnotify_inode_delete(struct inode *inode); extern void __fsnotify_inode_delete(struct inode *inode);
extern void __fsnotify_vfsmount_delete(struct vfsmount *mnt); extern void __fsnotify_vfsmount_delete(struct vfsmount *mnt);
extern void fsnotify_sb_delete(struct super_block *sb); extern void fsnotify_sb_delete(struct super_block *sb);
...@@ -500,21 +527,22 @@ extern void fsnotify_finish_user_wait(struct fsnotify_iter_info *iter_info); ...@@ -500,21 +527,22 @@ extern void fsnotify_finish_user_wait(struct fsnotify_iter_info *iter_info);
extern bool fsnotify_prepare_user_wait(struct fsnotify_iter_info *iter_info); extern bool fsnotify_prepare_user_wait(struct fsnotify_iter_info *iter_info);
static inline void fsnotify_init_event(struct fsnotify_event *event, static inline void fsnotify_init_event(struct fsnotify_event *event,
struct inode *inode) unsigned long objectid)
{ {
INIT_LIST_HEAD(&event->list); INIT_LIST_HEAD(&event->list);
event->inode = inode; event->objectid = objectid;
} }
#else #else
static inline int fsnotify(struct inode *to_tell, __u32 mask, const void *data, int data_is, static inline int fsnotify(struct inode *to_tell, __u32 mask, const void *data,
const struct qstr *name, u32 cookie) int data_type, const struct qstr *name, u32 cookie)
{ {
return 0; return 0;
} }
static inline int __fsnotify_parent(const struct path *path, struct dentry *dentry, __u32 mask) static inline int fsnotify_parent(struct dentry *dentry, __u32 mask,
const void *data, int data_type)
{ {
return 0; return 0;
} }
......
...@@ -24,10 +24,11 @@ ...@@ -24,10 +24,11 @@
#define FAN_OPEN_PERM 0x00010000 /* File open in perm check */ #define FAN_OPEN_PERM 0x00010000 /* File open in perm check */
#define FAN_ACCESS_PERM 0x00020000 /* File accessed in perm check */ #define FAN_ACCESS_PERM 0x00020000 /* File accessed in perm check */
#define FAN_OPEN_EXEC_PERM 0x00040000 /* File open/exec in perm check */ #define FAN_OPEN_EXEC_PERM 0x00040000 /* File open/exec in perm check */
#define FAN_DIR_MODIFY 0x00080000 /* Directory entry was modified */
#define FAN_ONDIR 0x40000000 /* event occurred against dir */ #define FAN_EVENT_ON_CHILD 0x08000000 /* Interested in child events */
#define FAN_EVENT_ON_CHILD 0x08000000 /* interested in child events */ #define FAN_ONDIR 0x40000000 /* Event occurred against dir */
/* helper events */ /* helper events */
#define FAN_CLOSE (FAN_CLOSE_WRITE | FAN_CLOSE_NOWRITE) /* close */ #define FAN_CLOSE (FAN_CLOSE_WRITE | FAN_CLOSE_NOWRITE) /* close */
...@@ -116,6 +117,7 @@ struct fanotify_event_metadata { ...@@ -116,6 +117,7 @@ struct fanotify_event_metadata {
}; };
#define FAN_EVENT_INFO_TYPE_FID 1 #define FAN_EVENT_INFO_TYPE_FID 1
#define FAN_EVENT_INFO_TYPE_DFID_NAME 2
/* Variable length info record following event metadata */ /* Variable length info record following event metadata */
struct fanotify_event_info_header { struct fanotify_event_info_header {
...@@ -124,7 +126,12 @@ struct fanotify_event_info_header { ...@@ -124,7 +126,12 @@ struct fanotify_event_info_header {
__u16 len; __u16 len;
}; };
/* Unique file identifier info record */ /*
* Unique file identifier info record. This is used both for
* FAN_EVENT_INFO_TYPE_FID records and for FAN_EVENT_INFO_TYPE_DFID_NAME
* records. For FAN_EVENT_INFO_TYPE_DFID_NAME there is additionally a null
* terminated name immediately after the file handle.
*/
struct fanotify_event_info_fid { struct fanotify_event_info_fid {
struct fanotify_event_info_header hdr; struct fanotify_event_info_header hdr;
__kernel_fsid_t fsid; __kernel_fsid_t fsid;
......
...@@ -160,23 +160,14 @@ static int audit_mark_handle_event(struct fsnotify_group *group, ...@@ -160,23 +160,14 @@ static int audit_mark_handle_event(struct fsnotify_group *group,
{ {
struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info); struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info);
struct audit_fsnotify_mark *audit_mark; struct audit_fsnotify_mark *audit_mark;
const struct inode *inode = NULL; const struct inode *inode = fsnotify_data_inode(data, data_type);
audit_mark = container_of(inode_mark, struct audit_fsnotify_mark, mark); audit_mark = container_of(inode_mark, struct audit_fsnotify_mark, mark);
BUG_ON(group != audit_fsnotify_group); BUG_ON(group != audit_fsnotify_group);
switch (data_type) { if (WARN_ON(!inode))
case (FSNOTIFY_EVENT_PATH):
inode = ((const struct path *)data)->dentry->d_inode;
break;
case (FSNOTIFY_EVENT_INODE):
inode = (const struct inode *)data;
break;
default:
BUG();
return 0; return 0;
}
if (mask & (FS_CREATE|FS_MOVED_TO|FS_DELETE|FS_MOVED_FROM)) { if (mask & (FS_CREATE|FS_MOVED_TO|FS_DELETE|FS_MOVED_FROM)) {
if (audit_compare_dname_path(dname, audit_mark->path, AUDIT_NAME_FULL)) if (audit_compare_dname_path(dname, audit_mark->path, AUDIT_NAME_FULL))
......
...@@ -471,25 +471,13 @@ static int audit_watch_handle_event(struct fsnotify_group *group, ...@@ -471,25 +471,13 @@ static int audit_watch_handle_event(struct fsnotify_group *group,
struct fsnotify_iter_info *iter_info) struct fsnotify_iter_info *iter_info)
{ {
struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info); struct fsnotify_mark *inode_mark = fsnotify_iter_inode_mark(iter_info);
const struct inode *inode; const struct inode *inode = fsnotify_data_inode(data, data_type);
struct audit_parent *parent; struct audit_parent *parent;
parent = container_of(inode_mark, struct audit_parent, mark); parent = container_of(inode_mark, struct audit_parent, mark);
BUG_ON(group != audit_watch_group); BUG_ON(group != audit_watch_group);
WARN_ON(!inode);
switch (data_type) {
case (FSNOTIFY_EVENT_PATH):
inode = d_backing_inode(((const struct path *)data)->dentry);
break;
case (FSNOTIFY_EVENT_INODE):
inode = (const struct inode *)data;
break;
default:
BUG();
inode = NULL;
break;
}
if (mask & (FS_CREATE|FS_MOVED_TO) && inode) if (mask & (FS_CREATE|FS_MOVED_TO) && inode)
audit_update_watch(parent, dname, inode->i_sb->s_dev, inode->i_ino, 0); audit_update_watch(parent, dname, inode->i_sb->s_dev, inode->i_ino, 0);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment