This commit brings back functions that set/clear EXTENT_WRITEBACK bits. These
are required to reliably clear PG_writeback page flag.

Signed-off-by: Chandan Rajendra <chan...@linux.vnet.ibm.com>
---
 fs/btrfs/extent_io.c | 47 +++++++++++++++++++++++++++--------------------
 fs/btrfs/inode.c     | 40 +++++++++++++++++++++++++++++++---------
 2 files changed, 58 insertions(+), 29 deletions(-)

diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c
index c98dfd8..57db008 100644
--- a/fs/btrfs/extent_io.c
+++ b/fs/btrfs/extent_io.c
@@ -1300,6 +1300,20 @@ int clear_extent_uptodate(struct extent_io_tree *tree, 
u64 start, u64 end,
                                cached_state, mask);
 }
 
+static int set_extent_writeback(struct extent_io_tree *tree, u64 start, u64 
end,
+                               struct extent_state **cached_state, gfp_t mask)
+{
+       return set_extent_bit(tree, start, end, EXTENT_WRITEBACK, NULL,
+                       cached_state, mask);
+}
+
+static int clear_extent_writeback(struct extent_io_tree *tree, u64 start, u64 
end,
+                               struct extent_state **cached_state, gfp_t mask)
+{
+       return clear_extent_bit(tree, start, end, EXTENT_WRITEBACK, 1, 0,
+                               cached_state, mask);
+}
+
 /*
  * either insert or lock state struct between start and end use mask to tell
  * us if waiting is desired.
@@ -1406,6 +1420,7 @@ static int set_range_writeback(struct extent_io_tree 
*tree, u64 start, u64 end)
                page_cache_release(page);
                index++;
        }
+       set_extent_writeback(tree, start, end, NULL, GFP_NOFS);
        return 0;
 }
 
@@ -2403,31 +2418,23 @@ static void end_bio_extent_writepage(struct bio *bio, 
int err)
 
        bio_for_each_segment_all(bvec, bio, i) {
                struct page *page = bvec->bv_page;
+               struct inode *inode = page->mapping->host;
+               struct extent_io_tree *tree = &BTRFS_I(inode)->io_tree;
+               u64 page_start, page_end;
 
-               /* We always issue full-page reads, but if some block
-                * in a page fails to read, blk_update_request() will
-                * advance bv_offset and adjust bv_len to compensate.
-                * Print a warning for nonzero offsets, and an error
-                * if they don't add up to a full page.  */
-               if (bvec->bv_offset || bvec->bv_len != PAGE_CACHE_SIZE) {
-                       if (bvec->bv_offset + bvec->bv_len != PAGE_CACHE_SIZE)
-                               
btrfs_err(BTRFS_I(page->mapping->host)->root->fs_info,
-                                  "partial page write in btrfs with offset %u 
and length %u",
-                                       bvec->bv_offset, bvec->bv_len);
-                       else
-                               
btrfs_info(BTRFS_I(page->mapping->host)->root->fs_info,
-                                  "incomplete page write in btrfs with offset 
%u and "
-                                  "length %u",
-                                       bvec->bv_offset, bvec->bv_len);
-               }
-
-               start = page_offset(page);
-               end = start + bvec->bv_offset + bvec->bv_len - 1;
+               start = page_offset(page) + bvec->bv_offset;
+               end = start + bvec->bv_len - 1;
 
                if (end_extent_writepage(page, err, start, end))
                        continue;
 
-               end_page_writeback(page);
+               clear_extent_writeback(tree, start, end, NULL, GFP_ATOMIC);
+
+               page_start = page_offset(page);
+               page_end = page_offset(page) + PAGE_CACHE_SIZE - 1;
+               if (!test_range_bit(tree, page_start, page_end,
+                                       EXTENT_WRITEBACK, 0, NULL))
+                       end_page_writeback(page);
        }
 
        bio_put(bio);
diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c
index 7309832..2ffb4df 100644
--- a/fs/btrfs/inode.c
+++ b/fs/btrfs/inode.c
@@ -2823,22 +2823,44 @@ static int btrfs_writepage_end_io_hook(struct page 
*page, u64 start, u64 end,
        struct btrfs_root *root = BTRFS_I(inode)->root;
        struct btrfs_ordered_extent *ordered_extent = NULL;
        struct btrfs_workqueue *workers;
+       u64 ordered_start, ordered_end;
+       int done;
 
        trace_btrfs_writepage_end_io_hook(page, start, end, uptodate);
 
        ClearPagePrivate2(page);
-       if (!btrfs_dec_test_ordered_pending(inode, &ordered_extent, start,
-                                           end - start + 1, uptodate))
-               return 0;
+loop:
+       ordered_extent = btrfs_lookup_ordered_range(inode, start,
+                                               end - start + 1);
+       if (!ordered_extent)
+               goto out;
 
-       btrfs_init_work(&ordered_extent->work, finish_ordered_fn, NULL, NULL);
+       ordered_start = max_t(u64, start, ordered_extent->file_offset);
+       ordered_end = min_t(u64, end,
+                       ordered_extent->file_offset + ordered_extent->len - 1);
 
-       if (btrfs_is_free_space_inode(inode))
-               workers = root->fs_info->endio_freespace_worker;
-       else
-               workers = root->fs_info->endio_write_workers;
-       btrfs_queue_work(workers, &ordered_extent->work);
+       done = btrfs_dec_test_ordered_pending(inode, &ordered_extent,
+                                       ordered_start,
+                                       ordered_end - ordered_start + 1,
+                                       uptodate);
+       if (done) {
+               btrfs_init_work(&ordered_extent->work, finish_ordered_fn, NULL, 
NULL);
+
+               if (btrfs_is_free_space_inode(inode))
+                       workers = root->fs_info->endio_freespace_worker;
+               else
+                       workers = root->fs_info->endio_write_workers;
 
+               btrfs_queue_work(workers, &ordered_extent->work);
+       }
+
+       btrfs_put_ordered_extent(ordered_extent);
+
+       start = ordered_end + 1;
+
+       if (start < end)
+               goto loop;
+out:
        return 0;
 }
 
-- 
2.1.0

--
To unsubscribe from this list: send the line "unsubscribe linux-btrfs" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to