aboutsummaryrefslogtreecommitdiffstats
path: root/fs/btrfs/disk-io.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/btrfs/disk-io.c')
-rw-r--r--fs/btrfs/disk-io.c13
1 files changed, 10 insertions, 3 deletions
diff --git a/fs/btrfs/disk-io.c b/fs/btrfs/disk-io.c
index 51372a521167..6f58911ece0d 100644
--- a/fs/btrfs/disk-io.c
+++ b/fs/btrfs/disk-io.c
@@ -2735,7 +2735,8 @@ int btrfs_read_buffer(struct extent_buffer *buf, u64 parent_transid)
2735 return ret; 2735 return ret;
2736} 2736}
2737 2737
2738int btree_lock_page_hook(struct page *page) 2738static int btree_lock_page_hook(struct page *page, void *data,
2739 void (*flush_fn)(void *))
2739{ 2740{
2740 struct inode *inode = page->mapping->host; 2741 struct inode *inode = page->mapping->host;
2741 struct btrfs_root *root = BTRFS_I(inode)->root; 2742 struct btrfs_root *root = BTRFS_I(inode)->root;
@@ -2752,7 +2753,10 @@ int btree_lock_page_hook(struct page *page)
2752 if (!eb) 2753 if (!eb)
2753 goto out; 2754 goto out;
2754 2755
2755 btrfs_tree_lock(eb); 2756 if (!btrfs_try_tree_write_lock(eb)) {
2757 flush_fn(data);
2758 btrfs_tree_lock(eb);
2759 }
2756 btrfs_set_header_flag(eb, BTRFS_HEADER_FLAG_WRITTEN); 2760 btrfs_set_header_flag(eb, BTRFS_HEADER_FLAG_WRITTEN);
2757 2761
2758 if (test_and_clear_bit(EXTENT_BUFFER_DIRTY, &eb->bflags)) { 2762 if (test_and_clear_bit(EXTENT_BUFFER_DIRTY, &eb->bflags)) {
@@ -2767,7 +2771,10 @@ int btree_lock_page_hook(struct page *page)
2767 btrfs_tree_unlock(eb); 2771 btrfs_tree_unlock(eb);
2768 free_extent_buffer(eb); 2772 free_extent_buffer(eb);
2769out: 2773out:
2770 lock_page(page); 2774 if (!trylock_page(page)) {
2775 flush_fn(data);
2776 lock_page(page);
2777 }
2771 return 0; 2778 return 0;
2772} 2779}
2773 2780