From: Jeff Layton <jlayton@kernel.org>
To: Oleg Nesterov <oleg@redhat.com>,
Christian Brauner <brauner@kernel.org>,
Linus Torvalds <torvalds@linux-foundation.org>
Cc: David Howells <dhowells@redhat.com>,
"Gautham R. Shenoy" <gautham.shenoy@amd.com>,
K Prateek Nayak <kprateek.nayak@amd.com>,
Mateusz Guzik <mjguzik@gmail.com>,
Neeraj Upadhyay <Neeraj.Upadhyay@amd.com>,
Oliver Sang <oliver.sang@intel.com>,
Swapnil Sapkal <swapnil.sapkal@amd.com>,
WangYuli <wangyuli@uniontech.com>,
linux-fsdevel@vger.kernel.org, linux-kernel@vger.kernel.org
Subject: Re: [PATCH] pipe: don't update {a,c,m}time for anonymous pipes
Date: Tue, 04 Feb 2025 09:53:13 -0500 [thread overview]
Message-ID: <10981ed475d3b7d2d0bef73ea6286bc60aa484b2.camel@kernel.org> (raw)
In-Reply-To: <20250204132153.GA20921@redhat.com>
On Tue, 2025-02-04 at 14:21 +0100, Oleg Nesterov wrote:
> These numbers are visible in fstat() but hopefully nobody uses this
> information and file_accessed/file_update_time are not that cheap.
> Stupid test-case:
>
> #include <stdio.h>
> #include <stdlib.h>
> #include <unistd.h>
> #include <assert.h>
> #include <sys/ioctl.h>
> #include <sys/time.h>
>
> static char buf[17 * 4096];
> static struct timeval TW, TR;
>
> int wr(int fd, int size)
> {
> int c, r;
> struct timeval t0, t1;
>
> gettimeofday(&t0, NULL);
> for (c = 0; (r = write(fd, buf, size)) > 0; c += r);
> gettimeofday(&t1, NULL);
> timeradd(&TW, &t1, &TW);
> timersub(&TW, &t0, &TW);
>
> return c;
> }
>
> int rd(int fd, int size)
> {
> int c, r;
> struct timeval t0, t1;
>
> gettimeofday(&t0, NULL);
> for (c = 0; (r = read(fd, buf, size)) > 0; c += r);
> gettimeofday(&t1, NULL);
> timeradd(&TR, &t1, &TR);
> timersub(&TR, &t0, &TR);
>
> return c;
> }
>
> int main(int argc, const char *argv[])
> {
> int fd[2], nb = 1, loop, size;
>
> assert(argc == 3);
> loop = atoi(argv[1]);
> size = atoi(argv[2]);
>
> assert(pipe(fd) == 0);
> assert(ioctl(fd[0], FIONBIO, &nb) == 0);
> assert(ioctl(fd[1], FIONBIO, &nb) == 0);
>
> assert(size <= sizeof(buf));
> while (loop--)
> assert(wr(fd[1], size) == rd(fd[0], size));
>
> struct timeval tt;
> timeradd(&TW, &TR, &tt);
> printf("TW = %lu.%03lu TR = %lu.%03lu TT = %lu.%03lu\n",
> TW.tv_sec, TW.tv_usec/1000,
> TR.tv_sec, TR.tv_usec/1000,
> tt.tv_sec, tt.tv_usec/1000);
>
> return 0;
> }
>
> Before:
> # for i in 1 2 3; do /host/tmp/test 10000 100; done
> TW = 8.047 TR = 5.845 TT = 13.893
> TW = 8.091 TR = 5.872 TT = 13.963
> TW = 8.083 TR = 5.885 TT = 13.969
> After:
> # for i in 1 2 3; do /host/tmp/test 10000 100; done
> TW = 4.752 TR = 4.664 TT = 9.416
> TW = 4.684 TR = 4.608 TT = 9.293
> TW = 4.736 TR = 4.652 TT = 9.388
>
> Signed-off-by: Oleg Nesterov <oleg@redhat.com>
> ---
> fs/pipe.c | 21 ++++++++++++++-------
> 1 file changed, 14 insertions(+), 7 deletions(-)
>
> diff --git a/fs/pipe.c b/fs/pipe.c
> index 94b59045ab44..baaa8c0817f1 100644
> --- a/fs/pipe.c
> +++ b/fs/pipe.c
> @@ -247,6 +247,11 @@ static inline unsigned int pipe_update_tail(struct pipe_inode_info *pipe,
> return tail;
> }
>
> +static inline bool is_pipe_inode(struct inode *inode)
> +{
> + return inode->i_sb->s_magic == PIPEFS_MAGIC;
> +}
> +
> static ssize_t
> pipe_read(struct kiocb *iocb, struct iov_iter *to)
> {
> @@ -404,7 +409,7 @@ pipe_read(struct kiocb *iocb, struct iov_iter *to)
> if (wake_next_reader)
> wake_up_interruptible_sync_poll(&pipe->rd_wait, EPOLLIN | EPOLLRDNORM);
> kill_fasync(&pipe->fasync_writers, SIGIO, POLL_OUT);
> - if (ret > 0)
> + if (ret > 0 && !is_pipe_inode(file_inode(filp)))
> file_accessed(filp);
> return ret;
> }
> @@ -604,11 +609,13 @@ pipe_write(struct kiocb *iocb, struct iov_iter *from)
> kill_fasync(&pipe->fasync_readers, SIGIO, POLL_IN);
> if (wake_next_writer)
> wake_up_interruptible_sync_poll(&pipe->wr_wait, EPOLLOUT | EPOLLWRNORM);
> - if (ret > 0 && sb_start_write_trylock(file_inode(filp)->i_sb)) {
> - int err = file_update_time(filp);
> - if (err)
> - ret = err;
> - sb_end_write(file_inode(filp)->i_sb);
> + if (ret > 0 && !is_pipe_inode(file_inode(filp))) {
> + if (sb_start_write_trylock(file_inode(filp)->i_sb)) {
> + int err = file_update_time(filp);
> + if (err)
> + ret = err;
> + sb_end_write(file_inode(filp)->i_sb);
> + }
> }
> return ret;
> }
> @@ -1108,7 +1115,7 @@ static void wake_up_partner(struct pipe_inode_info *pipe)
> static int fifo_open(struct inode *inode, struct file *filp)
> {
> struct pipe_inode_info *pipe;
> - bool is_pipe = inode->i_sb->s_magic == PIPEFS_MAGIC;
> + bool is_pipe = is_pipe_inode(inode);
> int ret;
>
> filp->f_pipe = 0;
Oh _anonymous_ pipes. You can disregard my earlier questions:
Reviewed-by: Jeff Layton <jlayton@kernel.org>
next prev parent reply other threads:[~2025-02-04 14:53 UTC|newest]
Thread overview: 9+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-02-04 13:21 [PATCH] pipe: don't update {a,c,m}time for anonymous pipes Oleg Nesterov
2025-02-04 14:21 ` Mateusz Guzik
2025-02-04 14:39 ` Oleg Nesterov
2025-02-04 16:34 ` Christian Brauner
2025-02-04 16:49 ` Linus Torvalds
2025-02-04 14:48 ` Jeff Layton
2025-02-04 14:53 ` Jeff Layton [this message]
2025-02-04 15:28 ` Linus Torvalds
2025-02-04 15:55 ` Oleg Nesterov
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=10981ed475d3b7d2d0bef73ea6286bc60aa484b2.camel@kernel.org \
--to=jlayton@kernel.org \
--cc=Neeraj.Upadhyay@amd.com \
--cc=brauner@kernel.org \
--cc=dhowells@redhat.com \
--cc=gautham.shenoy@amd.com \
--cc=kprateek.nayak@amd.com \
--cc=linux-fsdevel@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=mjguzik@gmail.com \
--cc=oleg@redhat.com \
--cc=oliver.sang@intel.com \
--cc=swapnil.sapkal@amd.com \
--cc=torvalds@linux-foundation.org \
--cc=wangyuli@uniontech.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox