forked from Minki/linux
Btrfs: Start btree concurrency work.
The allocation trees and the chunk trees are serialized via their own dedicated mutexes. This means allocation location is still not very fine grained. The main FS btree is protected by locks on each block in the btree. Locks are taken top / down, and as processing finishes on a given level of the tree, the lock is released after locking the lower level. The end result of a search is now a path where only the lowest level is locked. Releasing or freeing the path drops any locks held. Signed-off-by: Chris Mason <chris.mason@oracle.com>
This commit is contained in:
parent
1cc127b5d1
commit
925baeddc5
@ -6,7 +6,7 @@ btrfs-y := super.o ctree.o extent-tree.o print-tree.o root-tree.o dir-item.o \
|
|||||||
hash.o file-item.o inode-item.o inode-map.o disk-io.o \
|
hash.o file-item.o inode-item.o inode-map.o disk-io.o \
|
||||||
transaction.o bit-radix.o inode.o file.o tree-defrag.o \
|
transaction.o bit-radix.o inode.o file.o tree-defrag.o \
|
||||||
extent_map.o sysfs.o struct-funcs.o xattr.o ordered-data.o \
|
extent_map.o sysfs.o struct-funcs.o xattr.o ordered-data.o \
|
||||||
extent_io.o volumes.o async-thread.o ioctl.o
|
extent_io.o volumes.o async-thread.o ioctl.o locking.o
|
||||||
|
|
||||||
btrfs-$(CONFIG_FS_POSIX_ACL) += acl.o
|
btrfs-$(CONFIG_FS_POSIX_ACL) += acl.o
|
||||||
else
|
else
|
||||||
|
386
fs/btrfs/ctree.c
386
fs/btrfs/ctree.c
@ -21,6 +21,7 @@
|
|||||||
#include "disk-io.h"
|
#include "disk-io.h"
|
||||||
#include "transaction.h"
|
#include "transaction.h"
|
||||||
#include "print-tree.h"
|
#include "print-tree.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
static int split_node(struct btrfs_trans_handle *trans, struct btrfs_root
|
static int split_node(struct btrfs_trans_handle *trans, struct btrfs_root
|
||||||
*root, struct btrfs_path *path, int level);
|
*root, struct btrfs_path *path, int level);
|
||||||
@ -64,12 +65,47 @@ void btrfs_release_path(struct btrfs_root *root, struct btrfs_path *p)
|
|||||||
int i;
|
int i;
|
||||||
for (i = 0; i < BTRFS_MAX_LEVEL; i++) {
|
for (i = 0; i < BTRFS_MAX_LEVEL; i++) {
|
||||||
if (!p->nodes[i])
|
if (!p->nodes[i])
|
||||||
break;
|
continue;
|
||||||
|
if (p->locks[i]) {
|
||||||
|
btrfs_tree_unlock(p->nodes[i]);
|
||||||
|
p->locks[i] = 0;
|
||||||
|
}
|
||||||
free_extent_buffer(p->nodes[i]);
|
free_extent_buffer(p->nodes[i]);
|
||||||
}
|
}
|
||||||
memset(p, 0, sizeof(*p));
|
memset(p, 0, sizeof(*p));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct extent_buffer *btrfs_root_node(struct btrfs_root *root)
|
||||||
|
{
|
||||||
|
struct extent_buffer *eb;
|
||||||
|
spin_lock(&root->node_lock);
|
||||||
|
eb = root->node;
|
||||||
|
extent_buffer_get(eb);
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
return eb;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct extent_buffer *btrfs_lock_root_node(struct btrfs_root *root)
|
||||||
|
{
|
||||||
|
struct extent_buffer *eb;
|
||||||
|
|
||||||
|
while(1) {
|
||||||
|
eb = btrfs_root_node(root);
|
||||||
|
btrfs_tree_lock(eb);
|
||||||
|
|
||||||
|
spin_lock(&root->node_lock);
|
||||||
|
if (eb == root->node) {
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
|
||||||
|
btrfs_tree_unlock(eb);
|
||||||
|
free_extent_buffer(eb);
|
||||||
|
}
|
||||||
|
return eb;
|
||||||
|
}
|
||||||
|
|
||||||
static void add_root_to_dirty_list(struct btrfs_root *root)
|
static void add_root_to_dirty_list(struct btrfs_root *root)
|
||||||
{
|
{
|
||||||
if (root->track_dirty && list_empty(&root->dirty_list)) {
|
if (root->track_dirty && list_empty(&root->dirty_list)) {
|
||||||
@ -111,7 +147,7 @@ int btrfs_copy_root(struct btrfs_trans_handle *trans,
|
|||||||
} else {
|
} else {
|
||||||
first_key.objectid = 0;
|
first_key.objectid = 0;
|
||||||
}
|
}
|
||||||
cow = __btrfs_alloc_free_block(trans, new_root, buf->len,
|
cow = btrfs_alloc_free_block(trans, new_root, buf->len,
|
||||||
new_root_objectid,
|
new_root_objectid,
|
||||||
trans->transid, first_key.objectid,
|
trans->transid, first_key.objectid,
|
||||||
level, buf->start, 0);
|
level, buf->start, 0);
|
||||||
@ -151,8 +187,14 @@ int __btrfs_cow_block(struct btrfs_trans_handle *trans,
|
|||||||
int ret = 0;
|
int ret = 0;
|
||||||
int different_trans = 0;
|
int different_trans = 0;
|
||||||
int level;
|
int level;
|
||||||
|
int unlock_orig = 0;
|
||||||
struct btrfs_key first_key;
|
struct btrfs_key first_key;
|
||||||
|
|
||||||
|
if (*cow_ret == buf)
|
||||||
|
unlock_orig = 1;
|
||||||
|
|
||||||
|
WARN_ON(!btrfs_tree_locked(buf));
|
||||||
|
|
||||||
if (root->ref_cows) {
|
if (root->ref_cows) {
|
||||||
root_gen = trans->transid;
|
root_gen = trans->transid;
|
||||||
} else {
|
} else {
|
||||||
@ -172,7 +214,7 @@ int __btrfs_cow_block(struct btrfs_trans_handle *trans,
|
|||||||
} else {
|
} else {
|
||||||
first_key.objectid = 0;
|
first_key.objectid = 0;
|
||||||
}
|
}
|
||||||
cow = __btrfs_alloc_free_block(trans, root, buf->len,
|
cow = btrfs_alloc_free_block(trans, root, buf->len,
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
root_gen, first_key.objectid, level,
|
root_gen, first_key.objectid, level,
|
||||||
search_start, empty_size);
|
search_start, empty_size);
|
||||||
@ -196,9 +238,14 @@ int __btrfs_cow_block(struct btrfs_trans_handle *trans,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (buf == root->node) {
|
if (buf == root->node) {
|
||||||
|
WARN_ON(parent && parent != buf);
|
||||||
root_gen = btrfs_header_generation(buf);
|
root_gen = btrfs_header_generation(buf);
|
||||||
|
|
||||||
|
spin_lock(&root->node_lock);
|
||||||
root->node = cow;
|
root->node = cow;
|
||||||
extent_buffer_get(cow);
|
extent_buffer_get(cow);
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
|
||||||
if (buf != root->commit_root) {
|
if (buf != root->commit_root) {
|
||||||
btrfs_free_extent(trans, root, buf->start,
|
btrfs_free_extent(trans, root, buf->start,
|
||||||
buf->len, root->root_key.objectid,
|
buf->len, root->root_key.objectid,
|
||||||
@ -219,6 +266,8 @@ int __btrfs_cow_block(struct btrfs_trans_handle *trans,
|
|||||||
btrfs_header_owner(parent), root_gen,
|
btrfs_header_owner(parent), root_gen,
|
||||||
0, 0, 1);
|
0, 0, 1);
|
||||||
}
|
}
|
||||||
|
if (unlock_orig)
|
||||||
|
btrfs_tree_unlock(buf);
|
||||||
free_extent_buffer(buf);
|
free_extent_buffer(buf);
|
||||||
btrfs_mark_buffer_dirty(cow);
|
btrfs_mark_buffer_dirty(cow);
|
||||||
*cow_ret = cow;
|
*cow_ret = cow;
|
||||||
@ -316,6 +365,9 @@ int btrfs_realloc_node(struct btrfs_trans_handle *trans,
|
|||||||
int progress_passed = 0;
|
int progress_passed = 0;
|
||||||
struct btrfs_disk_key disk_key;
|
struct btrfs_disk_key disk_key;
|
||||||
|
|
||||||
|
/* FIXME this code needs locking */
|
||||||
|
return 0;
|
||||||
|
|
||||||
parent_level = btrfs_header_level(parent);
|
parent_level = btrfs_header_level(parent);
|
||||||
if (cache_only && parent_level != 1)
|
if (cache_only && parent_level != 1)
|
||||||
return 0;
|
return 0;
|
||||||
@ -729,6 +781,7 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
mid = path->nodes[level];
|
mid = path->nodes[level];
|
||||||
|
WARN_ON(!path->locks[level]);
|
||||||
WARN_ON(btrfs_header_generation(mid) != trans->transid);
|
WARN_ON(btrfs_header_generation(mid) != trans->transid);
|
||||||
|
|
||||||
orig_ptr = btrfs_node_blockptr(mid, orig_slot);
|
orig_ptr = btrfs_node_blockptr(mid, orig_slot);
|
||||||
@ -749,14 +802,21 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
|
|
||||||
/* promote the child to a root */
|
/* promote the child to a root */
|
||||||
child = read_node_slot(root, mid, 0);
|
child = read_node_slot(root, mid, 0);
|
||||||
|
btrfs_tree_lock(child);
|
||||||
BUG_ON(!child);
|
BUG_ON(!child);
|
||||||
ret = btrfs_cow_block(trans, root, child, mid, 0, &child);
|
ret = btrfs_cow_block(trans, root, child, mid, 0, &child);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
|
||||||
|
spin_lock(&root->node_lock);
|
||||||
root->node = child;
|
root->node = child;
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
|
||||||
add_root_to_dirty_list(root);
|
add_root_to_dirty_list(root);
|
||||||
|
btrfs_tree_unlock(child);
|
||||||
|
path->locks[level] = 0;
|
||||||
path->nodes[level] = NULL;
|
path->nodes[level] = NULL;
|
||||||
clean_tree_block(trans, root, mid);
|
clean_tree_block(trans, root, mid);
|
||||||
|
btrfs_tree_unlock(mid);
|
||||||
/* once for the path */
|
/* once for the path */
|
||||||
free_extent_buffer(mid);
|
free_extent_buffer(mid);
|
||||||
ret = btrfs_free_extent(trans, root, mid->start, mid->len,
|
ret = btrfs_free_extent(trans, root, mid->start, mid->len,
|
||||||
@ -775,6 +835,7 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
|
|
||||||
left = read_node_slot(root, parent, pslot - 1);
|
left = read_node_slot(root, parent, pslot - 1);
|
||||||
if (left) {
|
if (left) {
|
||||||
|
btrfs_tree_lock(left);
|
||||||
wret = btrfs_cow_block(trans, root, left,
|
wret = btrfs_cow_block(trans, root, left,
|
||||||
parent, pslot - 1, &left);
|
parent, pslot - 1, &left);
|
||||||
if (wret) {
|
if (wret) {
|
||||||
@ -784,6 +845,7 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
}
|
}
|
||||||
right = read_node_slot(root, parent, pslot + 1);
|
right = read_node_slot(root, parent, pslot + 1);
|
||||||
if (right) {
|
if (right) {
|
||||||
|
btrfs_tree_lock(right);
|
||||||
wret = btrfs_cow_block(trans, root, right,
|
wret = btrfs_cow_block(trans, root, right,
|
||||||
parent, pslot + 1, &right);
|
parent, pslot + 1, &right);
|
||||||
if (wret) {
|
if (wret) {
|
||||||
@ -815,6 +877,7 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
u32 blocksize = right->len;
|
u32 blocksize = right->len;
|
||||||
|
|
||||||
clean_tree_block(trans, root, right);
|
clean_tree_block(trans, root, right);
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
right = NULL;
|
right = NULL;
|
||||||
wret = del_ptr(trans, root, path, level + 1, pslot +
|
wret = del_ptr(trans, root, path, level + 1, pslot +
|
||||||
@ -862,7 +925,9 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
u64 root_gen = btrfs_header_generation(parent);
|
u64 root_gen = btrfs_header_generation(parent);
|
||||||
u64 bytenr = mid->start;
|
u64 bytenr = mid->start;
|
||||||
u32 blocksize = mid->len;
|
u32 blocksize = mid->len;
|
||||||
|
|
||||||
clean_tree_block(trans, root, mid);
|
clean_tree_block(trans, root, mid);
|
||||||
|
btrfs_tree_unlock(mid);
|
||||||
free_extent_buffer(mid);
|
free_extent_buffer(mid);
|
||||||
mid = NULL;
|
mid = NULL;
|
||||||
wret = del_ptr(trans, root, path, level + 1, pslot);
|
wret = del_ptr(trans, root, path, level + 1, pslot);
|
||||||
@ -885,11 +950,14 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
if (left) {
|
if (left) {
|
||||||
if (btrfs_header_nritems(left) > orig_slot) {
|
if (btrfs_header_nritems(left) > orig_slot) {
|
||||||
extent_buffer_get(left);
|
extent_buffer_get(left);
|
||||||
|
/* left was locked after cow */
|
||||||
path->nodes[level] = left;
|
path->nodes[level] = left;
|
||||||
path->slots[level + 1] -= 1;
|
path->slots[level + 1] -= 1;
|
||||||
path->slots[level] = orig_slot;
|
path->slots[level] = orig_slot;
|
||||||
if (mid)
|
if (mid) {
|
||||||
|
btrfs_tree_unlock(mid);
|
||||||
free_extent_buffer(mid);
|
free_extent_buffer(mid);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
orig_slot -= btrfs_header_nritems(left);
|
orig_slot -= btrfs_header_nritems(left);
|
||||||
path->slots[level] = orig_slot;
|
path->slots[level] = orig_slot;
|
||||||
@ -901,10 +969,15 @@ static int balance_level(struct btrfs_trans_handle *trans,
|
|||||||
btrfs_node_blockptr(path->nodes[level], path->slots[level]))
|
btrfs_node_blockptr(path->nodes[level], path->slots[level]))
|
||||||
BUG();
|
BUG();
|
||||||
enospc:
|
enospc:
|
||||||
if (right)
|
if (right) {
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
if (left)
|
}
|
||||||
|
if (left) {
|
||||||
|
if (path->nodes[level] != left)
|
||||||
|
btrfs_tree_unlock(left);
|
||||||
free_extent_buffer(left);
|
free_extent_buffer(left);
|
||||||
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -942,6 +1015,8 @@ static int noinline push_nodes_for_insert(struct btrfs_trans_handle *trans,
|
|||||||
/* first, try to make some room in the middle buffer */
|
/* first, try to make some room in the middle buffer */
|
||||||
if (left) {
|
if (left) {
|
||||||
u32 left_nr;
|
u32 left_nr;
|
||||||
|
|
||||||
|
btrfs_tree_lock(left);
|
||||||
left_nr = btrfs_header_nritems(left);
|
left_nr = btrfs_header_nritems(left);
|
||||||
if (left_nr >= BTRFS_NODEPTRS_PER_BLOCK(root) - 1) {
|
if (left_nr >= BTRFS_NODEPTRS_PER_BLOCK(root) - 1) {
|
||||||
wret = 1;
|
wret = 1;
|
||||||
@ -967,24 +1042,28 @@ static int noinline push_nodes_for_insert(struct btrfs_trans_handle *trans,
|
|||||||
path->nodes[level] = left;
|
path->nodes[level] = left;
|
||||||
path->slots[level + 1] -= 1;
|
path->slots[level + 1] -= 1;
|
||||||
path->slots[level] = orig_slot;
|
path->slots[level] = orig_slot;
|
||||||
|
btrfs_tree_unlock(mid);
|
||||||
free_extent_buffer(mid);
|
free_extent_buffer(mid);
|
||||||
} else {
|
} else {
|
||||||
orig_slot -=
|
orig_slot -=
|
||||||
btrfs_header_nritems(left);
|
btrfs_header_nritems(left);
|
||||||
path->slots[level] = orig_slot;
|
path->slots[level] = orig_slot;
|
||||||
|
btrfs_tree_unlock(left);
|
||||||
free_extent_buffer(left);
|
free_extent_buffer(left);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
btrfs_tree_unlock(left);
|
||||||
free_extent_buffer(left);
|
free_extent_buffer(left);
|
||||||
}
|
}
|
||||||
right= read_node_slot(root, parent, pslot + 1);
|
right = read_node_slot(root, parent, pslot + 1);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* then try to empty the right most buffer into the middle
|
* then try to empty the right most buffer into the middle
|
||||||
*/
|
*/
|
||||||
if (right) {
|
if (right) {
|
||||||
u32 right_nr;
|
u32 right_nr;
|
||||||
|
btrfs_tree_lock(right);
|
||||||
right_nr = btrfs_header_nritems(right);
|
right_nr = btrfs_header_nritems(right);
|
||||||
if (right_nr >= BTRFS_NODEPTRS_PER_BLOCK(root) - 1) {
|
if (right_nr >= BTRFS_NODEPTRS_PER_BLOCK(root) - 1) {
|
||||||
wret = 1;
|
wret = 1;
|
||||||
@ -1013,12 +1092,15 @@ static int noinline push_nodes_for_insert(struct btrfs_trans_handle *trans,
|
|||||||
path->slots[level + 1] += 1;
|
path->slots[level + 1] += 1;
|
||||||
path->slots[level] = orig_slot -
|
path->slots[level] = orig_slot -
|
||||||
btrfs_header_nritems(mid);
|
btrfs_header_nritems(mid);
|
||||||
|
btrfs_tree_unlock(mid);
|
||||||
free_extent_buffer(mid);
|
free_extent_buffer(mid);
|
||||||
} else {
|
} else {
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
@ -1050,6 +1132,8 @@ static void reada_for_search(struct btrfs_root *root, struct btrfs_path *path,
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
node = path->nodes[level];
|
node = path->nodes[level];
|
||||||
|
WARN_ON(!path->skip_locking && !btrfs_tree_locked(node));
|
||||||
|
|
||||||
search = btrfs_node_blockptr(node, slot);
|
search = btrfs_node_blockptr(node, slot);
|
||||||
blocksize = btrfs_level_size(root, level - 1);
|
blocksize = btrfs_level_size(root, level - 1);
|
||||||
eb = btrfs_find_tree_block(root, search, blocksize);
|
eb = btrfs_find_tree_block(root, search, blocksize);
|
||||||
@ -1098,6 +1182,39 @@ static void reada_for_search(struct btrfs_root *root, struct btrfs_path *path,
|
|||||||
highest_read = search;
|
highest_read = search;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void unlock_up(struct btrfs_path *path, int level, int lowest_unlock)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int skip_level = level;
|
||||||
|
struct extent_buffer *t;
|
||||||
|
|
||||||
|
for (i = level; i < BTRFS_MAX_LEVEL; i++) {
|
||||||
|
if (!path->nodes[i])
|
||||||
|
break;
|
||||||
|
if (!path->locks[i])
|
||||||
|
break;
|
||||||
|
if (path->slots[i] == 0) {
|
||||||
|
skip_level = i + 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (path->keep_locks) {
|
||||||
|
u32 nritems;
|
||||||
|
t = path->nodes[i];
|
||||||
|
nritems = btrfs_header_nritems(t);
|
||||||
|
if (path->slots[i] >= nritems - 1) {
|
||||||
|
skip_level = i + 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t = path->nodes[i];
|
||||||
|
if (i >= lowest_unlock && i > skip_level && path->locks[i]) {
|
||||||
|
btrfs_tree_unlock(t);
|
||||||
|
path->locks[i] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* look for key in the tree. path is filled in with nodes along the way
|
* look for key in the tree. path is filled in with nodes along the way
|
||||||
* if key is found, we return zero and you can find the item in the leaf
|
* if key is found, we return zero and you can find the item in the leaf
|
||||||
@ -1120,15 +1237,27 @@ int btrfs_search_slot(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
int ret;
|
int ret;
|
||||||
int level;
|
int level;
|
||||||
int should_reada = p->reada;
|
int should_reada = p->reada;
|
||||||
|
int lowest_unlock = 1;
|
||||||
u8 lowest_level = 0;
|
u8 lowest_level = 0;
|
||||||
|
|
||||||
lowest_level = p->lowest_level;
|
lowest_level = p->lowest_level;
|
||||||
WARN_ON(lowest_level && ins_len);
|
WARN_ON(lowest_level && ins_len);
|
||||||
WARN_ON(p->nodes[0] != NULL);
|
WARN_ON(p->nodes[0] != NULL);
|
||||||
WARN_ON(!mutex_is_locked(&root->fs_info->fs_mutex));
|
// WARN_ON(!mutex_is_locked(&root->fs_info->fs_mutex));
|
||||||
|
WARN_ON(root == root->fs_info->extent_root &&
|
||||||
|
!mutex_is_locked(&root->fs_info->alloc_mutex));
|
||||||
|
WARN_ON(root == root->fs_info->chunk_root &&
|
||||||
|
!mutex_is_locked(&root->fs_info->chunk_mutex));
|
||||||
|
WARN_ON(root == root->fs_info->dev_root &&
|
||||||
|
!mutex_is_locked(&root->fs_info->chunk_mutex));
|
||||||
|
if (ins_len < 0)
|
||||||
|
lowest_unlock = 2;
|
||||||
again:
|
again:
|
||||||
b = root->node;
|
if (!p->skip_locking)
|
||||||
extent_buffer_get(b);
|
b = btrfs_lock_root_node(root);
|
||||||
|
else
|
||||||
|
b = btrfs_root_node(root);
|
||||||
|
|
||||||
while (b) {
|
while (b) {
|
||||||
level = btrfs_header_level(b);
|
level = btrfs_header_level(b);
|
||||||
if (cow) {
|
if (cow) {
|
||||||
@ -1147,9 +1276,12 @@ again:
|
|||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
level = btrfs_header_level(b);
|
level = btrfs_header_level(b);
|
||||||
p->nodes[level] = b;
|
p->nodes[level] = b;
|
||||||
|
if (!p->skip_locking)
|
||||||
|
p->locks[level] = 1;
|
||||||
ret = check_block(root, p, level);
|
ret = check_block(root, p, level);
|
||||||
if (ret)
|
if (ret)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
ret = bin_search(b, key, level, &slot);
|
ret = bin_search(b, key, level, &slot);
|
||||||
if (level != 0) {
|
if (level != 0) {
|
||||||
if (ret && slot > 0)
|
if (ret && slot > 0)
|
||||||
@ -1177,14 +1309,19 @@ again:
|
|||||||
BUG_ON(btrfs_header_nritems(b) == 1);
|
BUG_ON(btrfs_header_nritems(b) == 1);
|
||||||
}
|
}
|
||||||
/* this is only true while dropping a snapshot */
|
/* this is only true while dropping a snapshot */
|
||||||
if (level == lowest_level)
|
if (level == lowest_level) {
|
||||||
|
unlock_up(p, level, lowest_unlock);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if (should_reada)
|
if (should_reada)
|
||||||
reada_for_search(root, p, level, slot,
|
reada_for_search(root, p, level, slot,
|
||||||
key->objectid);
|
key->objectid);
|
||||||
|
|
||||||
b = read_node_slot(root, b, slot);
|
b = read_node_slot(root, b, slot);
|
||||||
|
if (!p->skip_locking)
|
||||||
|
btrfs_tree_lock(b);
|
||||||
|
unlock_up(p, level, lowest_unlock);
|
||||||
} else {
|
} else {
|
||||||
p->slots[level] = slot;
|
p->slots[level] = slot;
|
||||||
if (ins_len > 0 && btrfs_leaf_free_space(root, b) <
|
if (ins_len > 0 && btrfs_leaf_free_space(root, b) <
|
||||||
@ -1195,6 +1332,7 @@ again:
|
|||||||
if (sret)
|
if (sret)
|
||||||
return sret;
|
return sret;
|
||||||
}
|
}
|
||||||
|
unlock_up(p, level, lowest_unlock);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1225,6 +1363,13 @@ static int fixup_low_keys(struct btrfs_trans_handle *trans,
|
|||||||
break;
|
break;
|
||||||
t = path->nodes[i];
|
t = path->nodes[i];
|
||||||
btrfs_set_node_key(t, key, tslot);
|
btrfs_set_node_key(t, key, tslot);
|
||||||
|
if (!btrfs_tree_locked(path->nodes[i])) {
|
||||||
|
int ii;
|
||||||
|
printk("fixup without lock on level %d\n", btrfs_header_level(path->nodes[i]));
|
||||||
|
for (ii = 0; ii < BTRFS_MAX_LEVEL; ii++) {
|
||||||
|
printk("level %d slot %d\n", ii, path->slots[ii]);
|
||||||
|
}
|
||||||
|
}
|
||||||
btrfs_mark_buffer_dirty(path->nodes[i]);
|
btrfs_mark_buffer_dirty(path->nodes[i]);
|
||||||
if (tslot != 0)
|
if (tslot != 0)
|
||||||
break;
|
break;
|
||||||
@ -1370,6 +1515,7 @@ static int noinline insert_new_root(struct btrfs_trans_handle *trans,
|
|||||||
u64 lower_gen;
|
u64 lower_gen;
|
||||||
struct extent_buffer *lower;
|
struct extent_buffer *lower;
|
||||||
struct extent_buffer *c;
|
struct extent_buffer *c;
|
||||||
|
struct extent_buffer *old;
|
||||||
struct btrfs_disk_key lower_key;
|
struct btrfs_disk_key lower_key;
|
||||||
|
|
||||||
BUG_ON(path->nodes[level]);
|
BUG_ON(path->nodes[level]);
|
||||||
@ -1386,12 +1532,13 @@ static int noinline insert_new_root(struct btrfs_trans_handle *trans,
|
|||||||
else
|
else
|
||||||
btrfs_node_key(lower, &lower_key, 0);
|
btrfs_node_key(lower, &lower_key, 0);
|
||||||
|
|
||||||
c = __btrfs_alloc_free_block(trans, root, root->nodesize,
|
c = btrfs_alloc_free_block(trans, root, root->nodesize,
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
root_gen, lower_key.objectid, level,
|
root_gen, lower_key.objectid, level,
|
||||||
root->node->start, 0);
|
root->node->start, 0);
|
||||||
if (IS_ERR(c))
|
if (IS_ERR(c))
|
||||||
return PTR_ERR(c);
|
return PTR_ERR(c);
|
||||||
|
|
||||||
memset_extent_buffer(c, 0, 0, root->nodesize);
|
memset_extent_buffer(c, 0, 0, root->nodesize);
|
||||||
btrfs_set_header_nritems(c, 1);
|
btrfs_set_header_nritems(c, 1);
|
||||||
btrfs_set_header_level(c, level);
|
btrfs_set_header_level(c, level);
|
||||||
@ -1416,23 +1563,31 @@ static int noinline insert_new_root(struct btrfs_trans_handle *trans,
|
|||||||
|
|
||||||
btrfs_mark_buffer_dirty(c);
|
btrfs_mark_buffer_dirty(c);
|
||||||
|
|
||||||
/* the super has an extra ref to root->node */
|
spin_lock(&root->node_lock);
|
||||||
free_extent_buffer(root->node);
|
old = root->node;
|
||||||
root->node = c;
|
root->node = c;
|
||||||
|
spin_unlock(&root->node_lock);
|
||||||
|
|
||||||
|
/* the super has an extra ref to root->node */
|
||||||
|
free_extent_buffer(old);
|
||||||
|
|
||||||
add_root_to_dirty_list(root);
|
add_root_to_dirty_list(root);
|
||||||
extent_buffer_get(c);
|
extent_buffer_get(c);
|
||||||
path->nodes[level] = c;
|
path->nodes[level] = c;
|
||||||
|
path->locks[level] = 1;
|
||||||
path->slots[level] = 0;
|
path->slots[level] = 0;
|
||||||
|
|
||||||
if (root->ref_cows && lower_gen != trans->transid) {
|
if (root->ref_cows && lower_gen != trans->transid) {
|
||||||
struct btrfs_path *back_path = btrfs_alloc_path();
|
struct btrfs_path *back_path = btrfs_alloc_path();
|
||||||
int ret;
|
int ret;
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
ret = btrfs_insert_extent_backref(trans,
|
ret = btrfs_insert_extent_backref(trans,
|
||||||
root->fs_info->extent_root,
|
root->fs_info->extent_root,
|
||||||
path, lower->start,
|
path, lower->start,
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
trans->transid, 0, 0);
|
trans->transid, 0, 0);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
btrfs_free_path(back_path);
|
btrfs_free_path(back_path);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
@ -1521,7 +1676,7 @@ static int split_node(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
root_gen = 0;
|
root_gen = 0;
|
||||||
|
|
||||||
btrfs_node_key(c, &disk_key, 0);
|
btrfs_node_key(c, &disk_key, 0);
|
||||||
split = __btrfs_alloc_free_block(trans, root, root->nodesize,
|
split = btrfs_alloc_free_block(trans, root, root->nodesize,
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
root_gen,
|
root_gen,
|
||||||
btrfs_disk_key_objectid(&disk_key),
|
btrfs_disk_key_objectid(&disk_key),
|
||||||
@ -1564,10 +1719,12 @@ static int split_node(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
|
|
||||||
if (path->slots[level] >= mid) {
|
if (path->slots[level] >= mid) {
|
||||||
path->slots[level] -= mid;
|
path->slots[level] -= mid;
|
||||||
|
btrfs_tree_unlock(c);
|
||||||
free_extent_buffer(c);
|
free_extent_buffer(c);
|
||||||
path->nodes[level] = split;
|
path->nodes[level] = split;
|
||||||
path->slots[level + 1] += 1;
|
path->slots[level + 1] += 1;
|
||||||
} else {
|
} else {
|
||||||
|
btrfs_tree_unlock(split);
|
||||||
free_extent_buffer(split);
|
free_extent_buffer(split);
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
@ -1648,30 +1805,24 @@ static int push_leaf_right(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
right = read_node_slot(root, upper, slot + 1);
|
right = read_node_slot(root, upper, slot + 1);
|
||||||
|
btrfs_tree_lock(right);
|
||||||
free_space = btrfs_leaf_free_space(root, right);
|
free_space = btrfs_leaf_free_space(root, right);
|
||||||
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
if (free_space < data_size + sizeof(struct btrfs_item))
|
||||||
free_extent_buffer(right);
|
goto out_unlock;
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* cow and double check */
|
/* cow and double check */
|
||||||
ret = btrfs_cow_block(trans, root, right, upper,
|
ret = btrfs_cow_block(trans, root, right, upper,
|
||||||
slot + 1, &right);
|
slot + 1, &right);
|
||||||
if (ret) {
|
if (ret)
|
||||||
free_extent_buffer(right);
|
goto out_unlock;
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
free_space = btrfs_leaf_free_space(root, right);
|
free_space = btrfs_leaf_free_space(root, right);
|
||||||
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
if (free_space < data_size + sizeof(struct btrfs_item))
|
||||||
free_extent_buffer(right);
|
goto out_unlock;
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
left_nritems = btrfs_header_nritems(left);
|
left_nritems = btrfs_header_nritems(left);
|
||||||
if (left_nritems == 0) {
|
if (left_nritems == 0)
|
||||||
free_extent_buffer(right);
|
goto out_unlock;
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (empty)
|
if (empty)
|
||||||
nr = 0;
|
nr = 0;
|
||||||
@ -1707,10 +1858,8 @@ static int push_leaf_right(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
left->map_token = NULL;
|
left->map_token = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (push_items == 0) {
|
if (push_items == 0)
|
||||||
free_extent_buffer(right);
|
goto out_unlock;
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty && push_items == left_nritems)
|
if (!empty && push_items == left_nritems)
|
||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
@ -1778,14 +1927,24 @@ static int push_leaf_right(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
/* then fixup the leaf pointer in the path */
|
/* then fixup the leaf pointer in the path */
|
||||||
if (path->slots[0] >= left_nritems) {
|
if (path->slots[0] >= left_nritems) {
|
||||||
path->slots[0] -= left_nritems;
|
path->slots[0] -= left_nritems;
|
||||||
|
if (btrfs_header_nritems(path->nodes[0]) == 0)
|
||||||
|
clean_tree_block(trans, root, path->nodes[0]);
|
||||||
|
btrfs_tree_unlock(path->nodes[0]);
|
||||||
free_extent_buffer(path->nodes[0]);
|
free_extent_buffer(path->nodes[0]);
|
||||||
path->nodes[0] = right;
|
path->nodes[0] = right;
|
||||||
path->slots[1] += 1;
|
path->slots[1] += 1;
|
||||||
} else {
|
} else {
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
out_unlock:
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
|
free_extent_buffer(right);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* push some data in the path leaf to the left, trying to free up at
|
* push some data in the path leaf to the left, trying to free up at
|
||||||
* least data_size bytes. returns zero if the push worked, nonzero otherwise
|
* least data_size bytes. returns zero if the push worked, nonzero otherwise
|
||||||
@ -1823,10 +1982,11 @@ static int push_leaf_left(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
}
|
}
|
||||||
|
|
||||||
left = read_node_slot(root, path->nodes[1], slot - 1);
|
left = read_node_slot(root, path->nodes[1], slot - 1);
|
||||||
|
btrfs_tree_lock(left);
|
||||||
free_space = btrfs_leaf_free_space(root, left);
|
free_space = btrfs_leaf_free_space(root, left);
|
||||||
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
||||||
free_extent_buffer(left);
|
ret = 1;
|
||||||
return 1;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* cow and double check */
|
/* cow and double check */
|
||||||
@ -1834,14 +1994,14 @@ static int push_leaf_left(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
path->nodes[1], slot - 1, &left);
|
path->nodes[1], slot - 1, &left);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
/* we hit -ENOSPC, but it isn't fatal here */
|
/* we hit -ENOSPC, but it isn't fatal here */
|
||||||
free_extent_buffer(left);
|
ret = 1;
|
||||||
return 1;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
free_space = btrfs_leaf_free_space(root, left);
|
free_space = btrfs_leaf_free_space(root, left);
|
||||||
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
if (free_space < data_size + sizeof(struct btrfs_item)) {
|
||||||
free_extent_buffer(left);
|
ret = 1;
|
||||||
return 1;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (empty)
|
if (empty)
|
||||||
@ -1876,8 +2036,8 @@ static int push_leaf_left(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (push_items == 0) {
|
if (push_items == 0) {
|
||||||
free_extent_buffer(left);
|
ret = 1;
|
||||||
return 1;
|
goto out;
|
||||||
}
|
}
|
||||||
if (!empty && push_items == btrfs_header_nritems(right))
|
if (!empty && push_items == btrfs_header_nritems(right))
|
||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
@ -1975,15 +2135,23 @@ static int push_leaf_left(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
/* then fixup the leaf pointer in the path */
|
/* then fixup the leaf pointer in the path */
|
||||||
if (path->slots[0] < push_items) {
|
if (path->slots[0] < push_items) {
|
||||||
path->slots[0] += old_left_nritems;
|
path->slots[0] += old_left_nritems;
|
||||||
|
if (btrfs_header_nritems(path->nodes[0]) == 0)
|
||||||
|
clean_tree_block(trans, root, path->nodes[0]);
|
||||||
|
btrfs_tree_unlock(path->nodes[0]);
|
||||||
free_extent_buffer(path->nodes[0]);
|
free_extent_buffer(path->nodes[0]);
|
||||||
path->nodes[0] = left;
|
path->nodes[0] = left;
|
||||||
path->slots[1] -= 1;
|
path->slots[1] -= 1;
|
||||||
} else {
|
} else {
|
||||||
|
btrfs_tree_unlock(left);
|
||||||
free_extent_buffer(left);
|
free_extent_buffer(left);
|
||||||
path->slots[0] -= push_items;
|
path->slots[0] -= push_items;
|
||||||
}
|
}
|
||||||
BUG_ON(path->slots[0] < 0);
|
BUG_ON(path->slots[0] < 0);
|
||||||
return ret;
|
return ret;
|
||||||
|
out:
|
||||||
|
btrfs_tree_unlock(left);
|
||||||
|
free_extent_buffer(left);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2052,7 +2220,7 @@ again:
|
|||||||
|
|
||||||
btrfs_item_key(l, &disk_key, 0);
|
btrfs_item_key(l, &disk_key, 0);
|
||||||
|
|
||||||
right = __btrfs_alloc_free_block(trans, root, root->leafsize,
|
right = btrfs_alloc_free_block(trans, root, root->leafsize,
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
root_gen, disk_key.objectid, 0,
|
root_gen, disk_key.objectid, 0,
|
||||||
l->start, 0);
|
l->start, 0);
|
||||||
@ -2085,6 +2253,8 @@ again:
|
|||||||
path->slots[1] + 1, 1);
|
path->slots[1] + 1, 1);
|
||||||
if (wret)
|
if (wret)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
|
|
||||||
|
btrfs_tree_unlock(path->nodes[0]);
|
||||||
free_extent_buffer(path->nodes[0]);
|
free_extent_buffer(path->nodes[0]);
|
||||||
path->nodes[0] = right;
|
path->nodes[0] = right;
|
||||||
path->slots[0] = 0;
|
path->slots[0] = 0;
|
||||||
@ -2111,6 +2281,7 @@ again:
|
|||||||
path->slots[1], 1);
|
path->slots[1], 1);
|
||||||
if (wret)
|
if (wret)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
|
btrfs_tree_unlock(path->nodes[0]);
|
||||||
free_extent_buffer(path->nodes[0]);
|
free_extent_buffer(path->nodes[0]);
|
||||||
path->nodes[0] = right;
|
path->nodes[0] = right;
|
||||||
path->slots[0] = 0;
|
path->slots[0] = 0;
|
||||||
@ -2184,12 +2355,15 @@ again:
|
|||||||
BUG_ON(path->slots[0] != slot);
|
BUG_ON(path->slots[0] != slot);
|
||||||
|
|
||||||
if (mid <= slot) {
|
if (mid <= slot) {
|
||||||
|
btrfs_tree_unlock(path->nodes[0]);
|
||||||
free_extent_buffer(path->nodes[0]);
|
free_extent_buffer(path->nodes[0]);
|
||||||
path->nodes[0] = right;
|
path->nodes[0] = right;
|
||||||
path->slots[0] -= mid;
|
path->slots[0] -= mid;
|
||||||
path->slots[1] += 1;
|
path->slots[1] += 1;
|
||||||
} else
|
} else {
|
||||||
|
btrfs_tree_unlock(right);
|
||||||
free_extent_buffer(right);
|
free_extent_buffer(right);
|
||||||
|
}
|
||||||
|
|
||||||
BUG_ON(path->slots[0] < 0);
|
BUG_ON(path->slots[0] < 0);
|
||||||
|
|
||||||
@ -2418,10 +2592,6 @@ int btrfs_insert_empty_items(struct btrfs_trans_handle *trans,
|
|||||||
total_data += data_size[i];
|
total_data += data_size[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* create a root if there isn't one */
|
|
||||||
if (!root->node)
|
|
||||||
BUG();
|
|
||||||
|
|
||||||
total_size = total_data + (nr - 1) * sizeof(struct btrfs_item);
|
total_size = total_data + (nr - 1) * sizeof(struct btrfs_item);
|
||||||
ret = btrfs_search_slot(trans, root, cpu_key, path, total_size, 1);
|
ret = btrfs_search_slot(trans, root, cpu_key, path, total_size, 1);
|
||||||
if (ret == 0) {
|
if (ret == 0) {
|
||||||
@ -2516,7 +2686,6 @@ int btrfs_insert_empty_items(struct btrfs_trans_handle *trans,
|
|||||||
btrfs_print_leaf(root, leaf);
|
btrfs_print_leaf(root, leaf);
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -2655,7 +2824,6 @@ int btrfs_del_items(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
btrfs_set_header_level(leaf, 0);
|
btrfs_set_header_level(leaf, 0);
|
||||||
} else {
|
} else {
|
||||||
u64 root_gen = btrfs_header_generation(path->nodes[1]);
|
u64 root_gen = btrfs_header_generation(path->nodes[1]);
|
||||||
clean_tree_block(trans, root, leaf);
|
|
||||||
wret = del_ptr(trans, root, path, 1, path->slots[1]);
|
wret = del_ptr(trans, root, path, 1, path->slots[1]);
|
||||||
if (wret)
|
if (wret)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
@ -2706,8 +2874,6 @@ int btrfs_del_items(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
root_gen = btrfs_header_generation(
|
root_gen = btrfs_header_generation(
|
||||||
path->nodes[1]);
|
path->nodes[1]);
|
||||||
|
|
||||||
clean_tree_block(trans, root, leaf);
|
|
||||||
|
|
||||||
wret = del_ptr(trans, root, path, 1, slot);
|
wret = del_ptr(trans, root, path, 1, slot);
|
||||||
if (wret)
|
if (wret)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
@ -2720,7 +2886,13 @@ int btrfs_del_items(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
if (wret)
|
if (wret)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
} else {
|
} else {
|
||||||
btrfs_mark_buffer_dirty(leaf);
|
/* if we're still in the path, make sure
|
||||||
|
* we're dirty. Otherwise, one of the
|
||||||
|
* push_leaf functions must have already
|
||||||
|
* dirtied this buffer
|
||||||
|
*/
|
||||||
|
if (path->nodes[0] == leaf)
|
||||||
|
btrfs_mark_buffer_dirty(leaf);
|
||||||
free_extent_buffer(leaf);
|
free_extent_buffer(leaf);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -2731,56 +2903,40 @@ int btrfs_del_items(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* walk up the tree as far as required to find the previous leaf.
|
* search the tree again to find a leaf with lesser keys
|
||||||
* returns 0 if it found something or 1 if there are no lesser leaves.
|
* returns 0 if it found something or 1 if there are no lesser leaves.
|
||||||
* returns < 0 on io errors.
|
* returns < 0 on io errors.
|
||||||
*/
|
*/
|
||||||
int btrfs_prev_leaf(struct btrfs_root *root, struct btrfs_path *path)
|
int btrfs_prev_leaf(struct btrfs_root *root, struct btrfs_path *path)
|
||||||
{
|
{
|
||||||
int slot;
|
struct btrfs_key key;
|
||||||
int level = 1;
|
struct btrfs_disk_key found_key;
|
||||||
struct extent_buffer *c;
|
int ret;
|
||||||
struct extent_buffer *next = NULL;
|
|
||||||
|
|
||||||
while(level < BTRFS_MAX_LEVEL) {
|
btrfs_item_key_to_cpu(path->nodes[0], &key, 0);
|
||||||
if (!path->nodes[level])
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
slot = path->slots[level];
|
if (key.offset > 0)
|
||||||
c = path->nodes[level];
|
key.offset--;
|
||||||
if (slot == 0) {
|
else if (key.type > 0)
|
||||||
level++;
|
key.type--;
|
||||||
if (level == BTRFS_MAX_LEVEL)
|
else if (key.objectid > 0)
|
||||||
return 1;
|
key.objectid--;
|
||||||
continue;
|
else
|
||||||
}
|
return 1;
|
||||||
slot--;
|
|
||||||
|
|
||||||
if (next)
|
btrfs_release_path(root, path);
|
||||||
free_extent_buffer(next);
|
ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
|
||||||
|
if (ret < 0)
|
||||||
next = read_node_slot(root, c, slot);
|
return ret;
|
||||||
break;
|
btrfs_item_key(path->nodes[0], &found_key, 0);
|
||||||
}
|
ret = comp_keys(&found_key, &key);
|
||||||
path->slots[level] = slot;
|
if (ret < 0)
|
||||||
while(1) {
|
return 0;
|
||||||
level--;
|
return 1;
|
||||||
c = path->nodes[level];
|
|
||||||
free_extent_buffer(c);
|
|
||||||
slot = btrfs_header_nritems(next);
|
|
||||||
if (slot != 0)
|
|
||||||
slot--;
|
|
||||||
path->nodes[level] = next;
|
|
||||||
path->slots[level] = slot;
|
|
||||||
if (!level)
|
|
||||||
break;
|
|
||||||
next = read_node_slot(root, next, slot);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* walk up the tree as far as required to find the next leaf.
|
* search the tree again to find a leaf with greater keys
|
||||||
* returns 0 if it found something or 1 if there are no greater leaves.
|
* returns 0 if it found something or 1 if there are no greater leaves.
|
||||||
* returns < 0 on io errors.
|
* returns < 0 on io errors.
|
||||||
*/
|
*/
|
||||||
@ -2790,6 +2946,28 @@ int btrfs_next_leaf(struct btrfs_root *root, struct btrfs_path *path)
|
|||||||
int level = 1;
|
int level = 1;
|
||||||
struct extent_buffer *c;
|
struct extent_buffer *c;
|
||||||
struct extent_buffer *next = NULL;
|
struct extent_buffer *next = NULL;
|
||||||
|
struct btrfs_key key;
|
||||||
|
u32 nritems;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
nritems = btrfs_header_nritems(path->nodes[0]);
|
||||||
|
if (nritems == 0) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
btrfs_item_key_to_cpu(path->nodes[0], &key, nritems - 1);
|
||||||
|
|
||||||
|
path->keep_locks = 1;
|
||||||
|
btrfs_release_path(root, path);
|
||||||
|
ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
|
||||||
|
path->keep_locks = 0;
|
||||||
|
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (path->slots[0] < nritems - 1) {
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
while(level < BTRFS_MAX_LEVEL) {
|
while(level < BTRFS_MAX_LEVEL) {
|
||||||
if (!path->nodes[level])
|
if (!path->nodes[level])
|
||||||
@ -2799,33 +2977,45 @@ int btrfs_next_leaf(struct btrfs_root *root, struct btrfs_path *path)
|
|||||||
c = path->nodes[level];
|
c = path->nodes[level];
|
||||||
if (slot >= btrfs_header_nritems(c)) {
|
if (slot >= btrfs_header_nritems(c)) {
|
||||||
level++;
|
level++;
|
||||||
if (level == BTRFS_MAX_LEVEL)
|
if (level == BTRFS_MAX_LEVEL) {
|
||||||
return 1;
|
return 1;
|
||||||
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (next)
|
if (next) {
|
||||||
|
btrfs_tree_unlock(next);
|
||||||
free_extent_buffer(next);
|
free_extent_buffer(next);
|
||||||
|
}
|
||||||
|
|
||||||
if (path->reada)
|
if (level == 1 && path->locks[1] && path->reada)
|
||||||
reada_for_search(root, path, level, slot, 0);
|
reada_for_search(root, path, level, slot, 0);
|
||||||
|
|
||||||
next = read_node_slot(root, c, slot);
|
next = read_node_slot(root, c, slot);
|
||||||
|
if (!path->skip_locking)
|
||||||
|
btrfs_tree_lock(next);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
path->slots[level] = slot;
|
path->slots[level] = slot;
|
||||||
while(1) {
|
while(1) {
|
||||||
level--;
|
level--;
|
||||||
c = path->nodes[level];
|
c = path->nodes[level];
|
||||||
|
if (path->locks[level])
|
||||||
|
btrfs_tree_unlock(c);
|
||||||
free_extent_buffer(c);
|
free_extent_buffer(c);
|
||||||
path->nodes[level] = next;
|
path->nodes[level] = next;
|
||||||
path->slots[level] = 0;
|
path->slots[level] = 0;
|
||||||
|
path->locks[level] = 1;
|
||||||
if (!level)
|
if (!level)
|
||||||
break;
|
break;
|
||||||
if (path->reada)
|
if (level == 1 && path->locks[1] && path->reada)
|
||||||
reada_for_search(root, path, level, 0, 0);
|
reada_for_search(root, path, level, slot, 0);
|
||||||
next = read_node_slot(root, next, 0);
|
next = read_node_slot(root, next, 0);
|
||||||
|
if (!path->skip_locking)
|
||||||
|
btrfs_tree_lock(next);
|
||||||
}
|
}
|
||||||
|
done:
|
||||||
|
unlock_up(path, 0, 1);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -330,8 +330,13 @@ struct btrfs_node {
|
|||||||
struct btrfs_path {
|
struct btrfs_path {
|
||||||
struct extent_buffer *nodes[BTRFS_MAX_LEVEL];
|
struct extent_buffer *nodes[BTRFS_MAX_LEVEL];
|
||||||
int slots[BTRFS_MAX_LEVEL];
|
int slots[BTRFS_MAX_LEVEL];
|
||||||
|
/* if there is real range locking, this locks field will change */
|
||||||
|
int locks[BTRFS_MAX_LEVEL];
|
||||||
int reada;
|
int reada;
|
||||||
|
/* keep some upper locks as we walk down */
|
||||||
|
int keep_locks;
|
||||||
int lowest_level;
|
int lowest_level;
|
||||||
|
int skip_locking;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -515,6 +520,8 @@ struct btrfs_fs_info {
|
|||||||
spinlock_t hash_lock;
|
spinlock_t hash_lock;
|
||||||
struct mutex trans_mutex;
|
struct mutex trans_mutex;
|
||||||
struct mutex fs_mutex;
|
struct mutex fs_mutex;
|
||||||
|
struct mutex alloc_mutex;
|
||||||
|
struct mutex chunk_mutex;
|
||||||
struct list_head trans_list;
|
struct list_head trans_list;
|
||||||
struct list_head hashers;
|
struct list_head hashers;
|
||||||
struct list_head dead_roots;
|
struct list_head dead_roots;
|
||||||
@ -576,6 +583,10 @@ struct btrfs_fs_info {
|
|||||||
*/
|
*/
|
||||||
struct btrfs_root {
|
struct btrfs_root {
|
||||||
struct extent_buffer *node;
|
struct extent_buffer *node;
|
||||||
|
|
||||||
|
/* the node lock is held while changing the node pointer */
|
||||||
|
spinlock_t node_lock;
|
||||||
|
|
||||||
struct extent_buffer *commit_root;
|
struct extent_buffer *commit_root;
|
||||||
struct btrfs_root_item root_item;
|
struct btrfs_root_item root_item;
|
||||||
struct btrfs_key root_key;
|
struct btrfs_key root_key;
|
||||||
@ -1353,13 +1364,7 @@ struct btrfs_block_group_cache *btrfs_find_block_group(struct btrfs_root *root,
|
|||||||
struct btrfs_block_group_cache
|
struct btrfs_block_group_cache
|
||||||
*hint, u64 search_start,
|
*hint, u64 search_start,
|
||||||
int data, int owner);
|
int data, int owner);
|
||||||
int btrfs_inc_root_ref(struct btrfs_trans_handle *trans,
|
|
||||||
struct btrfs_root *root, u64 owner_objectid);
|
|
||||||
struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root, u32 size,
|
|
||||||
u64 root_objectid,
|
|
||||||
u64 hint, u64 empty_size);
|
|
||||||
struct extent_buffer *__btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
|
||||||
struct btrfs_root *root,
|
struct btrfs_root *root,
|
||||||
u32 blocksize,
|
u32 blocksize,
|
||||||
u64 root_objectid,
|
u64 root_objectid,
|
||||||
@ -1368,8 +1373,6 @@ struct extent_buffer *__btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
|||||||
int level,
|
int level,
|
||||||
u64 hint,
|
u64 hint,
|
||||||
u64 empty_size);
|
u64 empty_size);
|
||||||
int btrfs_grow_extent_tree(struct btrfs_trans_handle *trans,
|
|
||||||
struct btrfs_root *root, u64 new_size);
|
|
||||||
int btrfs_shrink_extent_tree(struct btrfs_root *root, u64 new_size);
|
int btrfs_shrink_extent_tree(struct btrfs_root *root, u64 new_size);
|
||||||
int btrfs_insert_extent_backref(struct btrfs_trans_handle *trans,
|
int btrfs_insert_extent_backref(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root,
|
struct btrfs_root *root,
|
||||||
@ -1409,6 +1412,10 @@ int btrfs_make_block_group(struct btrfs_trans_handle *trans,
|
|||||||
int btrfs_previous_item(struct btrfs_root *root,
|
int btrfs_previous_item(struct btrfs_root *root,
|
||||||
struct btrfs_path *path, u64 min_objectid,
|
struct btrfs_path *path, u64 min_objectid,
|
||||||
int type);
|
int type);
|
||||||
|
|
||||||
|
struct extent_buffer *btrfs_root_node(struct btrfs_root *root);
|
||||||
|
struct extent_buffer *btrfs_lock_root_node(struct btrfs_root *root);
|
||||||
|
|
||||||
int btrfs_cow_block(struct btrfs_trans_handle *trans,
|
int btrfs_cow_block(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root, struct extent_buffer *buf,
|
struct btrfs_root *root, struct extent_buffer *buf,
|
||||||
struct extent_buffer *parent, int parent_slot,
|
struct extent_buffer *parent, int parent_slot,
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
#include "volumes.h"
|
#include "volumes.h"
|
||||||
#include "print-tree.h"
|
#include "print-tree.h"
|
||||||
#include "async-thread.h"
|
#include "async-thread.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
static int check_tree_block(struct btrfs_root *root, struct extent_buffer *buf)
|
static int check_tree_block(struct btrfs_root *root, struct extent_buffer *buf)
|
||||||
@ -681,9 +682,11 @@ int clean_tree_block(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
{
|
{
|
||||||
struct inode *btree_inode = root->fs_info->btree_inode;
|
struct inode *btree_inode = root->fs_info->btree_inode;
|
||||||
if (btrfs_header_generation(buf) ==
|
if (btrfs_header_generation(buf) ==
|
||||||
root->fs_info->running_transaction->transid)
|
root->fs_info->running_transaction->transid) {
|
||||||
|
WARN_ON(!btrfs_tree_locked(buf));
|
||||||
clear_extent_buffer_dirty(&BTRFS_I(btree_inode)->io_tree,
|
clear_extent_buffer_dirty(&BTRFS_I(btree_inode)->io_tree,
|
||||||
buf);
|
buf);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -720,6 +723,7 @@ static int __setup_root(u32 nodesize, u32 leafsize, u32 sectorsize,
|
|||||||
root->in_sysfs = 0;
|
root->in_sysfs = 0;
|
||||||
|
|
||||||
INIT_LIST_HEAD(&root->dirty_list);
|
INIT_LIST_HEAD(&root->dirty_list);
|
||||||
|
spin_lock_init(&root->node_lock);
|
||||||
memset(&root->root_key, 0, sizeof(root->root_key));
|
memset(&root->root_key, 0, sizeof(root->root_key));
|
||||||
memset(&root->root_item, 0, sizeof(root->root_item));
|
memset(&root->root_item, 0, sizeof(root->root_item));
|
||||||
memset(&root->defrag_progress, 0, sizeof(root->defrag_progress));
|
memset(&root->defrag_progress, 0, sizeof(root->defrag_progress));
|
||||||
@ -1196,6 +1200,8 @@ struct btrfs_root *open_ctree(struct super_block *sb,
|
|||||||
|
|
||||||
mutex_init(&fs_info->trans_mutex);
|
mutex_init(&fs_info->trans_mutex);
|
||||||
mutex_init(&fs_info->fs_mutex);
|
mutex_init(&fs_info->fs_mutex);
|
||||||
|
mutex_init(&fs_info->alloc_mutex);
|
||||||
|
mutex_init(&fs_info->chunk_mutex);
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
ret = add_hasher(fs_info, "crc32c");
|
ret = add_hasher(fs_info, "crc32c");
|
||||||
@ -1274,7 +1280,9 @@ struct btrfs_root *open_ctree(struct super_block *sb,
|
|||||||
|
|
||||||
mutex_lock(&fs_info->fs_mutex);
|
mutex_lock(&fs_info->fs_mutex);
|
||||||
|
|
||||||
|
mutex_lock(&fs_info->chunk_mutex);
|
||||||
ret = btrfs_read_sys_array(tree_root);
|
ret = btrfs_read_sys_array(tree_root);
|
||||||
|
mutex_unlock(&fs_info->chunk_mutex);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
printk("btrfs: failed to read the system array on %s\n",
|
printk("btrfs: failed to read the system array on %s\n",
|
||||||
sb->s_id);
|
sb->s_id);
|
||||||
@ -1296,7 +1304,9 @@ struct btrfs_root *open_ctree(struct super_block *sb,
|
|||||||
(unsigned long)btrfs_header_chunk_tree_uuid(chunk_root->node),
|
(unsigned long)btrfs_header_chunk_tree_uuid(chunk_root->node),
|
||||||
BTRFS_UUID_SIZE);
|
BTRFS_UUID_SIZE);
|
||||||
|
|
||||||
|
mutex_lock(&fs_info->chunk_mutex);
|
||||||
ret = btrfs_read_chunk_tree(chunk_root);
|
ret = btrfs_read_chunk_tree(chunk_root);
|
||||||
|
mutex_unlock(&fs_info->chunk_mutex);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
|
||||||
btrfs_close_extra_devices(fs_devices);
|
btrfs_close_extra_devices(fs_devices);
|
||||||
@ -1654,6 +1664,7 @@ void btrfs_mark_buffer_dirty(struct extent_buffer *buf)
|
|||||||
u64 transid = btrfs_header_generation(buf);
|
u64 transid = btrfs_header_generation(buf);
|
||||||
struct inode *btree_inode = root->fs_info->btree_inode;
|
struct inode *btree_inode = root->fs_info->btree_inode;
|
||||||
|
|
||||||
|
WARN_ON(!btrfs_tree_locked(buf));
|
||||||
if (transid != root->fs_info->generation) {
|
if (transid != root->fs_info->generation) {
|
||||||
printk(KERN_CRIT "transid mismatch buffer %llu, found %Lu running %Lu\n",
|
printk(KERN_CRIT "transid mismatch buffer %llu, found %Lu running %Lu\n",
|
||||||
(unsigned long long)buf->start,
|
(unsigned long long)buf->start,
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
#include "print-tree.h"
|
#include "print-tree.h"
|
||||||
#include "transaction.h"
|
#include "transaction.h"
|
||||||
#include "volumes.h"
|
#include "volumes.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
#define BLOCK_GROUP_DATA EXTENT_WRITEBACK
|
#define BLOCK_GROUP_DATA EXTENT_WRITEBACK
|
||||||
#define BLOCK_GROUP_METADATA EXTENT_UPTODATE
|
#define BLOCK_GROUP_METADATA EXTENT_UPTODATE
|
||||||
@ -36,7 +37,28 @@ static int finish_current_insert(struct btrfs_trans_handle *trans, struct
|
|||||||
btrfs_root *extent_root);
|
btrfs_root *extent_root);
|
||||||
static int del_pending_extents(struct btrfs_trans_handle *trans, struct
|
static int del_pending_extents(struct btrfs_trans_handle *trans, struct
|
||||||
btrfs_root *extent_root);
|
btrfs_root *extent_root);
|
||||||
|
static struct btrfs_block_group_cache *
|
||||||
|
__btrfs_find_block_group(struct btrfs_root *root,
|
||||||
|
struct btrfs_block_group_cache *hint,
|
||||||
|
u64 search_start, int data, int owner);
|
||||||
|
|
||||||
|
void maybe_lock_mutex(struct btrfs_root *root)
|
||||||
|
{
|
||||||
|
if (root != root->fs_info->extent_root &&
|
||||||
|
root != root->fs_info->chunk_root &&
|
||||||
|
root != root->fs_info->dev_root) {
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void maybe_unlock_mutex(struct btrfs_root *root)
|
||||||
|
{
|
||||||
|
if (root != root->fs_info->extent_root &&
|
||||||
|
root != root->fs_info->chunk_root &&
|
||||||
|
root != root->fs_info->dev_root) {
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int cache_block_group(struct btrfs_root *root,
|
static int cache_block_group(struct btrfs_root *root,
|
||||||
struct btrfs_block_group_cache *block_group)
|
struct btrfs_block_group_cache *block_group)
|
||||||
@ -66,6 +88,7 @@ static int cache_block_group(struct btrfs_root *root,
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
path->reada = 2;
|
path->reada = 2;
|
||||||
|
path->skip_locking = 1;
|
||||||
first_free = block_group->key.objectid;
|
first_free = block_group->key.objectid;
|
||||||
key.objectid = block_group->key.objectid;
|
key.objectid = block_group->key.objectid;
|
||||||
key.offset = 0;
|
key.offset = 0;
|
||||||
@ -290,7 +313,7 @@ no_cache:
|
|||||||
cache = btrfs_lookup_first_block_group(root->fs_info, last);
|
cache = btrfs_lookup_first_block_group(root->fs_info, last);
|
||||||
}
|
}
|
||||||
cache_miss = 0;
|
cache_miss = 0;
|
||||||
cache = btrfs_find_block_group(root, cache, last, data, 0);
|
cache = __btrfs_find_block_group(root, cache, last, data, 0);
|
||||||
if (!cache)
|
if (!cache)
|
||||||
goto no_cache;
|
goto no_cache;
|
||||||
*cache_ret = cache;
|
*cache_ret = cache;
|
||||||
@ -318,10 +341,10 @@ static int block_group_state_bits(u64 flags)
|
|||||||
return bits;
|
return bits;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct btrfs_block_group_cache *btrfs_find_block_group(struct btrfs_root *root,
|
static struct btrfs_block_group_cache *
|
||||||
struct btrfs_block_group_cache
|
__btrfs_find_block_group(struct btrfs_root *root,
|
||||||
*hint, u64 search_start,
|
struct btrfs_block_group_cache *hint,
|
||||||
int data, int owner)
|
u64 search_start, int data, int owner)
|
||||||
{
|
{
|
||||||
struct btrfs_block_group_cache *cache;
|
struct btrfs_block_group_cache *cache;
|
||||||
struct extent_io_tree *block_group_cache;
|
struct extent_io_tree *block_group_cache;
|
||||||
@ -411,6 +434,18 @@ found:
|
|||||||
return found_group;
|
return found_group;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct btrfs_block_group_cache *btrfs_find_block_group(struct btrfs_root *root,
|
||||||
|
struct btrfs_block_group_cache
|
||||||
|
*hint, u64 search_start,
|
||||||
|
int data, int owner)
|
||||||
|
{
|
||||||
|
|
||||||
|
struct btrfs_block_group_cache *ret;
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
|
ret = __btrfs_find_block_group(root, hint, search_start, data, owner);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
static u64 hash_extent_ref(u64 root_objectid, u64 ref_generation,
|
static u64 hash_extent_ref(u64 root_objectid, u64 ref_generation,
|
||||||
u64 owner, u64 owner_offset)
|
u64 owner, u64 owner_offset)
|
||||||
{
|
{
|
||||||
@ -646,7 +681,7 @@ out:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
|
static int __btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root,
|
struct btrfs_root *root,
|
||||||
u64 bytenr, u64 num_bytes,
|
u64 bytenr, u64 num_bytes,
|
||||||
u64 root_objectid, u64 ref_generation,
|
u64 root_objectid, u64 ref_generation,
|
||||||
@ -696,6 +731,22 @@ int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
|
||||||
|
struct btrfs_root *root,
|
||||||
|
u64 bytenr, u64 num_bytes,
|
||||||
|
u64 root_objectid, u64 ref_generation,
|
||||||
|
u64 owner, u64 owner_offset)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
|
ret = __btrfs_inc_extent_ref(trans, root, bytenr, num_bytes,
|
||||||
|
root_objectid, ref_generation,
|
||||||
|
owner, owner_offset);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
int btrfs_extent_post_op(struct btrfs_trans_handle *trans,
|
int btrfs_extent_post_op(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root)
|
struct btrfs_root *root)
|
||||||
{
|
{
|
||||||
@ -760,6 +811,10 @@ u32 btrfs_count_snapshots_in_path(struct btrfs_root *root,
|
|||||||
struct btrfs_extent_ref *ref_item;
|
struct btrfs_extent_ref *ref_item;
|
||||||
int level = -1;
|
int level = -1;
|
||||||
|
|
||||||
|
/* FIXME, needs locking */
|
||||||
|
BUG();
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
path = btrfs_alloc_path();
|
path = btrfs_alloc_path();
|
||||||
again:
|
again:
|
||||||
if (level == -1)
|
if (level == -1)
|
||||||
@ -854,33 +909,9 @@ again:
|
|||||||
|
|
||||||
out:
|
out:
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return total_count;
|
return total_count;
|
||||||
}
|
}
|
||||||
int btrfs_inc_root_ref(struct btrfs_trans_handle *trans,
|
|
||||||
struct btrfs_root *root, u64 owner_objectid)
|
|
||||||
{
|
|
||||||
u64 generation;
|
|
||||||
u64 key_objectid;
|
|
||||||
u64 level;
|
|
||||||
u32 nritems;
|
|
||||||
struct btrfs_disk_key disk_key;
|
|
||||||
|
|
||||||
level = btrfs_header_level(root->node);
|
|
||||||
generation = trans->transid;
|
|
||||||
nritems = btrfs_header_nritems(root->node);
|
|
||||||
if (nritems > 0) {
|
|
||||||
if (level == 0)
|
|
||||||
btrfs_item_key(root->node, &disk_key, 0);
|
|
||||||
else
|
|
||||||
btrfs_node_key(root->node, &disk_key, 0);
|
|
||||||
key_objectid = btrfs_disk_key_objectid(&disk_key);
|
|
||||||
} else {
|
|
||||||
key_objectid = 0;
|
|
||||||
}
|
|
||||||
return btrfs_inc_extent_ref(trans, root, root->node->start,
|
|
||||||
root->node->len, owner_objectid,
|
|
||||||
generation, level, key_objectid);
|
|
||||||
}
|
|
||||||
|
|
||||||
int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
||||||
struct extent_buffer *buf)
|
struct extent_buffer *buf)
|
||||||
@ -897,6 +928,7 @@ int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
if (!root->ref_cows)
|
if (!root->ref_cows)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
level = btrfs_header_level(buf);
|
level = btrfs_header_level(buf);
|
||||||
nritems = btrfs_header_nritems(buf);
|
nritems = btrfs_header_nritems(buf);
|
||||||
for (i = 0; i < nritems; i++) {
|
for (i = 0; i < nritems; i++) {
|
||||||
@ -913,7 +945,7 @@ int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
disk_bytenr = btrfs_file_extent_disk_bytenr(buf, fi);
|
disk_bytenr = btrfs_file_extent_disk_bytenr(buf, fi);
|
||||||
if (disk_bytenr == 0)
|
if (disk_bytenr == 0)
|
||||||
continue;
|
continue;
|
||||||
ret = btrfs_inc_extent_ref(trans, root, disk_bytenr,
|
ret = __btrfs_inc_extent_ref(trans, root, disk_bytenr,
|
||||||
btrfs_file_extent_disk_num_bytes(buf, fi),
|
btrfs_file_extent_disk_num_bytes(buf, fi),
|
||||||
root->root_key.objectid, trans->transid,
|
root->root_key.objectid, trans->transid,
|
||||||
key.objectid, key.offset);
|
key.objectid, key.offset);
|
||||||
@ -924,7 +956,7 @@ int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
} else {
|
} else {
|
||||||
bytenr = btrfs_node_blockptr(buf, i);
|
bytenr = btrfs_node_blockptr(buf, i);
|
||||||
btrfs_node_key_to_cpu(buf, &key, i);
|
btrfs_node_key_to_cpu(buf, &key, i);
|
||||||
ret = btrfs_inc_extent_ref(trans, root, bytenr,
|
ret = __btrfs_inc_extent_ref(trans, root, bytenr,
|
||||||
btrfs_level_size(root, level - 1),
|
btrfs_level_size(root, level - 1),
|
||||||
root->root_key.objectid,
|
root->root_key.objectid,
|
||||||
trans->transid,
|
trans->transid,
|
||||||
@ -935,6 +967,7 @@ int btrfs_inc_ref(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return 0;
|
return 0;
|
||||||
fail:
|
fail:
|
||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
@ -965,6 +998,7 @@ fail:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1019,6 +1053,7 @@ int btrfs_write_dirty_block_groups(struct btrfs_trans_handle *trans,
|
|||||||
if (!path)
|
if (!path)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_extent_bit(block_group_cache, last,
|
ret = find_first_extent_bit(block_group_cache, last,
|
||||||
&start, &end, BLOCK_GROUP_DIRTY);
|
&start, &end, BLOCK_GROUP_DIRTY);
|
||||||
@ -1045,6 +1080,7 @@ int btrfs_write_dirty_block_groups(struct btrfs_trans_handle *trans,
|
|||||||
BLOCK_GROUP_DIRTY, GFP_NOFS);
|
BLOCK_GROUP_DIRTY, GFP_NOFS);
|
||||||
}
|
}
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return werr;
|
return werr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1162,26 +1198,28 @@ static int do_chunk_alloc(struct btrfs_trans_handle *trans,
|
|||||||
space_info->force_alloc = 0;
|
space_info->force_alloc = 0;
|
||||||
}
|
}
|
||||||
if (space_info->full)
|
if (space_info->full)
|
||||||
return 0;
|
goto out;
|
||||||
|
|
||||||
thresh = div_factor(space_info->total_bytes, 6);
|
thresh = div_factor(space_info->total_bytes, 6);
|
||||||
if (!force &&
|
if (!force &&
|
||||||
(space_info->bytes_used + space_info->bytes_pinned + alloc_bytes) <
|
(space_info->bytes_used + space_info->bytes_pinned + alloc_bytes) <
|
||||||
thresh)
|
thresh)
|
||||||
return 0;
|
goto out;
|
||||||
|
|
||||||
|
mutex_lock(&extent_root->fs_info->chunk_mutex);
|
||||||
ret = btrfs_alloc_chunk(trans, extent_root, &start, &num_bytes, flags);
|
ret = btrfs_alloc_chunk(trans, extent_root, &start, &num_bytes, flags);
|
||||||
if (ret == -ENOSPC) {
|
if (ret == -ENOSPC) {
|
||||||
printk("space info full %Lu\n", flags);
|
printk("space info full %Lu\n", flags);
|
||||||
space_info->full = 1;
|
space_info->full = 1;
|
||||||
return 0;
|
goto out;
|
||||||
}
|
}
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
|
||||||
ret = btrfs_make_block_group(trans, extent_root, 0, flags,
|
ret = btrfs_make_block_group(trans, extent_root, 0, flags,
|
||||||
BTRFS_FIRST_CHUNK_TREE_OBJECTID, start, num_bytes);
|
BTRFS_FIRST_CHUNK_TREE_OBJECTID, start, num_bytes);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
mutex_unlock(&extent_root->fs_info->chunk_mutex);
|
||||||
|
out:
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1318,6 +1356,7 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
|
|||||||
struct extent_io_tree *free_space_cache;
|
struct extent_io_tree *free_space_cache;
|
||||||
free_space_cache = &root->fs_info->free_space_cache;
|
free_space_cache = &root->fs_info->free_space_cache;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_extent_bit(unpin, 0, &start, &end,
|
ret = find_first_extent_bit(unpin, 0, &start, &end,
|
||||||
EXTENT_DIRTY);
|
EXTENT_DIRTY);
|
||||||
@ -1327,6 +1366,7 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
|
|||||||
clear_extent_dirty(unpin, start, end, GFP_NOFS);
|
clear_extent_dirty(unpin, start, end, GFP_NOFS);
|
||||||
set_extent_dirty(free_space_cache, start, end, GFP_NOFS);
|
set_extent_dirty(free_space_cache, start, end, GFP_NOFS);
|
||||||
}
|
}
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1363,18 +1403,24 @@ static int finish_current_insert(struct btrfs_trans_handle *trans,
|
|||||||
GFP_NOFS);
|
GFP_NOFS);
|
||||||
eb = read_tree_block(extent_root, ins.objectid, ins.offset,
|
eb = read_tree_block(extent_root, ins.objectid, ins.offset,
|
||||||
trans->transid);
|
trans->transid);
|
||||||
|
btrfs_tree_lock(eb);
|
||||||
level = btrfs_header_level(eb);
|
level = btrfs_header_level(eb);
|
||||||
if (level == 0) {
|
if (level == 0) {
|
||||||
btrfs_item_key(eb, &first, 0);
|
btrfs_item_key(eb, &first, 0);
|
||||||
} else {
|
} else {
|
||||||
btrfs_node_key(eb, &first, 0);
|
btrfs_node_key(eb, &first, 0);
|
||||||
}
|
}
|
||||||
|
btrfs_tree_unlock(eb);
|
||||||
|
free_extent_buffer(eb);
|
||||||
|
/*
|
||||||
|
* the first key is just a hint, so the race we've created
|
||||||
|
* against reading it is fine
|
||||||
|
*/
|
||||||
err = btrfs_insert_extent_backref(trans, extent_root, path,
|
err = btrfs_insert_extent_backref(trans, extent_root, path,
|
||||||
start, extent_root->root_key.objectid,
|
start, extent_root->root_key.objectid,
|
||||||
0, level,
|
0, level,
|
||||||
btrfs_disk_key_objectid(&first));
|
btrfs_disk_key_objectid(&first));
|
||||||
BUG_ON(err);
|
BUG_ON(err);
|
||||||
free_extent_buffer(eb);
|
|
||||||
}
|
}
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
return 0;
|
return 0;
|
||||||
@ -1384,12 +1430,14 @@ static int pin_down_bytes(struct btrfs_root *root, u64 bytenr, u32 num_bytes,
|
|||||||
int pending)
|
int pending)
|
||||||
{
|
{
|
||||||
int err = 0;
|
int err = 0;
|
||||||
struct extent_buffer *buf;
|
|
||||||
|
|
||||||
if (!pending) {
|
if (!pending) {
|
||||||
|
#if 0
|
||||||
|
struct extent_buffer *buf;
|
||||||
buf = btrfs_find_tree_block(root, bytenr, num_bytes);
|
buf = btrfs_find_tree_block(root, bytenr, num_bytes);
|
||||||
if (buf) {
|
if (buf) {
|
||||||
if (btrfs_buffer_uptodate(buf, 0)) {
|
if (!btrfs_try_tree_lock(buf) &&
|
||||||
|
btrfs_buffer_uptodate(buf, 0)) {
|
||||||
u64 transid =
|
u64 transid =
|
||||||
root->fs_info->running_transaction->transid;
|
root->fs_info->running_transaction->transid;
|
||||||
u64 header_transid =
|
u64 header_transid =
|
||||||
@ -1398,12 +1446,15 @@ static int pin_down_bytes(struct btrfs_root *root, u64 bytenr, u32 num_bytes,
|
|||||||
!btrfs_header_flag(buf,
|
!btrfs_header_flag(buf,
|
||||||
BTRFS_HEADER_FLAG_WRITTEN)) {
|
BTRFS_HEADER_FLAG_WRITTEN)) {
|
||||||
clean_tree_block(NULL, root, buf);
|
clean_tree_block(NULL, root, buf);
|
||||||
|
btrfs_tree_unlock(buf);
|
||||||
free_extent_buffer(buf);
|
free_extent_buffer(buf);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
btrfs_tree_unlock(buf);
|
||||||
}
|
}
|
||||||
free_extent_buffer(buf);
|
free_extent_buffer(buf);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
update_pinned_extents(root, bytenr, num_bytes, 1);
|
update_pinned_extents(root, bytenr, num_bytes, 1);
|
||||||
} else {
|
} else {
|
||||||
set_extent_bits(&root->fs_info->pending_del,
|
set_extent_bits(&root->fs_info->pending_del,
|
||||||
@ -1586,10 +1637,11 @@ static int del_pending_extents(struct btrfs_trans_handle *trans, struct
|
|||||||
/*
|
/*
|
||||||
* remove an extent from the root, returns 0 on success
|
* remove an extent from the root, returns 0 on success
|
||||||
*/
|
*/
|
||||||
int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
|
static int __btrfs_free_extent(struct btrfs_trans_handle *trans,
|
||||||
*root, u64 bytenr, u64 num_bytes,
|
struct btrfs_root *root, u64 bytenr,
|
||||||
u64 root_objectid, u64 ref_generation,
|
u64 num_bytes, u64 root_objectid,
|
||||||
u64 owner_objectid, u64 owner_offset, int pin)
|
u64 ref_generation, u64 owner_objectid,
|
||||||
|
u64 owner_offset, int pin)
|
||||||
{
|
{
|
||||||
struct btrfs_root *extent_root = root->fs_info->extent_root;
|
struct btrfs_root *extent_root = root->fs_info->extent_root;
|
||||||
int pending_ret;
|
int pending_ret;
|
||||||
@ -1610,6 +1662,22 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
return ret ? ret : pending_ret;
|
return ret ? ret : pending_ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int btrfs_free_extent(struct btrfs_trans_handle *trans,
|
||||||
|
struct btrfs_root *root, u64 bytenr,
|
||||||
|
u64 num_bytes, u64 root_objectid,
|
||||||
|
u64 ref_generation, u64 owner_objectid,
|
||||||
|
u64 owner_offset, int pin)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
maybe_lock_mutex(root);
|
||||||
|
ret = __btrfs_free_extent(trans, root, bytenr, num_bytes,
|
||||||
|
root_objectid, ref_generation,
|
||||||
|
owner_objectid, owner_offset, pin);
|
||||||
|
maybe_unlock_mutex(root);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static u64 stripe_align(struct btrfs_root *root, u64 val)
|
static u64 stripe_align(struct btrfs_root *root, u64 val)
|
||||||
{
|
{
|
||||||
u64 mask = ((u64)root->stripesize - 1);
|
u64 mask = ((u64)root->stripesize - 1);
|
||||||
@ -1679,12 +1747,12 @@ static int noinline find_free_extent(struct btrfs_trans_handle *trans,
|
|||||||
block_group = btrfs_lookup_first_block_group(info, hint_byte);
|
block_group = btrfs_lookup_first_block_group(info, hint_byte);
|
||||||
if (!block_group)
|
if (!block_group)
|
||||||
hint_byte = search_start;
|
hint_byte = search_start;
|
||||||
block_group = btrfs_find_block_group(root, block_group,
|
block_group = __btrfs_find_block_group(root, block_group,
|
||||||
hint_byte, data, 1);
|
hint_byte, data, 1);
|
||||||
if (last_ptr && *last_ptr == 0 && block_group)
|
if (last_ptr && *last_ptr == 0 && block_group)
|
||||||
hint_byte = block_group->key.objectid;
|
hint_byte = block_group->key.objectid;
|
||||||
} else {
|
} else {
|
||||||
block_group = btrfs_find_block_group(root,
|
block_group = __btrfs_find_block_group(root,
|
||||||
trans->block_group,
|
trans->block_group,
|
||||||
search_start, data, 1);
|
search_start, data, 1);
|
||||||
}
|
}
|
||||||
@ -1806,7 +1874,7 @@ enospc:
|
|||||||
}
|
}
|
||||||
block_group = btrfs_lookup_first_block_group(info, search_start);
|
block_group = btrfs_lookup_first_block_group(info, search_start);
|
||||||
cond_resched();
|
cond_resched();
|
||||||
block_group = btrfs_find_block_group(root, block_group,
|
block_group = __btrfs_find_block_group(root, block_group,
|
||||||
search_start, data, 0);
|
search_start, data, 0);
|
||||||
goto check_failed;
|
goto check_failed;
|
||||||
|
|
||||||
@ -1843,6 +1911,8 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
|
|||||||
struct btrfs_path *path;
|
struct btrfs_path *path;
|
||||||
struct btrfs_key keys[2];
|
struct btrfs_key keys[2];
|
||||||
|
|
||||||
|
maybe_lock_mutex(root);
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
alloc_profile = info->avail_data_alloc_bits &
|
alloc_profile = info->avail_data_alloc_bits &
|
||||||
info->data_alloc_profile;
|
info->data_alloc_profile;
|
||||||
@ -1892,9 +1962,10 @@ again:
|
|||||||
if (ret) {
|
if (ret) {
|
||||||
printk("allocation failed flags %Lu\n", data);
|
printk("allocation failed flags %Lu\n", data);
|
||||||
}
|
}
|
||||||
BUG_ON(ret);
|
if (ret) {
|
||||||
if (ret)
|
BUG();
|
||||||
return ret;
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
/* block accounting for super block */
|
/* block accounting for super block */
|
||||||
super_used = btrfs_super_bytes_used(&info->super_copy);
|
super_used = btrfs_super_bytes_used(&info->super_copy);
|
||||||
@ -1953,11 +2024,11 @@ again:
|
|||||||
finish_current_insert(trans, extent_root);
|
finish_current_insert(trans, extent_root);
|
||||||
pending_ret = del_pending_extents(trans, extent_root);
|
pending_ret = del_pending_extents(trans, extent_root);
|
||||||
|
|
||||||
if (ret) {
|
if (ret)
|
||||||
return ret;
|
goto out;
|
||||||
}
|
|
||||||
if (pending_ret) {
|
if (pending_ret) {
|
||||||
return pending_ret;
|
ret = pending_ret;
|
||||||
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
update_block:
|
update_block:
|
||||||
@ -1967,36 +2038,15 @@ update_block:
|
|||||||
ins->objectid, ins->offset);
|
ins->objectid, ins->offset);
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
return 0;
|
out:
|
||||||
|
maybe_unlock_mutex(root);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* helper function to allocate a block for a given tree
|
* helper function to allocate a block for a given tree
|
||||||
* returns the tree buffer or NULL.
|
* returns the tree buffer or NULL.
|
||||||
*/
|
*/
|
||||||
struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_root *root,
|
|
||||||
u32 blocksize,
|
|
||||||
u64 root_objectid, u64 hint,
|
|
||||||
u64 empty_size)
|
|
||||||
{
|
|
||||||
u64 ref_generation;
|
|
||||||
|
|
||||||
if (root->ref_cows)
|
|
||||||
ref_generation = trans->transid;
|
|
||||||
else
|
|
||||||
ref_generation = 0;
|
|
||||||
|
|
||||||
|
|
||||||
return __btrfs_alloc_free_block(trans, root, blocksize, root_objectid,
|
|
||||||
ref_generation, 0, 0, hint, empty_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* helper function to allocate a block for a given tree
|
|
||||||
* returns the tree buffer or NULL.
|
|
||||||
*/
|
|
||||||
struct extent_buffer *__btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
|
||||||
struct btrfs_root *root,
|
struct btrfs_root *root,
|
||||||
u32 blocksize,
|
u32 blocksize,
|
||||||
u64 root_objectid,
|
u64 root_objectid,
|
||||||
@ -2026,6 +2076,7 @@ struct extent_buffer *__btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
|||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
btrfs_set_header_generation(buf, trans->transid);
|
btrfs_set_header_generation(buf, trans->transid);
|
||||||
|
btrfs_tree_lock(buf);
|
||||||
clean_tree_block(trans, root, buf);
|
clean_tree_block(trans, root, buf);
|
||||||
btrfs_set_buffer_uptodate(buf);
|
btrfs_set_buffer_uptodate(buf);
|
||||||
|
|
||||||
@ -2076,7 +2127,7 @@ static int noinline drop_leaf_ref(struct btrfs_trans_handle *trans,
|
|||||||
disk_bytenr = btrfs_file_extent_disk_bytenr(leaf, fi);
|
disk_bytenr = btrfs_file_extent_disk_bytenr(leaf, fi);
|
||||||
if (disk_bytenr == 0)
|
if (disk_bytenr == 0)
|
||||||
continue;
|
continue;
|
||||||
ret = btrfs_free_extent(trans, root, disk_bytenr,
|
ret = __btrfs_free_extent(trans, root, disk_bytenr,
|
||||||
btrfs_file_extent_disk_num_bytes(leaf, fi),
|
btrfs_file_extent_disk_num_bytes(leaf, fi),
|
||||||
leaf_owner, leaf_generation,
|
leaf_owner, leaf_generation,
|
||||||
key.objectid, key.offset, 0);
|
key.objectid, key.offset, 0);
|
||||||
@ -2151,6 +2202,8 @@ static int noinline walk_down_tree(struct btrfs_trans_handle *trans,
|
|||||||
int ret;
|
int ret;
|
||||||
u32 refs;
|
u32 refs;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
|
|
||||||
WARN_ON(*level < 0);
|
WARN_ON(*level < 0);
|
||||||
WARN_ON(*level >= BTRFS_MAX_LEVEL);
|
WARN_ON(*level >= BTRFS_MAX_LEVEL);
|
||||||
ret = lookup_extent_ref(trans, root,
|
ret = lookup_extent_ref(trans, root,
|
||||||
@ -2182,6 +2235,7 @@ static int noinline walk_down_tree(struct btrfs_trans_handle *trans,
|
|||||||
bytenr = btrfs_node_blockptr(cur, path->slots[*level]);
|
bytenr = btrfs_node_blockptr(cur, path->slots[*level]);
|
||||||
ptr_gen = btrfs_node_ptr_generation(cur, path->slots[*level]);
|
ptr_gen = btrfs_node_ptr_generation(cur, path->slots[*level]);
|
||||||
blocksize = btrfs_level_size(root, *level - 1);
|
blocksize = btrfs_level_size(root, *level - 1);
|
||||||
|
|
||||||
ret = lookup_extent_ref(trans, root, bytenr, blocksize, &refs);
|
ret = lookup_extent_ref(trans, root, bytenr, blocksize, &refs);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
if (refs != 1) {
|
if (refs != 1) {
|
||||||
@ -2189,7 +2243,7 @@ static int noinline walk_down_tree(struct btrfs_trans_handle *trans,
|
|||||||
root_owner = btrfs_header_owner(parent);
|
root_owner = btrfs_header_owner(parent);
|
||||||
root_gen = btrfs_header_generation(parent);
|
root_gen = btrfs_header_generation(parent);
|
||||||
path->slots[*level]++;
|
path->slots[*level]++;
|
||||||
ret = btrfs_free_extent(trans, root, bytenr,
|
ret = __btrfs_free_extent(trans, root, bytenr,
|
||||||
blocksize, root_owner,
|
blocksize, root_owner,
|
||||||
root_gen, 0, 0, 1);
|
root_gen, 0, 0, 1);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
@ -2201,9 +2255,11 @@ static int noinline walk_down_tree(struct btrfs_trans_handle *trans,
|
|||||||
reada_walk_down(root, cur, path->slots[*level]);
|
reada_walk_down(root, cur, path->slots[*level]);
|
||||||
|
|
||||||
mutex_unlock(&root->fs_info->fs_mutex);
|
mutex_unlock(&root->fs_info->fs_mutex);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
next = read_tree_block(root, bytenr, blocksize,
|
next = read_tree_block(root, bytenr, blocksize,
|
||||||
ptr_gen);
|
ptr_gen);
|
||||||
mutex_lock(&root->fs_info->fs_mutex);
|
mutex_lock(&root->fs_info->fs_mutex);
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
|
|
||||||
/* we've dropped the lock, double check */
|
/* we've dropped the lock, double check */
|
||||||
ret = lookup_extent_ref(trans, root, bytenr,
|
ret = lookup_extent_ref(trans, root, bytenr,
|
||||||
@ -2216,7 +2272,7 @@ static int noinline walk_down_tree(struct btrfs_trans_handle *trans,
|
|||||||
|
|
||||||
path->slots[*level]++;
|
path->slots[*level]++;
|
||||||
free_extent_buffer(next);
|
free_extent_buffer(next);
|
||||||
ret = btrfs_free_extent(trans, root, bytenr,
|
ret = __btrfs_free_extent(trans, root, bytenr,
|
||||||
blocksize,
|
blocksize,
|
||||||
root_owner,
|
root_owner,
|
||||||
root_gen, 0, 0, 1);
|
root_gen, 0, 0, 1);
|
||||||
@ -2244,13 +2300,14 @@ out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
root_gen = btrfs_header_generation(parent);
|
root_gen = btrfs_header_generation(parent);
|
||||||
ret = btrfs_free_extent(trans, root, path->nodes[*level]->start,
|
ret = __btrfs_free_extent(trans, root, path->nodes[*level]->start,
|
||||||
path->nodes[*level]->len,
|
path->nodes[*level]->len,
|
||||||
root_owner, root_gen, 0, 0, 1);
|
root_owner, root_gen, 0, 0, 1);
|
||||||
free_extent_buffer(path->nodes[*level]);
|
free_extent_buffer(path->nodes[*level]);
|
||||||
path->nodes[*level] = NULL;
|
path->nodes[*level] = NULL;
|
||||||
*level += 1;
|
*level += 1;
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2350,6 +2407,12 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
|
|||||||
btrfs_node_key(node, &found_key, path->slots[level]);
|
btrfs_node_key(node, &found_key, path->slots[level]);
|
||||||
WARN_ON(memcmp(&found_key, &root_item->drop_progress,
|
WARN_ON(memcmp(&found_key, &root_item->drop_progress,
|
||||||
sizeof(found_key)));
|
sizeof(found_key)));
|
||||||
|
for (i = 0; i < BTRFS_MAX_LEVEL; i++) {
|
||||||
|
if (path->nodes[i] && path->locks[i]) {
|
||||||
|
path->locks[i] = 0;
|
||||||
|
btrfs_tree_unlock(path->nodes[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
while(1) {
|
while(1) {
|
||||||
wret = walk_down_tree(trans, root, path, &level);
|
wret = walk_down_tree(trans, root, path, &level);
|
||||||
@ -2383,6 +2446,8 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
|
|||||||
u64 end;
|
u64 end;
|
||||||
u64 ptr;
|
u64 ptr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
mutex_lock(&info->alloc_mutex);
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_extent_bit(&info->block_group_cache, 0,
|
ret = find_first_extent_bit(&info->block_group_cache, 0,
|
||||||
&start, &end, (unsigned int)-1);
|
&start, &end, (unsigned int)-1);
|
||||||
@ -2402,6 +2467,7 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
|
|||||||
clear_extent_dirty(&info->free_space_cache, start,
|
clear_extent_dirty(&info->free_space_cache, start,
|
||||||
end, GFP_NOFS);
|
end, GFP_NOFS);
|
||||||
}
|
}
|
||||||
|
mutex_unlock(&info->alloc_mutex);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2678,6 +2744,7 @@ static int noinline relocate_one_reference(struct btrfs_root *extent_root,
|
|||||||
|
|
||||||
eb = read_tree_block(found_root, extent_key->objectid,
|
eb = read_tree_block(found_root, extent_key->objectid,
|
||||||
extent_key->offset, 0);
|
extent_key->offset, 0);
|
||||||
|
btrfs_tree_lock(eb);
|
||||||
level = btrfs_header_level(eb);
|
level = btrfs_header_level(eb);
|
||||||
|
|
||||||
if (level == 0)
|
if (level == 0)
|
||||||
@ -2685,6 +2752,7 @@ static int noinline relocate_one_reference(struct btrfs_root *extent_root,
|
|||||||
else
|
else
|
||||||
btrfs_node_key_to_cpu(eb, &found_key, 0);
|
btrfs_node_key_to_cpu(eb, &found_key, 0);
|
||||||
|
|
||||||
|
btrfs_tree_unlock(eb);
|
||||||
free_extent_buffer(eb);
|
free_extent_buffer(eb);
|
||||||
|
|
||||||
ret = find_root_for_ref(extent_root, path, &found_key,
|
ret = find_root_for_ref(extent_root, path, &found_key,
|
||||||
@ -2888,6 +2956,7 @@ int btrfs_shrink_extent_tree(struct btrfs_root *root, u64 shrink_start)
|
|||||||
int ret;
|
int ret;
|
||||||
int progress;
|
int progress;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
shrink_block_group = btrfs_lookup_block_group(root->fs_info,
|
shrink_block_group = btrfs_lookup_block_group(root->fs_info,
|
||||||
shrink_start);
|
shrink_start);
|
||||||
BUG_ON(!shrink_block_group);
|
BUG_ON(!shrink_block_group);
|
||||||
@ -3044,20 +3113,22 @@ next:
|
|||||||
(unsigned int)-1, GFP_NOFS);
|
(unsigned int)-1, GFP_NOFS);
|
||||||
out:
|
out:
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int find_first_block_group(struct btrfs_root *root, struct btrfs_path *path,
|
int find_first_block_group(struct btrfs_root *root, struct btrfs_path *path,
|
||||||
struct btrfs_key *key)
|
struct btrfs_key *key)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret = 0;
|
||||||
struct btrfs_key found_key;
|
struct btrfs_key found_key;
|
||||||
struct extent_buffer *leaf;
|
struct extent_buffer *leaf;
|
||||||
int slot;
|
int slot;
|
||||||
|
|
||||||
ret = btrfs_search_slot(NULL, root, key, path, 0, 0);
|
ret = btrfs_search_slot(NULL, root, key, path, 0, 0);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
goto out;
|
||||||
|
|
||||||
while(1) {
|
while(1) {
|
||||||
slot = path->slots[0];
|
slot = path->slots[0];
|
||||||
leaf = path->nodes[0];
|
leaf = path->nodes[0];
|
||||||
@ -3066,18 +3137,20 @@ int find_first_block_group(struct btrfs_root *root, struct btrfs_path *path,
|
|||||||
if (ret == 0)
|
if (ret == 0)
|
||||||
continue;
|
continue;
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto error;
|
goto out;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
btrfs_item_key_to_cpu(leaf, &found_key, slot);
|
btrfs_item_key_to_cpu(leaf, &found_key, slot);
|
||||||
|
|
||||||
if (found_key.objectid >= key->objectid &&
|
if (found_key.objectid >= key->objectid &&
|
||||||
found_key.type == BTRFS_BLOCK_GROUP_ITEM_KEY)
|
found_key.type == BTRFS_BLOCK_GROUP_ITEM_KEY) {
|
||||||
return 0;
|
ret = 0;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
path->slots[0]++;
|
path->slots[0]++;
|
||||||
}
|
}
|
||||||
ret = -ENOENT;
|
ret = -ENOENT;
|
||||||
error:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3103,6 +3176,7 @@ int btrfs_read_block_groups(struct btrfs_root *root)
|
|||||||
if (!path)
|
if (!path)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->alloc_mutex);
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_block_group(root, path, &key);
|
ret = find_first_block_group(root, path, &key);
|
||||||
if (ret > 0) {
|
if (ret > 0) {
|
||||||
@ -3158,6 +3232,7 @@ int btrfs_read_block_groups(struct btrfs_root *root)
|
|||||||
ret = 0;
|
ret = 0;
|
||||||
error:
|
error:
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
|
mutex_unlock(&root->fs_info->alloc_mutex);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3205,5 +3280,6 @@ int btrfs_make_block_group(struct btrfs_trans_handle *trans,
|
|||||||
ret = del_pending_extents(trans, extent_root);
|
ret = del_pending_extents(trans, extent_root);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
set_avail_alloc_bits(extent_root->fs_info, type);
|
set_avail_alloc_bits(extent_root->fs_info, type);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -2889,7 +2889,6 @@ int clear_extent_buffer_dirty(struct extent_io_tree *tree,
|
|||||||
|
|
||||||
for (i = 0; i < num_pages; i++) {
|
for (i = 0; i < num_pages; i++) {
|
||||||
page = extent_buffer_page(eb, i);
|
page = extent_buffer_page(eb, i);
|
||||||
lock_page(page);
|
|
||||||
if (i == 0)
|
if (i == 0)
|
||||||
set_page_extent_head(page, eb->len);
|
set_page_extent_head(page, eb->len);
|
||||||
else
|
else
|
||||||
@ -2907,7 +2906,6 @@ int clear_extent_buffer_dirty(struct extent_io_tree *tree,
|
|||||||
end = start + PAGE_CACHE_SIZE - 1;
|
end = start + PAGE_CACHE_SIZE - 1;
|
||||||
if (test_range_bit(tree, start, end,
|
if (test_range_bit(tree, start, end,
|
||||||
EXTENT_DIRTY, 0)) {
|
EXTENT_DIRTY, 0)) {
|
||||||
unlock_page(page);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2919,7 +2917,6 @@ int clear_extent_buffer_dirty(struct extent_io_tree *tree,
|
|||||||
PAGECACHE_TAG_DIRTY);
|
PAGECACHE_TAG_DIRTY);
|
||||||
}
|
}
|
||||||
read_unlock_irq(&page->mapping->tree_lock);
|
read_unlock_irq(&page->mapping->tree_lock);
|
||||||
unlock_page(page);
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -2948,17 +2945,12 @@ int set_extent_buffer_dirty(struct extent_io_tree *tree,
|
|||||||
* on us if the page isn't already dirty.
|
* on us if the page isn't already dirty.
|
||||||
*/
|
*/
|
||||||
if (i == 0) {
|
if (i == 0) {
|
||||||
lock_page(page);
|
|
||||||
set_page_extent_head(page, eb->len);
|
set_page_extent_head(page, eb->len);
|
||||||
} else if (PagePrivate(page) &&
|
} else if (PagePrivate(page) &&
|
||||||
page->private != EXTENT_PAGE_PRIVATE) {
|
page->private != EXTENT_PAGE_PRIVATE) {
|
||||||
lock_page(page);
|
|
||||||
set_page_extent_mapped(page);
|
set_page_extent_mapped(page);
|
||||||
unlock_page(page);
|
|
||||||
}
|
}
|
||||||
__set_page_dirty_nobuffers(extent_buffer_page(eb, i));
|
__set_page_dirty_nobuffers(extent_buffer_page(eb, i));
|
||||||
if (i == 0)
|
|
||||||
unlock_page(page);
|
|
||||||
}
|
}
|
||||||
return set_extent_dirty(tree, eb->start,
|
return set_extent_dirty(tree, eb->start,
|
||||||
eb->start + eb->len - 1, GFP_NOFS);
|
eb->start + eb->len - 1, GFP_NOFS);
|
||||||
|
@ -115,6 +115,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
|
|||||||
trans = btrfs_start_transaction(root, 1);
|
trans = btrfs_start_transaction(root, 1);
|
||||||
BUG_ON(!trans);
|
BUG_ON(!trans);
|
||||||
btrfs_set_trans_block_group(trans, inode);
|
btrfs_set_trans_block_group(trans, inode);
|
||||||
|
mutex_unlock(&root->fs_info->fs_mutex);
|
||||||
|
|
||||||
num_bytes = (end - start + blocksize) & ~(blocksize - 1);
|
num_bytes = (end - start + blocksize) & ~(blocksize - 1);
|
||||||
num_bytes = max(blocksize, num_bytes);
|
num_bytes = max(blocksize, num_bytes);
|
||||||
@ -159,6 +160,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
|
|||||||
btrfs_add_ordered_inode(inode);
|
btrfs_add_ordered_inode(inode);
|
||||||
btrfs_update_inode(trans, root, inode);
|
btrfs_update_inode(trans, root, inode);
|
||||||
out:
|
out:
|
||||||
|
mutex_lock(&root->fs_info->fs_mutex);
|
||||||
btrfs_end_transaction(trans, root);
|
btrfs_end_transaction(trans, root);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -349,10 +351,12 @@ int __btrfs_submit_bio_hook(struct inode *inode, int rw, struct bio *bio,
|
|||||||
|
|
||||||
mutex_lock(&root->fs_info->fs_mutex);
|
mutex_lock(&root->fs_info->fs_mutex);
|
||||||
trans = btrfs_start_transaction(root, 1);
|
trans = btrfs_start_transaction(root, 1);
|
||||||
|
mutex_unlock(&root->fs_info->fs_mutex);
|
||||||
|
|
||||||
btrfs_set_trans_block_group(trans, inode);
|
btrfs_set_trans_block_group(trans, inode);
|
||||||
btrfs_csum_file_blocks(trans, root, inode, bio, sums);
|
btrfs_csum_file_blocks(trans, root, inode, bio, sums);
|
||||||
|
|
||||||
|
mutex_lock(&root->fs_info->fs_mutex);
|
||||||
ret = btrfs_end_transaction(trans, root);
|
ret = btrfs_end_transaction(trans, root);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
mutex_unlock(&root->fs_info->fs_mutex);
|
mutex_unlock(&root->fs_info->fs_mutex);
|
||||||
@ -807,6 +811,7 @@ static int btrfs_unlink_trans(struct btrfs_trans_handle *trans,
|
|||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
ret = btrfs_delete_one_dir_name(trans, root, path, di);
|
ret = btrfs_delete_one_dir_name(trans, root, path, di);
|
||||||
|
btrfs_release_path(root, path);
|
||||||
|
|
||||||
dentry->d_inode->i_ctime = dir->i_ctime;
|
dentry->d_inode->i_ctime = dir->i_ctime;
|
||||||
ret = btrfs_del_inode_ref(trans, root, name, name_len,
|
ret = btrfs_del_inode_ref(trans, root, name, name_len,
|
||||||
@ -881,8 +886,9 @@ static int btrfs_rmdir(struct inode *dir, struct dentry *dentry)
|
|||||||
struct btrfs_trans_handle *trans;
|
struct btrfs_trans_handle *trans;
|
||||||
unsigned long nr = 0;
|
unsigned long nr = 0;
|
||||||
|
|
||||||
if (inode->i_size > BTRFS_EMPTY_DIR_SIZE)
|
if (inode->i_size > BTRFS_EMPTY_DIR_SIZE) {
|
||||||
return -ENOTEMPTY;
|
return -ENOTEMPTY;
|
||||||
|
}
|
||||||
|
|
||||||
mutex_lock(&root->fs_info->fs_mutex);
|
mutex_lock(&root->fs_info->fs_mutex);
|
||||||
ret = btrfs_check_free_space(root, 1, 1);
|
ret = btrfs_check_free_space(root, 1, 1);
|
||||||
|
@ -43,6 +43,7 @@
|
|||||||
#include "ioctl.h"
|
#include "ioctl.h"
|
||||||
#include "print-tree.h"
|
#include "print-tree.h"
|
||||||
#include "volumes.h"
|
#include "volumes.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -75,9 +76,9 @@ static noinline int create_subvol(struct btrfs_root *root, char *name,
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
leaf = __btrfs_alloc_free_block(trans, root, root->leafsize,
|
leaf = btrfs_alloc_free_block(trans, root, root->leafsize,
|
||||||
objectid, trans->transid, 0, 0,
|
objectid, trans->transid, 0, 0,
|
||||||
0, 0);
|
0, 0);
|
||||||
if (IS_ERR(leaf))
|
if (IS_ERR(leaf))
|
||||||
return PTR_ERR(leaf);
|
return PTR_ERR(leaf);
|
||||||
|
|
||||||
@ -108,6 +109,7 @@ static noinline int create_subvol(struct btrfs_root *root, char *name,
|
|||||||
memset(&root_item.drop_progress, 0, sizeof(root_item.drop_progress));
|
memset(&root_item.drop_progress, 0, sizeof(root_item.drop_progress));
|
||||||
root_item.drop_level = 0;
|
root_item.drop_level = 0;
|
||||||
|
|
||||||
|
btrfs_tree_unlock(leaf);
|
||||||
free_extent_buffer(leaf);
|
free_extent_buffer(leaf);
|
||||||
leaf = NULL;
|
leaf = NULL;
|
||||||
|
|
||||||
|
50
fs/btrfs/locking.c
Normal file
50
fs/btrfs/locking.c
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2008 Oracle. All rights reserved.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public
|
||||||
|
* License v2 as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public
|
||||||
|
* License along with this program; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 021110-1307, USA.
|
||||||
|
*/
|
||||||
|
#include <linux/sched.h>
|
||||||
|
#include <linux/gfp.h>
|
||||||
|
#include <linux/pagemap.h>
|
||||||
|
#include <linux/spinlock.h>
|
||||||
|
#include <linux/page-flags.h>
|
||||||
|
#include <linux/bug.h>
|
||||||
|
#include "ctree.h"
|
||||||
|
#include "extent_io.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
|
int btrfs_tree_lock(struct extent_buffer *eb)
|
||||||
|
{
|
||||||
|
lock_page(eb->first_page);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int btrfs_try_tree_lock(struct extent_buffer *eb)
|
||||||
|
{
|
||||||
|
return TestSetPageLocked(eb->first_page);
|
||||||
|
}
|
||||||
|
|
||||||
|
int btrfs_tree_unlock(struct extent_buffer *eb)
|
||||||
|
{
|
||||||
|
WARN_ON(!PageLocked(eb->first_page));
|
||||||
|
unlock_page(eb->first_page);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int btrfs_tree_locked(struct extent_buffer *eb)
|
||||||
|
{
|
||||||
|
return PageLocked(eb->first_page);
|
||||||
|
}
|
||||||
|
|
26
fs/btrfs/locking.h
Normal file
26
fs/btrfs/locking.h
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2008 Oracle. All rights reserved.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public
|
||||||
|
* License v2 as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public
|
||||||
|
* License along with this program; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 021110-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __BTRFS_LOCKING_
|
||||||
|
#define __BTRFS_LOCKING_
|
||||||
|
|
||||||
|
int btrfs_tree_lock(struct extent_buffer *eb);
|
||||||
|
int btrfs_tree_unlock(struct extent_buffer *eb);
|
||||||
|
int btrfs_tree_locked(struct extent_buffer *eb);
|
||||||
|
int btrfs_try_tree_lock(struct extent_buffer *eb);
|
||||||
|
#endif
|
@ -23,6 +23,7 @@
|
|||||||
#include "ctree.h"
|
#include "ctree.h"
|
||||||
#include "disk-io.h"
|
#include "disk-io.h"
|
||||||
#include "transaction.h"
|
#include "transaction.h"
|
||||||
|
#include "locking.h"
|
||||||
|
|
||||||
static int total_trans = 0;
|
static int total_trans = 0;
|
||||||
extern struct kmem_cache *btrfs_trans_handle_cachep;
|
extern struct kmem_cache *btrfs_trans_handle_cachep;
|
||||||
@ -96,8 +97,7 @@ static noinline int record_root_in_trans(struct btrfs_root *root)
|
|||||||
radix_tree_tag_set(&root->fs_info->fs_roots_radix,
|
radix_tree_tag_set(&root->fs_info->fs_roots_radix,
|
||||||
(unsigned long)root->root_key.objectid,
|
(unsigned long)root->root_key.objectid,
|
||||||
BTRFS_ROOT_DEFRAG_TAG);
|
BTRFS_ROOT_DEFRAG_TAG);
|
||||||
root->commit_root = root->node;
|
root->commit_root = btrfs_root_node(root);
|
||||||
extent_buffer_get(root->node);
|
|
||||||
} else {
|
} else {
|
||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
}
|
}
|
||||||
@ -559,6 +559,7 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
|
|||||||
struct btrfs_root *tree_root = fs_info->tree_root;
|
struct btrfs_root *tree_root = fs_info->tree_root;
|
||||||
struct btrfs_root *root = pending->root;
|
struct btrfs_root *root = pending->root;
|
||||||
struct extent_buffer *tmp;
|
struct extent_buffer *tmp;
|
||||||
|
struct extent_buffer *old;
|
||||||
int ret;
|
int ret;
|
||||||
int namelen;
|
int namelen;
|
||||||
u64 objectid;
|
u64 objectid;
|
||||||
@ -578,16 +579,18 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
|
|||||||
key.offset = 1;
|
key.offset = 1;
|
||||||
btrfs_set_key_type(&key, BTRFS_ROOT_ITEM_KEY);
|
btrfs_set_key_type(&key, BTRFS_ROOT_ITEM_KEY);
|
||||||
|
|
||||||
extent_buffer_get(root->node);
|
old = btrfs_lock_root_node(root);
|
||||||
btrfs_cow_block(trans, root, root->node, NULL, 0, &tmp);
|
btrfs_cow_block(trans, root, old, NULL, 0, &old);
|
||||||
free_extent_buffer(tmp);
|
|
||||||
|
|
||||||
btrfs_copy_root(trans, root, root->node, &tmp, objectid);
|
btrfs_copy_root(trans, root, old, &tmp, objectid);
|
||||||
|
btrfs_tree_unlock(old);
|
||||||
|
free_extent_buffer(old);
|
||||||
|
|
||||||
btrfs_set_root_bytenr(new_root_item, tmp->start);
|
btrfs_set_root_bytenr(new_root_item, tmp->start);
|
||||||
btrfs_set_root_level(new_root_item, btrfs_header_level(tmp));
|
btrfs_set_root_level(new_root_item, btrfs_header_level(tmp));
|
||||||
ret = btrfs_insert_root(trans, root->fs_info->tree_root, &key,
|
ret = btrfs_insert_root(trans, root->fs_info->tree_root, &key,
|
||||||
new_root_item);
|
new_root_item);
|
||||||
|
btrfs_tree_unlock(tmp);
|
||||||
free_extent_buffer(tmp);
|
free_extent_buffer(tmp);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
@ -181,6 +181,8 @@ int btrfs_defrag_leaves(struct btrfs_trans_handle *trans,
|
|||||||
if (root->fs_info->extent_root == root)
|
if (root->fs_info->extent_root == root)
|
||||||
is_extent = 1;
|
is_extent = 1;
|
||||||
|
|
||||||
|
goto out;
|
||||||
|
|
||||||
if (root->ref_cows == 0 && !is_extent)
|
if (root->ref_cows == 0 && !is_extent)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user