Re: [PATCH] ceph: return the real size readed when hit EOF

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On 10/19/21 8:59 PM, Jeff Layton wrote:
On Tue, 2021-10-19 at 19:51 +0800, xiubli@xxxxxxxxxx wrote:
From: Xiubo Li <xiubli@xxxxxxxxxx>

At the same time set the ki_pos to the file size.

It would be good to put the comments in your follow up email into the
patch description here, so that it explains what you're fixing and why.
Sure.

Signed-off-by: Xiubo Li <xiubli@xxxxxxxxxx>
---
  fs/ceph/file.c | 14 +++++++++-----
  1 file changed, 9 insertions(+), 5 deletions(-)

diff --git a/fs/ceph/file.c b/fs/ceph/file.c
index 91173d3aa161..1abc3b591740 100644
--- a/fs/ceph/file.c
+++ b/fs/ceph/file.c
@@ -847,6 +847,7 @@ static ssize_t ceph_sync_read(struct kiocb *iocb, struct iov_iter *to,
  	ssize_t ret;
  	u64 off = iocb->ki_pos;
  	u64 len = iov_iter_count(to);
+	u64 i_size = i_size_read(inode);
Was there a reason to change where the i_size is fetched, or did you
just not see the point in fetching it on each loop? I wonder if we can
hit any races vs. truncates with this?

From my reading from the code, it shouldn't.

While doing the truncate it will down_write(&inode->i_rwsem). And the sync read will hold down_read() it. It should be safe.


Oh well, all of this non-buffered I/O seems somewhat racy anyway. ;)

  	dout("sync_read on file %p %llu~%u %s\n", file, off, (unsigned)len,
  	     (file->f_flags & O_DIRECT) ? "O_DIRECT" : "");
@@ -870,7 +871,6 @@ static ssize_t ceph_sync_read(struct kiocb *iocb, struct iov_iter *to,
  		struct page **pages;
  		int num_pages;
  		size_t page_off;
-		u64 i_size;
  		bool more;
  		int idx;
  		size_t left;
@@ -909,7 +909,6 @@ static ssize_t ceph_sync_read(struct kiocb *iocb, struct iov_iter *to,
ceph_osdc_put_request(req); - i_size = i_size_read(inode);
  		dout("sync_read %llu~%llu got %zd i_size %llu%s\n",
  		     off, len, ret, i_size, (more ? " MORE" : ""));
@@ -954,10 +953,15 @@ static ssize_t ceph_sync_read(struct kiocb *iocb, struct iov_iter *to, if (off > iocb->ki_pos) {
  		if (ret >= 0 &&
Do we need to check ret here?

It seems this check makes no sense even in the following case I mentioned. Will check it more and try to remove it.


I think that if ret < 0, then "off" must
be smaller than "i_size", no?

From current code, there has one case that it's no, such as if the file size is 10, and the ceph may return 4K contents and then the read length will be 4K too, and if it just fails in case:

                        copied = copy_page_to_iter(pages[idx++],
                                                   page_off, len, to);
                        off += copied;
                        left -= copied;
                        if (copied < len) {
                                ret = -EFAULT;
                                break;
                        }

And if the "copied = 1K" for some reasons, the "off" will be larger than the "i_size" but ret < 0.

BRs

Xiubo




-		    iov_iter_count(to) > 0 && off >= i_size_read(inode))
+		    iov_iter_count(to) > 0 &&
+		    off >= i_size_read(inode)) {
  			*retry_op = CHECK_EOF;
-		ret = off - iocb->ki_pos;
-		iocb->ki_pos = off;
+			ret = i_size - iocb->ki_pos;
+			iocb->ki_pos = i_size;
+		} else {
+			ret = off - iocb->ki_pos;
+			iocb->ki_pos = off;
+		}
  	}
dout("sync_read result %zd retry_op %d\n", ret, *retry_op);




[Index of Archives]     [CEPH Users]     [Ceph Large]     [Ceph Dev]     [Information on CEPH]     [Linux BTRFS]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux