Skip to content

Commit

Permalink
eventfs: Clean up dentry ops and add revalidate function
Browse files Browse the repository at this point in the history
In order for the dentries to stay up-to-date with the eventfs changes,
just add a 'd_revalidate' function that checks the 'is_freed' bit.

Also, clean up the dentry release to actually use d_release() rather
than the slightly odd d_iput() function.  We don't care about the inode,
all we want to do is to get rid of the refcount to the eventfs data
added by dentry->d_fsdata.

It would probably be cleaner to make eventfs its own filesystem, or at
least set its own dentry ops when looking up eventfs files.  But as it
is, only eventfs dentries use d_fsdata, so we don't really need to split
these things up by use.

Another thing that might be worth doing is to make all eventfs lookups
mark their dentries as not worth caching.  We could do that with
d_delete(), but the DCACHE_DONTCACHE flag would likely be even better.

As it is, the dentries are all freeable, but they only tend to get freed
at memory pressure rather than more proactively.  But that's a separate
issue.

Link: https://lore.kernel.org/linux-trace-kernel/[email protected]/
Link: https://lore.kernel.org/linux-trace-kernel/[email protected]

Cc: [email protected]
Cc: Masami Hiramatsu <[email protected]>
Cc: Mark Rutland <[email protected]>
Cc: Mathieu Desnoyers <[email protected]>
Cc: Christian Brauner <[email protected]>
Cc: Al Viro <[email protected]>
Cc: Ajay Kaher <[email protected]>
Cc: Greg Kroah-Hartman <[email protected]>
Fixes: c1504e5 ("eventfs: Implement eventfs dir creation functions")
Signed-off-by: Linus Torvalds <[email protected]>
Signed-off-by: Steven Rostedt (Google) <[email protected]>
  • Loading branch information
torvalds authored and rostedt committed Feb 1, 2024
1 parent 408600b commit 8dce06e
Show file tree
Hide file tree
Showing 3 changed files with 22 additions and 13 deletions.
5 changes: 2 additions & 3 deletions fs/tracefs/event_inode.c
Original file line number Diff line number Diff line change
Expand Up @@ -378,13 +378,12 @@ static void free_ei(struct eventfs_inode *ei)
}

/**
* eventfs_set_ei_status_free - remove the dentry reference from an eventfs_inode
* @ti: the tracefs_inode of the dentry
* eventfs_d_release - dentry is going away
* @dentry: dentry which has the reference to remove.
*
* Remove the association between a dentry from an eventfs_inode.
*/
void eventfs_set_ei_status_free(struct tracefs_inode *ti, struct dentry *dentry)
void eventfs_d_release(struct dentry *dentry)
{
struct eventfs_inode *ei;
int i;
Expand Down
27 changes: 18 additions & 9 deletions fs/tracefs/inode.c
Original file line number Diff line number Diff line change
Expand Up @@ -377,21 +377,30 @@ static const struct super_operations tracefs_super_operations = {
.show_options = tracefs_show_options,
};

static void tracefs_dentry_iput(struct dentry *dentry, struct inode *inode)
/*
* It would be cleaner if eventfs had its own dentry ops.
*
* Note that d_revalidate is called potentially under RCU,
* so it can't take the eventfs mutex etc. It's fine - if
* we open a file just as it's marked dead, things will
* still work just fine, and just see the old stale case.
*/
static void tracefs_d_release(struct dentry *dentry)
{
struct tracefs_inode *ti;
if (dentry->d_fsdata)
eventfs_d_release(dentry);
}

if (!dentry || !inode)
return;
static int tracefs_d_revalidate(struct dentry *dentry, unsigned int flags)
{
struct eventfs_inode *ei = dentry->d_fsdata;

ti = get_tracefs(inode);
if (ti && ti->flags & TRACEFS_EVENT_INODE)
eventfs_set_ei_status_free(ti, dentry);
iput(inode);
return !(ei && ei->is_freed);
}

static const struct dentry_operations tracefs_dentry_operations = {
.d_iput = tracefs_dentry_iput,
.d_revalidate = tracefs_d_revalidate,
.d_release = tracefs_d_release,
};

static int trace_fill_super(struct super_block *sb, void *data, int silent)
Expand Down
3 changes: 2 additions & 1 deletion fs/tracefs/internal.h
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ struct dentry *tracefs_start_creating(const char *name, struct dentry *parent);
struct dentry *tracefs_end_creating(struct dentry *dentry);
struct dentry *tracefs_failed_creating(struct dentry *dentry);
struct inode *tracefs_get_inode(struct super_block *sb);
void eventfs_set_ei_status_free(struct tracefs_inode *ti, struct dentry *dentry);

void eventfs_d_release(struct dentry *dentry);

#endif /* _TRACEFS_INTERNAL_H */

0 comments on commit 8dce06e

Please sign in to comment.