diff mbox series

fs: use wq_has_sleeper() in end_dir_add()

Message ID 20250316232421.1642758-1-mjguzik@gmail.com (mailing list archive)
State New
Headers show
Series fs: use wq_has_sleeper() in end_dir_add() | expand

Commit Message

Mateusz Guzik March 16, 2025, 11:24 p.m. UTC
The routine is used a lot, while the wakeup almost never has anyone to
deal with.

wake_up_all() takes an irq-protected spinlock, wq_has_sleeper() "only"
contains a full fence -- not free by any means, but still cheaper.

Sample result tracing waiters using a custom probe during -j 20 kernel
build (0 - no waiters, 1 - waiters):

@[
    wakeprobe+5
    __wake_up_common+63
    __wake_up+54
    __d_add+234
    d_splice_alias+146
    ext4_lookup+439
    path_openat+1746
    do_filp_open+195
    do_sys_openat2+153
    __x64_sys_openat+86
    do_syscall_64+82
    entry_SYSCALL_64_after_hwframe+118
]:
[0, 1)             13999 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@|
[1, ...)               1 |                                                    |

So that 14000 calls in total from this backtrace, where only one time
had a waiter.

Signed-off-by: Mateusz Guzik <mjguzik@gmail.com>
---
 fs/dcache.c | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

Comments

Mateusz Guzik March 16, 2025, 11:27 p.m. UTC | #1
On Mon, Mar 17, 2025 at 12:24 AM Mateusz Guzik <mjguzik@gmail.com> wrote:
>
> The routine is used a lot, while the wakeup almost never has anyone to
> deal with.
>
> wake_up_all() takes an irq-protected spinlock, wq_has_sleeper() "only"
> contains a full fence -- not free by any means, but still cheaper.
>
> Sample result tracing waiters using a custom probe during -j 20 kernel
> build (0 - no waiters, 1 - waiters):
>
> @[
>     wakeprobe+5
>     __wake_up_common+63
>     __wake_up+54
>     __d_add+234
>     d_splice_alias+146
>     ext4_lookup+439
>     path_openat+1746
>     do_filp_open+195
>     do_sys_openat2+153
>     __x64_sys_openat+86
>     do_syscall_64+82
>     entry_SYSCALL_64_after_hwframe+118
> ]:
> [0, 1)             13999 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@|
> [1, ...)               1 |                                                    |
>
> So that 14000 calls in total from this backtrace, where only one time
> had a waiter.

I noticed unusually weird engrish after sending.

how about this sentence instead:
> Only 1 call out of 14000 with this backtrace had waiters.


>
> Signed-off-by: Mateusz Guzik <mjguzik@gmail.com>
> ---
>  fs/dcache.c | 3 ++-
>  1 file changed, 2 insertions(+), 1 deletion(-)
>
> diff --git a/fs/dcache.c b/fs/dcache.c
> index df8833fe9986..bd5aa136153a 100644
> --- a/fs/dcache.c
> +++ b/fs/dcache.c
> @@ -2497,7 +2497,8 @@ static inline void end_dir_add(struct inode *dir, unsigned int n,
>  {
>         smp_store_release(&dir->i_dir_seq, n + 2);
>         preempt_enable_nested();
> -       wake_up_all(d_wait);
> +       if (wq_has_sleeper(d_wait))
> +               wake_up_all(d_wait);
>  }
>
>  static void d_wait_lookup(struct dentry *dentry)
> --
> 2.43.0
>
Christian Brauner March 17, 2025, 3:06 p.m. UTC | #2
On Mon, 17 Mar 2025 00:24:21 +0100, Mateusz Guzik wrote:
> The routine is used a lot, while the wakeup almost never has anyone to
> deal with.
> 
> wake_up_all() takes an irq-protected spinlock, wq_has_sleeper() "only"
> contains a full fence -- not free by any means, but still cheaper.
> 
> Sample result tracing waiters using a custom probe during -j 20 kernel
> build (0 - no waiters, 1 - waiters):
> 
> [...]

Applied to the vfs-6.15.misc branch of the vfs/vfs.git tree.
Patches in the vfs-6.15.misc branch should appear in linux-next soon.

Please report any outstanding bugs that were missed during review in a
new review to the original patch series allowing us to drop it.

It's encouraged to provide Acked-bys and Reviewed-bys even though the
patch has now been applied. If possible patch trailers will be updated.

Note that commit hashes shown below are subject to change due to rebase,
trailer updates or similar. If in doubt, please check the listed branch.

tree:   https://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs.git
branch: vfs-6.15.misc

[1/1] fs: use wq_has_sleeper() in end_dir_add()
      https://git.kernel.org/vfs/vfs/c/f81ad90aa97e
Jan Kara March 17, 2025, 5:03 p.m. UTC | #3
On Mon 17-03-25 00:24:21, Mateusz Guzik wrote:
> The routine is used a lot, while the wakeup almost never has anyone to
> deal with.
> 
> wake_up_all() takes an irq-protected spinlock, wq_has_sleeper() "only"
> contains a full fence -- not free by any means, but still cheaper.
> 
> Sample result tracing waiters using a custom probe during -j 20 kernel
> build (0 - no waiters, 1 - waiters):
> 
> @[
>     wakeprobe+5
>     __wake_up_common+63
>     __wake_up+54
>     __d_add+234
>     d_splice_alias+146
>     ext4_lookup+439
>     path_openat+1746
>     do_filp_open+195
>     do_sys_openat2+153
>     __x64_sys_openat+86
>     do_syscall_64+82
>     entry_SYSCALL_64_after_hwframe+118
> ]:
> [0, 1)             13999 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@|
> [1, ...)               1 |                                                    |
> 
> So that 14000 calls in total from this backtrace, where only one time
> had a waiter.
> 
> Signed-off-by: Mateusz Guzik <mjguzik@gmail.com>

Looks good. Feel free to add:

Reviewed-by: Jan Kara <jack@suse.cz>

								Honza

> ---
>  fs/dcache.c | 3 ++-
>  1 file changed, 2 insertions(+), 1 deletion(-)
> 
> diff --git a/fs/dcache.c b/fs/dcache.c
> index df8833fe9986..bd5aa136153a 100644
> --- a/fs/dcache.c
> +++ b/fs/dcache.c
> @@ -2497,7 +2497,8 @@ static inline void end_dir_add(struct inode *dir, unsigned int n,
>  {
>  	smp_store_release(&dir->i_dir_seq, n + 2);
>  	preempt_enable_nested();
> -	wake_up_all(d_wait);
> +	if (wq_has_sleeper(d_wait))
> +		wake_up_all(d_wait);
>  }
>  
>  static void d_wait_lookup(struct dentry *dentry)
> -- 
> 2.43.0
>
diff mbox series

Patch

diff --git a/fs/dcache.c b/fs/dcache.c
index df8833fe9986..bd5aa136153a 100644
--- a/fs/dcache.c
+++ b/fs/dcache.c
@@ -2497,7 +2497,8 @@  static inline void end_dir_add(struct inode *dir, unsigned int n,
 {
 	smp_store_release(&dir->i_dir_seq, n + 2);
 	preempt_enable_nested();
-	wake_up_all(d_wait);
+	if (wq_has_sleeper(d_wait))
+		wake_up_all(d_wait);
 }
 
 static void d_wait_lookup(struct dentry *dentry)