Commit 1cc16990 authored by Jeff Layton's avatar Jeff Layton Committed by Ilya Dryomov

ceph: fold ceph_update_writeable_page into ceph_write_begin

...and reorganize the loop for better clarity.
Signed-off-by: default avatarJeff Layton <jlayton@kernel.org>
Signed-off-by: default avatarIlya Dryomov <idryomov@gmail.com>
parent 6390987f
...@@ -1306,104 +1306,84 @@ ceph_find_incompatible(struct page *page) ...@@ -1306,104 +1306,84 @@ ceph_find_incompatible(struct page *page)
/* /*
* We are only allowed to write into/dirty the page if the page is * We are only allowed to write into/dirty the page if the page is
* clean, or already dirty within the same snap context. * clean, or already dirty within the same snap context.
*
* called with page locked.
* return success with page locked,
* or any failure (incl -EAGAIN) with page unlocked.
*/ */
static int ceph_update_writeable_page(struct file *file, static int ceph_write_begin(struct file *file, struct address_space *mapping,
loff_t pos, unsigned len, loff_t pos, unsigned len, unsigned flags,
struct page *page) struct page **pagep, void **fsdata)
{ {
struct inode *inode = file_inode(file); struct inode *inode = file_inode(file);
struct ceph_inode_info *ci = ceph_inode(inode); struct ceph_inode_info *ci = ceph_inode(inode);
struct ceph_snap_context *snapc; struct ceph_snap_context *snapc;
loff_t page_off = pos & PAGE_MASK; struct page *page = NULL;
pgoff_t index = pos >> PAGE_SHIFT;
int pos_in_page = pos & ~PAGE_MASK; int pos_in_page = pos & ~PAGE_MASK;
int end_in_page = pos_in_page + len; int r = 0;
loff_t i_size;
int r;
retry_locked: dout("write_begin file %p inode %p page %p %d~%d\n", file, inode, page, (int)pos, (int)len);
snapc = ceph_find_incompatible(page);
if (snapc) { for (;;) {
if (IS_ERR(snapc)) { page = grab_cache_page_write_begin(mapping, index, 0);
r = PTR_ERR(snapc); if (!page) {
goto fail_unlock; r = -ENOMEM;
break;
} }
unlock_page(page);
ceph_queue_writeback(inode);
r = wait_event_killable(ci->i_cap_wq,
context_is_writeable_or_written(inode, snapc));
ceph_put_snap_context(snapc);
return -EAGAIN;
}
if (PageUptodate(page)) { snapc = ceph_find_incompatible(page);
dout(" page %p already uptodate\n", page); if (snapc) {
return 0; if (IS_ERR(snapc)) {
} r = PTR_ERR(snapc);
break;
}
unlock_page(page);
put_page(page);
page = NULL;
ceph_queue_writeback(inode);
r = wait_event_killable(ci->i_cap_wq,
context_is_writeable_or_written(inode, snapc));
ceph_put_snap_context(snapc);
if (r != 0)
break;
continue;
}
/* full page? */ if (PageUptodate(page)) {
if (pos_in_page == 0 && len == PAGE_SIZE) dout(" page %p already uptodate\n", page);
return 0; break;
}
/* past end of file? */ /*
i_size = i_size_read(inode); * In some cases we don't need to read at all:
* - full page write
if (page_off >= i_size || * - write that lies completely beyond EOF
(pos_in_page == 0 && (pos+len) >= i_size && * - write that covers the the page from start to EOF or beyond it
end_in_page - pos_in_page != PAGE_SIZE)) { */
dout(" zeroing %p 0 - %d and %d - %d\n", if ((pos_in_page == 0 && len == PAGE_SIZE) ||
page, pos_in_page, end_in_page, (int)PAGE_SIZE); (pos >= i_size_read(inode)) ||
zero_user_segments(page, (pos_in_page == 0 && (pos + len) >= i_size_read(inode))) {
0, pos_in_page, zero_user_segments(page, 0, pos_in_page,
end_in_page, PAGE_SIZE); pos_in_page + len, PAGE_SIZE);
return 0; break;
} }
/* we need to read it. */ /*
r = ceph_do_readpage(file, page); * We need to read it. If we get back -EINPROGRESS, then the page was
if (r < 0) { * handed off to fscache and it will be unlocked when the read completes.
if (r == -EINPROGRESS) * Refind the page in that case so we can reacquire the page lock. Otherwise
return -EAGAIN; * we got a hard error or the read was completed synchronously.
goto fail_unlock; */
r = ceph_do_readpage(file, page);
if (r != -EINPROGRESS)
break;
} }
goto retry_locked;
fail_unlock:
unlock_page(page);
return r;
}
/*
* We are only allowed to write into/dirty the page if the page is
* clean, or already dirty within the same snap context.
*/
static int ceph_write_begin(struct file *file, struct address_space *mapping,
loff_t pos, unsigned len, unsigned flags,
struct page **pagep, void **fsdata)
{
struct inode *inode = file_inode(file);
struct page *page;
pgoff_t index = pos >> PAGE_SHIFT;
int r;
do {
/* get a page */
page = grab_cache_page_write_begin(mapping, index, 0);
if (!page)
return -ENOMEM;
dout("write_begin file %p inode %p page %p %d~%d\n", file,
inode, page, (int)pos, (int)len);
r = ceph_update_writeable_page(file, pos, len, page); if (r < 0) {
if (r < 0) if (page) {
unlock_page(page);
put_page(page); put_page(page);
else }
*pagep = page; } else {
} while (r == -EAGAIN); *pagep = page;
}
return r; return r;
} }
......
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