Commit 972a278f authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'for-5.19-rc7-tag' of git://git.kernel.org/pub/scm/linux/kernel/git/kdave/linux

Pull btrfs reverts from David Sterba:
 "Due to a recent report [1] we need to revert the radix tree to xarray
  conversion patches.

  There's a problem with sleeping under spinlock, when xa_insert could
  allocate memory under pressure. We use GFP_NOFS so this is a real
  problem that we unfortunately did not discover during review.

  I'm sorry to do such change at rc6 time but the revert is IMO the
  safer option, there are patches to use mutex instead of the spin locks
  but that would need more testing. The revert branch has been tested on
  a few setups, all seem ok.

  The conversion to xarray will be revisited in the future"

Link: https://lore.kernel.org/linux-btrfs/cover.1657097693.git.fdmanana@suse.com/ [1]

* tag 'for-5.19-rc7-tag' of git://git.kernel.org/pub/scm/linux/kernel/git/kdave/linux:
  Revert "btrfs: turn delayed_nodes_tree into an XArray"
  Revert "btrfs: turn name_cache radix tree into XArray in send_ctx"
  Revert "btrfs: turn fs_info member buffer_radix into XArray"
  Revert "btrfs: turn fs_roots_radix in btrfs_fs_info into an XArray"
parents c5fe7a97 088aea3b
...@@ -675,9 +675,8 @@ struct btrfs_fs_info { ...@@ -675,9 +675,8 @@ struct btrfs_fs_info {
rwlock_t global_root_lock; rwlock_t global_root_lock;
struct rb_root global_root_tree; struct rb_root global_root_tree;
/* The xarray that holds all the FS roots */ spinlock_t fs_roots_radix_lock;
spinlock_t fs_roots_lock; struct radix_tree_root fs_roots_radix;
struct xarray fs_roots;
/* block group cache stuff */ /* block group cache stuff */
rwlock_t block_group_cache_lock; rwlock_t block_group_cache_lock;
...@@ -995,10 +994,10 @@ struct btrfs_fs_info { ...@@ -995,10 +994,10 @@ struct btrfs_fs_info {
struct btrfs_delayed_root *delayed_root; struct btrfs_delayed_root *delayed_root;
/* Extent buffer xarray */ /* Extent buffer radix tree */
spinlock_t buffer_lock; spinlock_t buffer_lock;
/* Entries are eb->start / sectorsize */ /* Entries are eb->start / sectorsize */
struct xarray extent_buffers; struct radix_tree_root buffer_radix;
/* next backup root to be overwritten */ /* next backup root to be overwritten */
int backup_root_index; int backup_root_index;
...@@ -1119,8 +1118,7 @@ enum { ...@@ -1119,8 +1118,7 @@ enum {
*/ */
BTRFS_ROOT_SHAREABLE, BTRFS_ROOT_SHAREABLE,
BTRFS_ROOT_TRACK_DIRTY, BTRFS_ROOT_TRACK_DIRTY,
/* The root is tracked in fs_info::fs_roots */ BTRFS_ROOT_IN_RADIX,
BTRFS_ROOT_REGISTERED,
BTRFS_ROOT_ORPHAN_ITEM_INSERTED, BTRFS_ROOT_ORPHAN_ITEM_INSERTED,
BTRFS_ROOT_DEFRAG_RUNNING, BTRFS_ROOT_DEFRAG_RUNNING,
BTRFS_ROOT_FORCE_COW, BTRFS_ROOT_FORCE_COW,
...@@ -1224,10 +1222,10 @@ struct btrfs_root { ...@@ -1224,10 +1222,10 @@ struct btrfs_root {
struct rb_root inode_tree; struct rb_root inode_tree;
/* /*
* Xarray that keeps track of delayed nodes of every inode, protected * radix tree that keeps track of delayed nodes of every inode,
* by inode_lock * protected by inode_lock
*/ */
struct xarray delayed_nodes; struct radix_tree_root delayed_nodes_tree;
/* /*
* right now this just gets used so that a root has its own devid * right now this just gets used so that a root has its own devid
* for stat. It may be used for more later * for stat. It may be used for more later
......
...@@ -78,7 +78,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node( ...@@ -78,7 +78,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
} }
spin_lock(&root->inode_lock); spin_lock(&root->inode_lock);
node = xa_load(&root->delayed_nodes, ino); node = radix_tree_lookup(&root->delayed_nodes_tree, ino);
if (node) { if (node) {
if (btrfs_inode->delayed_node) { if (btrfs_inode->delayed_node) {
...@@ -90,9 +90,9 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node( ...@@ -90,9 +90,9 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
/* /*
* It's possible that we're racing into the middle of removing * It's possible that we're racing into the middle of removing
* this node from the xarray. In this case, the refcount * this node from the radix tree. In this case, the refcount
* was zero and it should never go back to one. Just return * was zero and it should never go back to one. Just return
* NULL like it was never in the xarray at all; our release * NULL like it was never in the radix at all; our release
* function is in the process of removing it. * function is in the process of removing it.
* *
* Some implementations of refcount_inc refuse to bump the * Some implementations of refcount_inc refuse to bump the
...@@ -100,7 +100,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node( ...@@ -100,7 +100,7 @@ static struct btrfs_delayed_node *btrfs_get_delayed_node(
* here, refcount_inc() may decide to just WARN_ONCE() instead * here, refcount_inc() may decide to just WARN_ONCE() instead
* of actually bumping the refcount. * of actually bumping the refcount.
* *
* If this node is properly in the xarray, we want to bump the * If this node is properly in the radix, we want to bump the
* refcount twice, once for the inode and once for this get * refcount twice, once for the inode and once for this get
* operation. * operation.
*/ */
...@@ -128,30 +128,36 @@ static struct btrfs_delayed_node *btrfs_get_or_create_delayed_node( ...@@ -128,30 +128,36 @@ static struct btrfs_delayed_node *btrfs_get_or_create_delayed_node(
u64 ino = btrfs_ino(btrfs_inode); u64 ino = btrfs_ino(btrfs_inode);
int ret; int ret;
do { again:
node = btrfs_get_delayed_node(btrfs_inode); node = btrfs_get_delayed_node(btrfs_inode);
if (node) if (node)
return node; return node;
node = kmem_cache_zalloc(delayed_node_cache, GFP_NOFS); node = kmem_cache_zalloc(delayed_node_cache, GFP_NOFS);
if (!node) if (!node)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
btrfs_init_delayed_node(node, root, ino); btrfs_init_delayed_node(node, root, ino);
/* Cached in the inode and can be accessed */ /* cached in the btrfs inode and can be accessed */
refcount_set(&node->refs, 2); refcount_set(&node->refs, 2);
spin_lock(&root->inode_lock); ret = radix_tree_preload(GFP_NOFS);
ret = xa_insert(&root->delayed_nodes, ino, node, GFP_NOFS); if (ret) {
if (ret) { kmem_cache_free(delayed_node_cache, node);
spin_unlock(&root->inode_lock); return ERR_PTR(ret);
kmem_cache_free(delayed_node_cache, node); }
if (ret != -EBUSY)
return ERR_PTR(ret); spin_lock(&root->inode_lock);
} ret = radix_tree_insert(&root->delayed_nodes_tree, ino, node);
} while (ret); if (ret == -EEXIST) {
spin_unlock(&root->inode_lock);
kmem_cache_free(delayed_node_cache, node);
radix_tree_preload_end();
goto again;
}
btrfs_inode->delayed_node = node; btrfs_inode->delayed_node = node;
spin_unlock(&root->inode_lock); spin_unlock(&root->inode_lock);
radix_tree_preload_end();
return node; return node;
} }
...@@ -270,7 +276,8 @@ static void __btrfs_release_delayed_node( ...@@ -270,7 +276,8 @@ static void __btrfs_release_delayed_node(
* back up. We can delete it now. * back up. We can delete it now.
*/ */
ASSERT(refcount_read(&delayed_node->refs) == 0); ASSERT(refcount_read(&delayed_node->refs) == 0);
xa_erase(&root->delayed_nodes, delayed_node->inode_id); radix_tree_delete(&root->delayed_nodes_tree,
delayed_node->inode_id);
spin_unlock(&root->inode_lock); spin_unlock(&root->inode_lock);
kmem_cache_free(delayed_node_cache, delayed_node); kmem_cache_free(delayed_node_cache, delayed_node);
} }
...@@ -1863,35 +1870,34 @@ void btrfs_kill_delayed_inode_items(struct btrfs_inode *inode) ...@@ -1863,35 +1870,34 @@ void btrfs_kill_delayed_inode_items(struct btrfs_inode *inode)
void btrfs_kill_all_delayed_nodes(struct btrfs_root *root) void btrfs_kill_all_delayed_nodes(struct btrfs_root *root)
{ {
unsigned long index = 0; u64 inode_id = 0;
struct btrfs_delayed_node *delayed_node;
struct btrfs_delayed_node *delayed_nodes[8]; struct btrfs_delayed_node *delayed_nodes[8];
int i, n;
while (1) { while (1) {
int n = 0;
spin_lock(&root->inode_lock); spin_lock(&root->inode_lock);
if (xa_empty(&root->delayed_nodes)) { n = radix_tree_gang_lookup(&root->delayed_nodes_tree,
(void **)delayed_nodes, inode_id,
ARRAY_SIZE(delayed_nodes));
if (!n) {
spin_unlock(&root->inode_lock); spin_unlock(&root->inode_lock);
return; break;
} }
xa_for_each_start(&root->delayed_nodes, index, delayed_node, index) { inode_id = delayed_nodes[n - 1]->inode_id + 1;
for (i = 0; i < n; i++) {
/* /*
* Don't increase refs in case the node is dead and * Don't increase refs in case the node is dead and
* about to be removed from the tree in the loop below * about to be removed from the tree in the loop below
*/ */
if (refcount_inc_not_zero(&delayed_node->refs)) { if (!refcount_inc_not_zero(&delayed_nodes[i]->refs))
delayed_nodes[n] = delayed_node; delayed_nodes[i] = NULL;
n++;
}
if (n >= ARRAY_SIZE(delayed_nodes))
break;
} }
index++;
spin_unlock(&root->inode_lock); spin_unlock(&root->inode_lock);
for (int i = 0; i < n; i++) { for (i = 0; i < n; i++) {
if (!delayed_nodes[i])
continue;
__btrfs_kill_delayed_node(delayed_nodes[i]); __btrfs_kill_delayed_node(delayed_nodes[i]);
btrfs_release_delayed_node(delayed_nodes[i]); btrfs_release_delayed_node(delayed_nodes[i]);
} }
......
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#include <linux/fs.h> #include <linux/fs.h>
#include <linux/blkdev.h> #include <linux/blkdev.h>
#include <linux/radix-tree.h>
#include <linux/writeback.h> #include <linux/writeback.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <linux/kthread.h> #include <linux/kthread.h>
...@@ -485,7 +486,7 @@ static int csum_dirty_subpage_buffers(struct btrfs_fs_info *fs_info, ...@@ -485,7 +486,7 @@ static int csum_dirty_subpage_buffers(struct btrfs_fs_info *fs_info,
uptodate = btrfs_subpage_test_uptodate(fs_info, page, cur, uptodate = btrfs_subpage_test_uptodate(fs_info, page, cur,
fs_info->nodesize); fs_info->nodesize);
/* A dirty eb shouldn't disappear from extent_buffers */ /* A dirty eb shouldn't disappear from buffer_radix */
if (WARN_ON(!eb)) if (WARN_ON(!eb))
return -EUCLEAN; return -EUCLEAN;
...@@ -1158,7 +1159,7 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info, ...@@ -1158,7 +1159,7 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info,
root->nr_delalloc_inodes = 0; root->nr_delalloc_inodes = 0;
root->nr_ordered_extents = 0; root->nr_ordered_extents = 0;
root->inode_tree = RB_ROOT; root->inode_tree = RB_ROOT;
xa_init_flags(&root->delayed_nodes, GFP_ATOMIC); INIT_RADIX_TREE(&root->delayed_nodes_tree, GFP_ATOMIC);
btrfs_init_root_block_rsv(root); btrfs_init_root_block_rsv(root);
...@@ -1210,9 +1211,9 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info, ...@@ -1210,9 +1211,9 @@ static void __setup_root(struct btrfs_root *root, struct btrfs_fs_info *fs_info,
btrfs_qgroup_init_swapped_blocks(&root->swapped_blocks); btrfs_qgroup_init_swapped_blocks(&root->swapped_blocks);
#ifdef CONFIG_BTRFS_DEBUG #ifdef CONFIG_BTRFS_DEBUG
INIT_LIST_HEAD(&root->leak_list); INIT_LIST_HEAD(&root->leak_list);
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
list_add_tail(&root->leak_list, &fs_info->allocated_roots); list_add_tail(&root->leak_list, &fs_info->allocated_roots);
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
#endif #endif
} }
...@@ -1659,11 +1660,12 @@ static struct btrfs_root *btrfs_lookup_fs_root(struct btrfs_fs_info *fs_info, ...@@ -1659,11 +1660,12 @@ static struct btrfs_root *btrfs_lookup_fs_root(struct btrfs_fs_info *fs_info,
{ {
struct btrfs_root *root; struct btrfs_root *root;
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
root = xa_load(&fs_info->fs_roots, (unsigned long)root_id); root = radix_tree_lookup(&fs_info->fs_roots_radix,
(unsigned long)root_id);
if (root) if (root)
root = btrfs_grab_root(root); root = btrfs_grab_root(root);
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
return root; return root;
} }
...@@ -1705,14 +1707,20 @@ int btrfs_insert_fs_root(struct btrfs_fs_info *fs_info, ...@@ -1705,14 +1707,20 @@ int btrfs_insert_fs_root(struct btrfs_fs_info *fs_info,
{ {
int ret; int ret;
spin_lock(&fs_info->fs_roots_lock); ret = radix_tree_preload(GFP_NOFS);
ret = xa_insert(&fs_info->fs_roots, (unsigned long)root->root_key.objectid, if (ret)
root, GFP_NOFS); return ret;
spin_lock(&fs_info->fs_roots_radix_lock);
ret = radix_tree_insert(&fs_info->fs_roots_radix,
(unsigned long)root->root_key.objectid,
root);
if (ret == 0) { if (ret == 0) {
btrfs_grab_root(root); btrfs_grab_root(root);
set_bit(BTRFS_ROOT_REGISTERED, &root->state); set_bit(BTRFS_ROOT_IN_RADIX, &root->state);
} }
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
radix_tree_preload_end();
return ret; return ret;
} }
...@@ -2342,9 +2350,9 @@ void btrfs_put_root(struct btrfs_root *root) ...@@ -2342,9 +2350,9 @@ void btrfs_put_root(struct btrfs_root *root)
btrfs_drew_lock_destroy(&root->snapshot_lock); btrfs_drew_lock_destroy(&root->snapshot_lock);
free_root_extent_buffers(root); free_root_extent_buffers(root);
#ifdef CONFIG_BTRFS_DEBUG #ifdef CONFIG_BTRFS_DEBUG
spin_lock(&root->fs_info->fs_roots_lock); spin_lock(&root->fs_info->fs_roots_radix_lock);
list_del_init(&root->leak_list); list_del_init(&root->leak_list);
spin_unlock(&root->fs_info->fs_roots_lock); spin_unlock(&root->fs_info->fs_roots_radix_lock);
#endif #endif
kfree(root); kfree(root);
} }
...@@ -2352,21 +2360,28 @@ void btrfs_put_root(struct btrfs_root *root) ...@@ -2352,21 +2360,28 @@ void btrfs_put_root(struct btrfs_root *root)
void btrfs_free_fs_roots(struct btrfs_fs_info *fs_info) void btrfs_free_fs_roots(struct btrfs_fs_info *fs_info)
{ {
struct btrfs_root *root; int ret;
unsigned long index = 0; struct btrfs_root *gang[8];
int i;
while (!list_empty(&fs_info->dead_roots)) { while (!list_empty(&fs_info->dead_roots)) {
root = list_entry(fs_info->dead_roots.next, gang[0] = list_entry(fs_info->dead_roots.next,
struct btrfs_root, root_list); struct btrfs_root, root_list);
list_del(&root->root_list); list_del(&gang[0]->root_list);
if (test_bit(BTRFS_ROOT_REGISTERED, &root->state)) if (test_bit(BTRFS_ROOT_IN_RADIX, &gang[0]->state))
btrfs_drop_and_free_fs_root(fs_info, root); btrfs_drop_and_free_fs_root(fs_info, gang[0]);
btrfs_put_root(root); btrfs_put_root(gang[0]);
} }
xa_for_each(&fs_info->fs_roots, index, root) { while (1) {
btrfs_drop_and_free_fs_root(fs_info, root); ret = radix_tree_gang_lookup(&fs_info->fs_roots_radix,
(void **)gang, 0,
ARRAY_SIZE(gang));
if (!ret)
break;
for (i = 0; i < ret; i++)
btrfs_drop_and_free_fs_root(fs_info, gang[i]);
} }
} }
...@@ -3134,8 +3149,8 @@ static int __cold init_tree_roots(struct btrfs_fs_info *fs_info) ...@@ -3134,8 +3149,8 @@ static int __cold init_tree_roots(struct btrfs_fs_info *fs_info)
void btrfs_init_fs_info(struct btrfs_fs_info *fs_info) void btrfs_init_fs_info(struct btrfs_fs_info *fs_info)
{ {
xa_init_flags(&fs_info->fs_roots, GFP_ATOMIC); INIT_RADIX_TREE(&fs_info->fs_roots_radix, GFP_ATOMIC);
xa_init_flags(&fs_info->extent_buffers, GFP_ATOMIC); INIT_RADIX_TREE(&fs_info->buffer_radix, GFP_ATOMIC);
INIT_LIST_HEAD(&fs_info->trans_list); INIT_LIST_HEAD(&fs_info->trans_list);
INIT_LIST_HEAD(&fs_info->dead_roots); INIT_LIST_HEAD(&fs_info->dead_roots);
INIT_LIST_HEAD(&fs_info->delayed_iputs); INIT_LIST_HEAD(&fs_info->delayed_iputs);
...@@ -3143,7 +3158,7 @@ void btrfs_init_fs_info(struct btrfs_fs_info *fs_info) ...@@ -3143,7 +3158,7 @@ void btrfs_init_fs_info(struct btrfs_fs_info *fs_info)
INIT_LIST_HEAD(&fs_info->caching_block_groups); INIT_LIST_HEAD(&fs_info->caching_block_groups);
spin_lock_init(&fs_info->delalloc_root_lock); spin_lock_init(&fs_info->delalloc_root_lock);
spin_lock_init(&fs_info->trans_lock); spin_lock_init(&fs_info->trans_lock);
spin_lock_init(&fs_info->fs_roots_lock); spin_lock_init(&fs_info->fs_roots_radix_lock);
spin_lock_init(&fs_info->delayed_iput_lock); spin_lock_init(&fs_info->delayed_iput_lock);
spin_lock_init(&fs_info->defrag_inodes_lock); spin_lock_init(&fs_info->defrag_inodes_lock);
spin_lock_init(&fs_info->super_lock); spin_lock_init(&fs_info->super_lock);
...@@ -3374,7 +3389,7 @@ int btrfs_start_pre_rw_mount(struct btrfs_fs_info *fs_info) ...@@ -3374,7 +3389,7 @@ int btrfs_start_pre_rw_mount(struct btrfs_fs_info *fs_info)
/* /*
* btrfs_find_orphan_roots() is responsible for finding all the dead * btrfs_find_orphan_roots() is responsible for finding all the dead
* roots (with 0 refs), flag them with BTRFS_ROOT_DEAD_TREE and load * roots (with 0 refs), flag them with BTRFS_ROOT_DEAD_TREE and load
* them into the fs_info->fs_roots. This must be done before * them into the fs_info->fs_roots_radix tree. This must be done before
* calling btrfs_orphan_cleanup() on the tree root. If we don't do it * calling btrfs_orphan_cleanup() on the tree root. If we don't do it
* first, then btrfs_orphan_cleanup() will delete a dead root's orphan * first, then btrfs_orphan_cleanup() will delete a dead root's orphan
* item before the root's tree is deleted - this means that if we unmount * item before the root's tree is deleted - this means that if we unmount
...@@ -4499,11 +4514,12 @@ void btrfs_drop_and_free_fs_root(struct btrfs_fs_info *fs_info, ...@@ -4499,11 +4514,12 @@ void btrfs_drop_and_free_fs_root(struct btrfs_fs_info *fs_info,
{ {
bool drop_ref = false; bool drop_ref = false;
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
xa_erase(&fs_info->fs_roots, (unsigned long)root->root_key.objectid); radix_tree_delete(&fs_info->fs_roots_radix,
if (test_and_clear_bit(BTRFS_ROOT_REGISTERED, &root->state)) (unsigned long)root->root_key.objectid);
if (test_and_clear_bit(BTRFS_ROOT_IN_RADIX, &root->state))
drop_ref = true; drop_ref = true;
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
if (BTRFS_FS_ERROR(fs_info)) { if (BTRFS_FS_ERROR(fs_info)) {
ASSERT(root->log_root == NULL); ASSERT(root->log_root == NULL);
...@@ -4519,48 +4535,50 @@ void btrfs_drop_and_free_fs_root(struct btrfs_fs_info *fs_info, ...@@ -4519,48 +4535,50 @@ void btrfs_drop_and_free_fs_root(struct btrfs_fs_info *fs_info,
int btrfs_cleanup_fs_roots(struct btrfs_fs_info *fs_info) int btrfs_cleanup_fs_roots(struct btrfs_fs_info *fs_info)
{ {
struct btrfs_root *roots[8]; u64 root_objectid = 0;
unsigned long index = 0; struct btrfs_root *gang[8];
int i; int i = 0;
int err = 0; int err = 0;
int grabbed; unsigned int ret = 0;
while (1) { while (1) {
struct btrfs_root *root; spin_lock(&fs_info->fs_roots_radix_lock);
ret = radix_tree_gang_lookup(&fs_info->fs_roots_radix,
spin_lock(&fs_info->fs_roots_lock); (void **)gang, root_objectid,
if (!xa_find(&fs_info->fs_roots, &index, ULONG_MAX, XA_PRESENT)) { ARRAY_SIZE(gang));
spin_unlock(&fs_info->fs_roots_lock); if (!ret) {
return err; spin_unlock(&fs_info->fs_roots_radix_lock);
break;
} }
root_objectid = gang[ret - 1]->root_key.objectid + 1;
grabbed = 0; for (i = 0; i < ret; i++) {
xa_for_each_start(&fs_info->fs_roots, index, root, index) { /* Avoid to grab roots in dead_roots */
/* Avoid grabbing roots in dead_roots */ if (btrfs_root_refs(&gang[i]->root_item) == 0) {
if (btrfs_root_refs(&root->root_item) > 0) gang[i] = NULL;
roots[grabbed++] = btrfs_grab_root(root); continue;
if (grabbed >= ARRAY_SIZE(roots)) }
break; /* grab all the search result for later use */
gang[i] = btrfs_grab_root(gang[i]);
} }
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
for (i = 0; i < grabbed; i++) { for (i = 0; i < ret; i++) {
if (!roots[i]) if (!gang[i])
continue; continue;
index = roots[i]->root_key.objectid; root_objectid = gang[i]->root_key.objectid;
err = btrfs_orphan_cleanup(roots[i]); err = btrfs_orphan_cleanup(gang[i]);
if (err) if (err)
goto out; break;
btrfs_put_root(roots[i]); btrfs_put_root(gang[i]);
} }
index++; root_objectid++;
} }
out: /* release the uncleaned roots due to error */
/* Release the roots that remain uncleaned due to error */ for (; i < ret; i++) {
for (; i < grabbed; i++) { if (gang[i])
if (roots[i]) btrfs_put_root(gang[i]);
btrfs_put_root(roots[i]);
} }
return err; return err;
} }
...@@ -4879,28 +4897,31 @@ static void btrfs_error_commit_super(struct btrfs_fs_info *fs_info) ...@@ -4879,28 +4897,31 @@ static void btrfs_error_commit_super(struct btrfs_fs_info *fs_info)
static void btrfs_drop_all_logs(struct btrfs_fs_info *fs_info) static void btrfs_drop_all_logs(struct btrfs_fs_info *fs_info)
{ {
unsigned long index = 0; struct btrfs_root *gang[8];
int grabbed = 0; u64 root_objectid = 0;
struct btrfs_root *roots[8]; int ret;
spin_lock(&fs_info->fs_roots_radix_lock);
while ((ret = radix_tree_gang_lookup(&fs_info->fs_roots_radix,
(void **)gang, root_objectid,
ARRAY_SIZE(gang))) != 0) {
int i;
spin_lock(&fs_info->fs_roots_lock); for (i = 0; i < ret; i++)
while ((grabbed = xa_extract(&fs_info->fs_roots, (void **)roots, index, gang[i] = btrfs_grab_root(gang[i]);
ULONG_MAX, 8, XA_PRESENT))) { spin_unlock(&fs_info->fs_roots_radix_lock);
for (int i = 0; i < grabbed; i++)
roots[i] = btrfs_grab_root(roots[i]);
spin_unlock(&fs_info->fs_roots_lock);
for (int i = 0; i < grabbed; i++) { for (i = 0; i < ret; i++) {
if (!roots[i]) if (!gang[i])
continue; continue;
index = roots[i]->root_key.objectid; root_objectid = gang[i]->root_key.objectid;
btrfs_free_log(NULL, roots[i]); btrfs_free_log(NULL, gang[i]);
btrfs_put_root(roots[i]); btrfs_put_root(gang[i]);
} }
index++; root_objectid++;
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
} }
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
btrfs_free_log_root_tree(NULL, fs_info); btrfs_free_log_root_tree(NULL, fs_info);
} }
......
...@@ -5829,7 +5829,7 @@ int btrfs_drop_snapshot(struct btrfs_root *root, int update_ref, int for_reloc) ...@@ -5829,7 +5829,7 @@ int btrfs_drop_snapshot(struct btrfs_root *root, int update_ref, int for_reloc)
btrfs_qgroup_convert_reserved_meta(root, INT_MAX); btrfs_qgroup_convert_reserved_meta(root, INT_MAX);
btrfs_qgroup_free_meta_all_pertrans(root); btrfs_qgroup_free_meta_all_pertrans(root);
if (test_bit(BTRFS_ROOT_REGISTERED, &root->state)) if (test_bit(BTRFS_ROOT_IN_RADIX, &root->state))
btrfs_add_dropped_root(trans, root); btrfs_add_dropped_root(trans, root);
else else
btrfs_put_root(root); btrfs_put_root(root);
......
...@@ -2966,7 +2966,7 @@ static void begin_page_read(struct btrfs_fs_info *fs_info, struct page *page) ...@@ -2966,7 +2966,7 @@ static void begin_page_read(struct btrfs_fs_info *fs_info, struct page *page)
} }
/* /*
* Find extent buffer for a given bytenr. * Find extent buffer for a givne bytenr.
* *
* This is for end_bio_extent_readpage(), thus we can't do any unsafe locking * This is for end_bio_extent_readpage(), thus we can't do any unsafe locking
* in endio context. * in endio context.
...@@ -2985,9 +2985,11 @@ static struct extent_buffer *find_extent_buffer_readpage( ...@@ -2985,9 +2985,11 @@ static struct extent_buffer *find_extent_buffer_readpage(
return (struct extent_buffer *)page->private; return (struct extent_buffer *)page->private;
} }
/* For subpage case, we need to lookup extent buffer xarray */ /* For subpage case, we need to lookup buffer radix tree */
eb = xa_load(&fs_info->extent_buffers, rcu_read_lock();
bytenr >> fs_info->sectorsize_bits); eb = radix_tree_lookup(&fs_info->buffer_radix,
bytenr >> fs_info->sectorsize_bits);
rcu_read_unlock();
ASSERT(eb); ASSERT(eb);
return eb; return eb;
} }
...@@ -4435,8 +4437,8 @@ static struct extent_buffer *find_extent_buffer_nolock( ...@@ -4435,8 +4437,8 @@ static struct extent_buffer *find_extent_buffer_nolock(
struct extent_buffer *eb; struct extent_buffer *eb;
rcu_read_lock(); rcu_read_lock();
eb = xa_load(&fs_info->extent_buffers, eb = radix_tree_lookup(&fs_info->buffer_radix,
start >> fs_info->sectorsize_bits); start >> fs_info->sectorsize_bits);
if (eb && atomic_inc_not_zero(&eb->refs)) { if (eb && atomic_inc_not_zero(&eb->refs)) {
rcu_read_unlock(); rcu_read_unlock();
return eb; return eb;
...@@ -6129,22 +6131,24 @@ struct extent_buffer *alloc_test_extent_buffer(struct btrfs_fs_info *fs_info, ...@@ -6129,22 +6131,24 @@ struct extent_buffer *alloc_test_extent_buffer(struct btrfs_fs_info *fs_info,
if (!eb) if (!eb)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
eb->fs_info = fs_info; eb->fs_info = fs_info;
again:
do { ret = radix_tree_preload(GFP_NOFS);
ret = xa_insert(&fs_info->extent_buffers, if (ret) {
start >> fs_info->sectorsize_bits, exists = ERR_PTR(ret);
eb, GFP_NOFS); goto free_eb;
if (ret == -ENOMEM) { }
exists = ERR_PTR(ret); spin_lock(&fs_info->buffer_lock);
ret = radix_tree_insert(&fs_info->buffer_radix,
start >> fs_info->sectorsize_bits, eb);
spin_unlock(&fs_info->buffer_lock);
radix_tree_preload_end();
if (ret == -EEXIST) {
exists = find_extent_buffer(fs_info, start);
if (exists)
goto free_eb; goto free_eb;
} else
if (ret == -EBUSY) { goto again;
exists = find_extent_buffer(fs_info, start); }
if (exists)
goto free_eb;
}
} while (ret);
check_buffer_tree_ref(eb); check_buffer_tree_ref(eb);
set_bit(EXTENT_BUFFER_IN_TREE, &eb->bflags); set_bit(EXTENT_BUFFER_IN_TREE, &eb->bflags);
...@@ -6319,22 +6323,25 @@ struct extent_buffer *alloc_extent_buffer(struct btrfs_fs_info *fs_info, ...@@ -6319,22 +6323,25 @@ struct extent_buffer *alloc_extent_buffer(struct btrfs_fs_info *fs_info,
} }
if (uptodate) if (uptodate)
set_bit(EXTENT_BUFFER_UPTODATE, &eb->bflags); set_bit(EXTENT_BUFFER_UPTODATE, &eb->bflags);
again:
do { ret = radix_tree_preload(GFP_NOFS);
ret = xa_insert(&fs_info->extent_buffers, if (ret) {
start >> fs_info->sectorsize_bits, exists = ERR_PTR(ret);
eb, GFP_NOFS); goto free_eb;
if (ret == -ENOMEM) { }
exists = ERR_PTR(ret);
spin_lock(&fs_info->buffer_lock);
ret = radix_tree_insert(&fs_info->buffer_radix,
start >> fs_info->sectorsize_bits, eb);
spin_unlock(&fs_info->buffer_lock);
radix_tree_preload_end();
if (ret == -EEXIST) {
exists = find_extent_buffer(fs_info, start);
if (exists)
goto free_eb; goto free_eb;
} else
if (ret == -EBUSY) { goto again;
exists = find_extent_buffer(fs_info, start); }
if (exists)
goto free_eb;
}
} while (ret);
/* add one reference for the tree */ /* add one reference for the tree */
check_buffer_tree_ref(eb); check_buffer_tree_ref(eb);
set_bit(EXTENT_BUFFER_IN_TREE, &eb->bflags); set_bit(EXTENT_BUFFER_IN_TREE, &eb->bflags);
...@@ -6379,8 +6386,10 @@ static int release_extent_buffer(struct extent_buffer *eb) ...@@ -6379,8 +6386,10 @@ static int release_extent_buffer(struct extent_buffer *eb)
spin_unlock(&eb->refs_lock); spin_unlock(&eb->refs_lock);
xa_erase(&fs_info->extent_buffers, spin_lock(&fs_info->buffer_lock);
eb->start >> fs_info->sectorsize_bits); radix_tree_delete(&fs_info->buffer_radix,
eb->start >> fs_info->sectorsize_bits);
spin_unlock(&fs_info->buffer_lock);
} else { } else {
spin_unlock(&eb->refs_lock); spin_unlock(&eb->refs_lock);
} }
...@@ -7325,25 +7334,42 @@ void memmove_extent_buffer(const struct extent_buffer *dst, ...@@ -7325,25 +7334,42 @@ void memmove_extent_buffer(const struct extent_buffer *dst,
} }
} }
#define GANG_LOOKUP_SIZE 16
static struct extent_buffer *get_next_extent_buffer( static struct extent_buffer *get_next_extent_buffer(
struct btrfs_fs_info *fs_info, struct page *page, u64 bytenr) struct btrfs_fs_info *fs_info, struct page *page, u64 bytenr)
{ {
struct extent_buffer *eb; struct extent_buffer *gang[GANG_LOOKUP_SIZE];
unsigned long index; struct extent_buffer *found = NULL;
u64 page_start = page_offset(page); u64 page_start = page_offset(page);
u64 cur = page_start;
ASSERT(in_range(bytenr, page_start, PAGE_SIZE)); ASSERT(in_range(bytenr, page_start, PAGE_SIZE));
lockdep_assert_held(&fs_info->buffer_lock); lockdep_assert_held(&fs_info->buffer_lock);
xa_for_each_start(&fs_info->extent_buffers, index, eb, while (cur < page_start + PAGE_SIZE) {
page_start >> fs_info->sectorsize_bits) { int ret;
if (in_range(eb->start, page_start, PAGE_SIZE)) int i;
return eb;
else if (eb->start >= page_start + PAGE_SIZE) ret = radix_tree_gang_lookup(&fs_info->buffer_radix,
/* Already beyond page end */ (void **)gang, cur >> fs_info->sectorsize_bits,
return NULL; min_t(unsigned int, GANG_LOOKUP_SIZE,
PAGE_SIZE / fs_info->nodesize));
if (ret == 0)
goto out;
for (i = 0; i < ret; i++) {
/* Already beyond page end */
if (gang[i]->start >= page_start + PAGE_SIZE)
goto out;
/* Found one */
if (gang[i]->start >= bytenr) {
found = gang[i];
goto out;
}
}
cur = gang[ret - 1]->start + gang[ret - 1]->len;
} }
return NULL; out:
return found;
} }
static int try_release_subpage_extent_buffer(struct page *page) static int try_release_subpage_extent_buffer(struct page *page)
......
...@@ -3578,7 +3578,6 @@ int btrfs_orphan_cleanup(struct btrfs_root *root) ...@@ -3578,7 +3578,6 @@ int btrfs_orphan_cleanup(struct btrfs_root *root)
u64 last_objectid = 0; u64 last_objectid = 0;
int ret = 0, nr_unlink = 0; int ret = 0, nr_unlink = 0;
/* Bail out if the cleanup is already running. */
if (test_and_set_bit(BTRFS_ROOT_ORPHAN_CLEANUP, &root->state)) if (test_and_set_bit(BTRFS_ROOT_ORPHAN_CLEANUP, &root->state))
return 0; return 0;
...@@ -3661,17 +3660,17 @@ int btrfs_orphan_cleanup(struct btrfs_root *root) ...@@ -3661,17 +3660,17 @@ int btrfs_orphan_cleanup(struct btrfs_root *root)
* *
* btrfs_find_orphan_roots() ran before us, which has * btrfs_find_orphan_roots() ran before us, which has
* found all deleted roots and loaded them into * found all deleted roots and loaded them into
* fs_info->fs_roots. So here we can find if an * fs_info->fs_roots_radix. So here we can find if an
* orphan item corresponds to a deleted root by looking * orphan item corresponds to a deleted root by looking
* up the root from that xarray. * up the root from that radix tree.
*/ */
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
dead_root = xa_load(&fs_info->fs_roots, dead_root = radix_tree_lookup(&fs_info->fs_roots_radix,
(unsigned long)found_key.objectid); (unsigned long)found_key.objectid);
if (dead_root && btrfs_root_refs(&dead_root->root_item) == 0) if (dead_root && btrfs_root_refs(&dead_root->root_item) == 0)
is_dead_root = 1; is_dead_root = 1;
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
if (is_dead_root) { if (is_dead_root) {
/* prevent this orphan from being found again */ /* prevent this orphan from being found again */
...@@ -3911,7 +3910,7 @@ static int btrfs_read_locked_inode(struct inode *inode, ...@@ -3911,7 +3910,7 @@ static int btrfs_read_locked_inode(struct inode *inode,
* cache. * cache.
* *
* This is required for both inode re-read from disk and delayed inode * This is required for both inode re-read from disk and delayed inode
* in the delayed_nodes xarray. * in delayed_nodes_tree.
*/ */
if (BTRFS_I(inode)->last_trans == fs_info->generation) if (BTRFS_I(inode)->last_trans == fs_info->generation)
set_bit(BTRFS_INODE_NEEDS_FULL_SYNC, set_bit(BTRFS_INODE_NEEDS_FULL_SYNC,
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <linux/mount.h> #include <linux/mount.h>
#include <linux/xattr.h> #include <linux/xattr.h>
#include <linux/posix_acl_xattr.h> #include <linux/posix_acl_xattr.h>
#include <linux/radix-tree.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/string.h> #include <linux/string.h>
#include <linux/compat.h> #include <linux/compat.h>
...@@ -127,7 +128,7 @@ struct send_ctx { ...@@ -127,7 +128,7 @@ struct send_ctx {
struct list_head new_refs; struct list_head new_refs;
struct list_head deleted_refs; struct list_head deleted_refs;
struct xarray name_cache; struct radix_tree_root name_cache;
struct list_head name_cache_list; struct list_head name_cache_list;
int name_cache_size; int name_cache_size;
...@@ -268,13 +269,14 @@ struct orphan_dir_info { ...@@ -268,13 +269,14 @@ struct orphan_dir_info {
struct name_cache_entry { struct name_cache_entry {
struct list_head list; struct list_head list;
/* /*
* On 32bit kernels, xarray has only 32bit indices, but we need to * radix_tree has only 32bit entries but we need to handle 64bit inums.
* handle 64bit inums. We use the lower 32bit of the 64bit inum to store * We use the lower 32bit of the 64bit inum to store it in the tree. If
* it in the tree. If more than one inum would fall into the same entry, * more then one inum would fall into the same entry, we use radix_list
* we use inum_aliases to store the additional entries. inum_aliases is * to store the additional entries. radix_list is also used to store
* also used to store entries with the same inum but different generations. * entries where two entries have the same inum but different
* generations.
*/ */
struct list_head inum_aliases; struct list_head radix_list;
u64 ino; u64 ino;
u64 gen; u64 gen;
u64 parent_ino; u64 parent_ino;
...@@ -2024,9 +2026,9 @@ static int did_overwrite_first_ref(struct send_ctx *sctx, u64 ino, u64 gen) ...@@ -2024,9 +2026,9 @@ static int did_overwrite_first_ref(struct send_ctx *sctx, u64 ino, u64 gen)
} }
/* /*
* Insert a name cache entry. On 32bit kernels the xarray index is 32bit, * Insert a name cache entry. On 32bit kernels the radix tree index is 32bit,
* so we need to do some special handling in case we have clashes. This function * so we need to do some special handling in case we have clashes. This function
* takes care of this with the help of name_cache_entry::inum_aliases. * takes care of this with the help of name_cache_entry::radix_list.
* In case of error, nce is kfreed. * In case of error, nce is kfreed.
*/ */
static int name_cache_insert(struct send_ctx *sctx, static int name_cache_insert(struct send_ctx *sctx,
...@@ -2035,7 +2037,8 @@ static int name_cache_insert(struct send_ctx *sctx, ...@@ -2035,7 +2037,8 @@ static int name_cache_insert(struct send_ctx *sctx,
int ret = 0; int ret = 0;
struct list_head *nce_head; struct list_head *nce_head;
nce_head = xa_load(&sctx->name_cache, (unsigned long)nce->ino); nce_head = radix_tree_lookup(&sctx->name_cache,
(unsigned long)nce->ino);
if (!nce_head) { if (!nce_head) {
nce_head = kmalloc(sizeof(*nce_head), GFP_KERNEL); nce_head = kmalloc(sizeof(*nce_head), GFP_KERNEL);
if (!nce_head) { if (!nce_head) {
...@@ -2044,14 +2047,14 @@ static int name_cache_insert(struct send_ctx *sctx, ...@@ -2044,14 +2047,14 @@ static int name_cache_insert(struct send_ctx *sctx,
} }
INIT_LIST_HEAD(nce_head); INIT_LIST_HEAD(nce_head);
ret = xa_insert(&sctx->name_cache, nce->ino, nce_head, GFP_KERNEL); ret = radix_tree_insert(&sctx->name_cache, nce->ino, nce_head);
if (ret < 0) { if (ret < 0) {
kfree(nce_head); kfree(nce_head);
kfree(nce); kfree(nce);
return ret; return ret;
} }
} }
list_add_tail(&nce->inum_aliases, nce_head); list_add_tail(&nce->radix_list, nce_head);
list_add_tail(&nce->list, &sctx->name_cache_list); list_add_tail(&nce->list, &sctx->name_cache_list);
sctx->name_cache_size++; sctx->name_cache_size++;
...@@ -2063,14 +2066,15 @@ static void name_cache_delete(struct send_ctx *sctx, ...@@ -2063,14 +2066,15 @@ static void name_cache_delete(struct send_ctx *sctx,
{ {
struct list_head *nce_head; struct list_head *nce_head;
nce_head = xa_load(&sctx->name_cache, (unsigned long)nce->ino); nce_head = radix_tree_lookup(&sctx->name_cache,
(unsigned long)nce->ino);
if (!nce_head) { if (!nce_head) {
btrfs_err(sctx->send_root->fs_info, btrfs_err(sctx->send_root->fs_info,
"name_cache_delete lookup failed ino %llu cache size %d, leaking memory", "name_cache_delete lookup failed ino %llu cache size %d, leaking memory",
nce->ino, sctx->name_cache_size); nce->ino, sctx->name_cache_size);
} }
list_del(&nce->inum_aliases); list_del(&nce->radix_list);
list_del(&nce->list); list_del(&nce->list);
sctx->name_cache_size--; sctx->name_cache_size--;
...@@ -2078,7 +2082,7 @@ static void name_cache_delete(struct send_ctx *sctx, ...@@ -2078,7 +2082,7 @@ static void name_cache_delete(struct send_ctx *sctx,
* We may not get to the final release of nce_head if the lookup fails * We may not get to the final release of nce_head if the lookup fails
*/ */
if (nce_head && list_empty(nce_head)) { if (nce_head && list_empty(nce_head)) {
xa_erase(&sctx->name_cache, (unsigned long)nce->ino); radix_tree_delete(&sctx->name_cache, (unsigned long)nce->ino);
kfree(nce_head); kfree(nce_head);
} }
} }
...@@ -2089,11 +2093,11 @@ static struct name_cache_entry *name_cache_search(struct send_ctx *sctx, ...@@ -2089,11 +2093,11 @@ static struct name_cache_entry *name_cache_search(struct send_ctx *sctx,
struct list_head *nce_head; struct list_head *nce_head;
struct name_cache_entry *cur; struct name_cache_entry *cur;
nce_head = xa_load(&sctx->name_cache, (unsigned long)ino); nce_head = radix_tree_lookup(&sctx->name_cache, (unsigned long)ino);
if (!nce_head) if (!nce_head)
return NULL; return NULL;
list_for_each_entry(cur, nce_head, inum_aliases) { list_for_each_entry(cur, nce_head, radix_list) {
if (cur->ino == ino && cur->gen == gen) if (cur->ino == ino && cur->gen == gen)
return cur; return cur;
} }
...@@ -7518,7 +7522,7 @@ long btrfs_ioctl_send(struct inode *inode, struct btrfs_ioctl_send_args *arg) ...@@ -7518,7 +7522,7 @@ long btrfs_ioctl_send(struct inode *inode, struct btrfs_ioctl_send_args *arg)
INIT_LIST_HEAD(&sctx->new_refs); INIT_LIST_HEAD(&sctx->new_refs);
INIT_LIST_HEAD(&sctx->deleted_refs); INIT_LIST_HEAD(&sctx->deleted_refs);
xa_init_flags(&sctx->name_cache, GFP_KERNEL); INIT_RADIX_TREE(&sctx->name_cache, GFP_KERNEL);
INIT_LIST_HEAD(&sctx->name_cache_list); INIT_LIST_HEAD(&sctx->name_cache_list);
sctx->flags = arg->flags; sctx->flags = arg->flags;
......
...@@ -150,8 +150,8 @@ struct btrfs_fs_info *btrfs_alloc_dummy_fs_info(u32 nodesize, u32 sectorsize) ...@@ -150,8 +150,8 @@ struct btrfs_fs_info *btrfs_alloc_dummy_fs_info(u32 nodesize, u32 sectorsize)
void btrfs_free_dummy_fs_info(struct btrfs_fs_info *fs_info) void btrfs_free_dummy_fs_info(struct btrfs_fs_info *fs_info)
{ {
unsigned long index; struct radix_tree_iter iter;
struct extent_buffer *eb; void **slot;
struct btrfs_device *dev, *tmp; struct btrfs_device *dev, *tmp;
if (!fs_info) if (!fs_info)
...@@ -163,9 +163,25 @@ void btrfs_free_dummy_fs_info(struct btrfs_fs_info *fs_info) ...@@ -163,9 +163,25 @@ void btrfs_free_dummy_fs_info(struct btrfs_fs_info *fs_info)
test_mnt->mnt_sb->s_fs_info = NULL; test_mnt->mnt_sb->s_fs_info = NULL;
xa_for_each(&fs_info->extent_buffers, index, eb) { spin_lock(&fs_info->buffer_lock);
radix_tree_for_each_slot(slot, &fs_info->buffer_radix, &iter, 0) {
struct extent_buffer *eb;
eb = radix_tree_deref_slot_protected(slot, &fs_info->buffer_lock);
if (!eb)
continue;
/* Shouldn't happen but that kind of thinking creates CVE's */
if (radix_tree_exception(eb)) {
if (radix_tree_deref_retry(eb))
slot = radix_tree_iter_retry(&iter);
continue;
}
slot = radix_tree_iter_resume(slot, &iter);
spin_unlock(&fs_info->buffer_lock);
free_extent_buffer_stale(eb); free_extent_buffer_stale(eb);
spin_lock(&fs_info->buffer_lock);
} }
spin_unlock(&fs_info->buffer_lock);
btrfs_mapping_tree_free(&fs_info->mapping_tree); btrfs_mapping_tree_free(&fs_info->mapping_tree);
list_for_each_entry_safe(dev, tmp, &fs_info->fs_devices->devices, list_for_each_entry_safe(dev, tmp, &fs_info->fs_devices->devices,
...@@ -186,7 +202,7 @@ void btrfs_free_dummy_root(struct btrfs_root *root) ...@@ -186,7 +202,7 @@ void btrfs_free_dummy_root(struct btrfs_root *root)
if (!root) if (!root)
return; return;
/* Will be freed by btrfs_free_fs_roots */ /* Will be freed by btrfs_free_fs_roots */
if (WARN_ON(test_bit(BTRFS_ROOT_REGISTERED, &root->state))) if (WARN_ON(test_bit(BTRFS_ROOT_IN_RADIX, &root->state)))
return; return;
btrfs_global_root_delete(root); btrfs_global_root_delete(root);
btrfs_put_root(root); btrfs_put_root(root);
......
...@@ -23,7 +23,7 @@ ...@@ -23,7 +23,7 @@
#include "space-info.h" #include "space-info.h"
#include "zoned.h" #include "zoned.h"
#define BTRFS_ROOT_TRANS_TAG XA_MARK_0 #define BTRFS_ROOT_TRANS_TAG 0
/* /*
* Transaction states and transitions * Transaction states and transitions
...@@ -437,15 +437,15 @@ static int record_root_in_trans(struct btrfs_trans_handle *trans, ...@@ -437,15 +437,15 @@ static int record_root_in_trans(struct btrfs_trans_handle *trans,
*/ */
smp_wmb(); smp_wmb();
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
if (root->last_trans == trans->transid && !force) { if (root->last_trans == trans->transid && !force) {
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
return 0; return 0;
} }
xa_set_mark(&fs_info->fs_roots, radix_tree_tag_set(&fs_info->fs_roots_radix,
(unsigned long)root->root_key.objectid, (unsigned long)root->root_key.objectid,
BTRFS_ROOT_TRANS_TAG); BTRFS_ROOT_TRANS_TAG);
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
root->last_trans = trans->transid; root->last_trans = trans->transid;
/* this is pretty tricky. We don't want to /* this is pretty tricky. We don't want to
...@@ -487,9 +487,11 @@ void btrfs_add_dropped_root(struct btrfs_trans_handle *trans, ...@@ -487,9 +487,11 @@ void btrfs_add_dropped_root(struct btrfs_trans_handle *trans,
spin_unlock(&cur_trans->dropped_roots_lock); spin_unlock(&cur_trans->dropped_roots_lock);
/* Make sure we don't try to update the root at commit time */ /* Make sure we don't try to update the root at commit time */
xa_clear_mark(&fs_info->fs_roots, spin_lock(&fs_info->fs_roots_radix_lock);
(unsigned long)root->root_key.objectid, radix_tree_tag_clear(&fs_info->fs_roots_radix,
BTRFS_ROOT_TRANS_TAG); (unsigned long)root->root_key.objectid,
BTRFS_ROOT_TRANS_TAG);
spin_unlock(&fs_info->fs_roots_radix_lock);
} }
int btrfs_record_root_in_trans(struct btrfs_trans_handle *trans, int btrfs_record_root_in_trans(struct btrfs_trans_handle *trans,
...@@ -1402,8 +1404,9 @@ void btrfs_add_dead_root(struct btrfs_root *root) ...@@ -1402,8 +1404,9 @@ void btrfs_add_dead_root(struct btrfs_root *root)
static noinline int commit_fs_roots(struct btrfs_trans_handle *trans) static noinline int commit_fs_roots(struct btrfs_trans_handle *trans)
{ {
struct btrfs_fs_info *fs_info = trans->fs_info; struct btrfs_fs_info *fs_info = trans->fs_info;
struct btrfs_root *root; struct btrfs_root *gang[8];
unsigned long index; int i;
int ret;
/* /*
* At this point no one can be using this transaction to modify any tree * At this point no one can be using this transaction to modify any tree
...@@ -1411,46 +1414,57 @@ static noinline int commit_fs_roots(struct btrfs_trans_handle *trans) ...@@ -1411,46 +1414,57 @@ static noinline int commit_fs_roots(struct btrfs_trans_handle *trans)
*/ */
ASSERT(trans->transaction->state == TRANS_STATE_COMMIT_DOING); ASSERT(trans->transaction->state == TRANS_STATE_COMMIT_DOING);
spin_lock(&fs_info->fs_roots_lock); spin_lock(&fs_info->fs_roots_radix_lock);
xa_for_each_marked(&fs_info->fs_roots, index, root, BTRFS_ROOT_TRANS_TAG) { while (1) {
int ret; ret = radix_tree_gang_lookup_tag(&fs_info->fs_roots_radix,
(void **)gang, 0,
/* ARRAY_SIZE(gang),
* At this point we can neither have tasks logging inodes BTRFS_ROOT_TRANS_TAG);
* from a root nor trying to commit a log tree. if (ret == 0)
*/ break;
ASSERT(atomic_read(&root->log_writers) == 0); for (i = 0; i < ret; i++) {
ASSERT(atomic_read(&root->log_commit[0]) == 0); struct btrfs_root *root = gang[i];
ASSERT(atomic_read(&root->log_commit[1]) == 0); int ret2;
xa_clear_mark(&fs_info->fs_roots, /*
(unsigned long)root->root_key.objectid, * At this point we can neither have tasks logging inodes
BTRFS_ROOT_TRANS_TAG); * from a root nor trying to commit a log tree.
spin_unlock(&fs_info->fs_roots_lock); */
ASSERT(atomic_read(&root->log_writers) == 0);
btrfs_free_log(trans, root); ASSERT(atomic_read(&root->log_commit[0]) == 0);
ret = btrfs_update_reloc_root(trans, root); ASSERT(atomic_read(&root->log_commit[1]) == 0);
if (ret)
return ret; radix_tree_tag_clear(&fs_info->fs_roots_radix,
(unsigned long)root->root_key.objectid,
/* See comments in should_cow_block() */ BTRFS_ROOT_TRANS_TAG);
clear_bit(BTRFS_ROOT_FORCE_COW, &root->state); spin_unlock(&fs_info->fs_roots_radix_lock);
smp_mb__after_atomic();
btrfs_free_log(trans, root);
ret2 = btrfs_update_reloc_root(trans, root);
if (ret2)
return ret2;
/* see comments in should_cow_block() */
clear_bit(BTRFS_ROOT_FORCE_COW, &root->state);
smp_mb__after_atomic();
if (root->commit_root != root->node) {
list_add_tail(&root->dirty_list,
&trans->transaction->switch_commits);
btrfs_set_root_node(&root->root_item,
root->node);
}
if (root->commit_root != root->node) { ret2 = btrfs_update_root(trans, fs_info->tree_root,
list_add_tail(&root->dirty_list, &root->root_key,
&trans->transaction->switch_commits); &root->root_item);
btrfs_set_root_node(&root->root_item, root->node); if (ret2)
return ret2;
spin_lock(&fs_info->fs_roots_radix_lock);
btrfs_qgroup_free_meta_all_pertrans(root);
} }
ret = btrfs_update_root(trans, fs_info->tree_root,
&root->root_key, &root->root_item);
if (ret)
return ret;
spin_lock(&fs_info->fs_roots_lock);
btrfs_qgroup_free_meta_all_pertrans(root);
} }
spin_unlock(&fs_info->fs_roots_lock); spin_unlock(&fs_info->fs_roots_radix_lock);
return 0; return 0;
} }
......
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