Btrfs: Fix extent_buffer and extent_state leaks
Signed-off-by: Chris Mason <chris.mason@oracle.com>
This commit is contained in:
parent
ae5252bd51
commit
f510cfecfc
|
@ -87,6 +87,7 @@ static int __btrfs_cow_block(struct btrfs_trans_handle *trans,
|
||||||
if (IS_ERR(cow))
|
if (IS_ERR(cow))
|
||||||
return PTR_ERR(cow);
|
return PTR_ERR(cow);
|
||||||
|
|
||||||
|
cow->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
if (buf->len != root->sectorsize || cow->len != root->sectorsize)
|
if (buf->len != root->sectorsize || cow->len != root->sectorsize)
|
||||||
WARN_ON(1);
|
WARN_ON(1);
|
||||||
|
|
||||||
|
@ -132,6 +133,7 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
|
||||||
struct extent_buffer **cow_ret)
|
struct extent_buffer **cow_ret)
|
||||||
{
|
{
|
||||||
u64 search_start;
|
u64 search_start;
|
||||||
|
int ret;
|
||||||
if (trans->transaction != root->fs_info->running_transaction) {
|
if (trans->transaction != root->fs_info->running_transaction) {
|
||||||
printk(KERN_CRIT "trans %Lu running %Lu\n", trans->transid,
|
printk(KERN_CRIT "trans %Lu running %Lu\n", trans->transid,
|
||||||
root->fs_info->running_transaction->transid);
|
root->fs_info->running_transaction->transid);
|
||||||
|
@ -148,8 +150,10 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
|
||||||
}
|
}
|
||||||
|
|
||||||
search_start = extent_buffer_blocknr(buf) & ~((u64)65535);
|
search_start = extent_buffer_blocknr(buf) & ~((u64)65535);
|
||||||
return __btrfs_cow_block(trans, root, buf, parent,
|
ret = __btrfs_cow_block(trans, root, buf, parent,
|
||||||
parent_slot, cow_ret, search_start, 0);
|
parent_slot, cow_ret, search_start, 0);
|
||||||
|
(*cow_ret)->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int close_blocks(u64 blocknr, u64 other)
|
static int close_blocks(u64 blocknr, u64 other)
|
||||||
|
@ -1013,8 +1017,10 @@ again:
|
||||||
if (sret)
|
if (sret)
|
||||||
return sret;
|
return sret;
|
||||||
b = p->nodes[level];
|
b = p->nodes[level];
|
||||||
if (!b)
|
if (!b) {
|
||||||
|
btrfs_release_path(NULL, p);
|
||||||
goto again;
|
goto again;
|
||||||
|
}
|
||||||
slot = p->slots[level];
|
slot = p->slots[level];
|
||||||
BUG_ON(btrfs_header_nritems(b) == 1);
|
BUG_ON(btrfs_header_nritems(b) == 1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -303,8 +303,8 @@ struct btrfs_fs_info {
|
||||||
struct radix_tree_root pinned_radix;
|
struct radix_tree_root pinned_radix;
|
||||||
struct radix_tree_root block_group_radix;
|
struct radix_tree_root block_group_radix;
|
||||||
struct radix_tree_root block_group_data_radix;
|
struct radix_tree_root block_group_data_radix;
|
||||||
struct radix_tree_root extent_map_radix;
|
|
||||||
struct radix_tree_root extent_ins_radix;
|
struct radix_tree_root extent_ins_radix;
|
||||||
|
struct extent_map_tree free_space_cache;
|
||||||
u64 generation;
|
u64 generation;
|
||||||
u64 last_trans_committed;
|
u64 last_trans_committed;
|
||||||
struct btrfs_transaction *running_transaction;
|
struct btrfs_transaction *running_transaction;
|
||||||
|
|
|
@ -46,18 +46,25 @@ struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
|
||||||
u64 blocknr)
|
u64 blocknr)
|
||||||
{
|
{
|
||||||
struct inode *btree_inode = root->fs_info->btree_inode;
|
struct inode *btree_inode = root->fs_info->btree_inode;
|
||||||
return find_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
|
struct extent_buffer *eb;
|
||||||
|
eb = find_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
|
||||||
blocknr * root->sectorsize,
|
blocknr * root->sectorsize,
|
||||||
root->sectorsize, GFP_NOFS);
|
root->sectorsize, GFP_NOFS);
|
||||||
|
if (eb)
|
||||||
|
eb->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
|
return eb;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
|
struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
|
||||||
u64 blocknr)
|
u64 blocknr)
|
||||||
{
|
{
|
||||||
struct inode *btree_inode = root->fs_info->btree_inode;
|
struct inode *btree_inode = root->fs_info->btree_inode;
|
||||||
return alloc_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
|
struct extent_buffer *eb;
|
||||||
|
eb = alloc_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
|
||||||
blocknr * root->sectorsize,
|
blocknr * root->sectorsize,
|
||||||
root->sectorsize, GFP_NOFS);
|
root->sectorsize, GFP_NOFS);
|
||||||
|
eb->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
|
return eb;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct extent_map *btree_get_extent(struct inode *inode, struct page *page,
|
struct extent_map *btree_get_extent(struct inode *inode, struct page *page,
|
||||||
|
@ -226,6 +233,7 @@ struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 blocknr)
|
||||||
return NULL;
|
return NULL;
|
||||||
read_extent_buffer_pages(&BTRFS_I(btree_inode)->extent_tree,
|
read_extent_buffer_pages(&BTRFS_I(btree_inode)->extent_tree,
|
||||||
buf, 1);
|
buf, 1);
|
||||||
|
buf->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -426,7 +434,6 @@ struct btrfs_root *open_ctree(struct super_block *sb)
|
||||||
}
|
}
|
||||||
init_bit_radix(&fs_info->pinned_radix);
|
init_bit_radix(&fs_info->pinned_radix);
|
||||||
init_bit_radix(&fs_info->pending_del_radix);
|
init_bit_radix(&fs_info->pending_del_radix);
|
||||||
init_bit_radix(&fs_info->extent_map_radix);
|
|
||||||
init_bit_radix(&fs_info->extent_ins_radix);
|
init_bit_radix(&fs_info->extent_ins_radix);
|
||||||
INIT_RADIX_TREE(&fs_info->fs_roots_radix, GFP_NOFS);
|
INIT_RADIX_TREE(&fs_info->fs_roots_radix, GFP_NOFS);
|
||||||
INIT_RADIX_TREE(&fs_info->block_group_radix, GFP_KERNEL);
|
INIT_RADIX_TREE(&fs_info->block_group_radix, GFP_KERNEL);
|
||||||
|
@ -449,6 +456,8 @@ struct btrfs_root *open_ctree(struct super_block *sb)
|
||||||
extent_map_tree_init(&BTRFS_I(fs_info->btree_inode)->extent_tree,
|
extent_map_tree_init(&BTRFS_I(fs_info->btree_inode)->extent_tree,
|
||||||
fs_info->btree_inode->i_mapping,
|
fs_info->btree_inode->i_mapping,
|
||||||
GFP_NOFS);
|
GFP_NOFS);
|
||||||
|
extent_map_tree_init(&fs_info->free_space_cache,
|
||||||
|
fs_info->btree_inode->i_mapping, GFP_NOFS);
|
||||||
fs_info->do_barriers = 1;
|
fs_info->do_barriers = 1;
|
||||||
fs_info->closing = 0;
|
fs_info->closing = 0;
|
||||||
|
|
||||||
|
@ -594,8 +603,10 @@ int close_ctree(struct btrfs_root *root)
|
||||||
|
|
||||||
if (fs_info->extent_root->node)
|
if (fs_info->extent_root->node)
|
||||||
free_extent_buffer(fs_info->extent_root->node);
|
free_extent_buffer(fs_info->extent_root->node);
|
||||||
|
|
||||||
if (fs_info->tree_root->node)
|
if (fs_info->tree_root->node)
|
||||||
free_extent_buffer(fs_info->tree_root->node);
|
free_extent_buffer(fs_info->tree_root->node);
|
||||||
|
|
||||||
free_extent_buffer(fs_info->sb_buffer);
|
free_extent_buffer(fs_info->sb_buffer);
|
||||||
truncate_inode_pages(fs_info->btree_inode->i_mapping, 0);
|
truncate_inode_pages(fs_info->btree_inode->i_mapping, 0);
|
||||||
iput(fs_info->btree_inode);
|
iput(fs_info->btree_inode);
|
||||||
|
|
|
@ -34,21 +34,19 @@ static int cache_block_group(struct btrfs_root *root,
|
||||||
int ret;
|
int ret;
|
||||||
struct btrfs_key key;
|
struct btrfs_key key;
|
||||||
struct extent_buffer *leaf;
|
struct extent_buffer *leaf;
|
||||||
struct radix_tree_root *extent_radix;
|
struct extent_map_tree *free_space_cache;
|
||||||
int slot;
|
int slot;
|
||||||
u64 i;
|
|
||||||
u64 last = 0;
|
u64 last = 0;
|
||||||
u64 hole_size;
|
u64 hole_size;
|
||||||
u64 first_free;
|
u64 first_free;
|
||||||
int found = 0;
|
int found = 0;
|
||||||
|
|
||||||
root = root->fs_info->extent_root;
|
root = root->fs_info->extent_root;
|
||||||
extent_radix = &root->fs_info->extent_map_radix;
|
free_space_cache = &root->fs_info->free_space_cache;
|
||||||
|
|
||||||
if (block_group->cached)
|
if (block_group->cached)
|
||||||
return 0;
|
return 0;
|
||||||
if (block_group->data)
|
|
||||||
return 0;
|
|
||||||
path = btrfs_alloc_path();
|
path = btrfs_alloc_path();
|
||||||
if (!path)
|
if (!path)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
@ -98,9 +96,11 @@ static int cache_block_group(struct btrfs_root *root,
|
||||||
last = first_free;
|
last = first_free;
|
||||||
found = 1;
|
found = 1;
|
||||||
}
|
}
|
||||||
hole_size = key.objectid - last;
|
if (key.objectid > last) {
|
||||||
for (i = 0; i < hole_size; i++) {
|
hole_size = key.objectid - last;
|
||||||
set_radix_bit(extent_radix, last + i);
|
set_extent_dirty(free_space_cache, last,
|
||||||
|
last + hole_size - 1,
|
||||||
|
GFP_NOFS);
|
||||||
}
|
}
|
||||||
last = key.objectid + key.offset;
|
last = key.objectid + key.offset;
|
||||||
}
|
}
|
||||||
|
@ -114,9 +114,8 @@ next:
|
||||||
block_group->key.offset > last) {
|
block_group->key.offset > last) {
|
||||||
hole_size = block_group->key.objectid +
|
hole_size = block_group->key.objectid +
|
||||||
block_group->key.offset - last;
|
block_group->key.offset - last;
|
||||||
for (i = 0; i < hole_size; i++) {
|
set_extent_dirty(free_space_cache, last,
|
||||||
set_radix_bit(extent_radix, last + i);
|
last + hole_size - 1, GFP_NOFS);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
block_group->cached = 1;
|
block_group->cached = 1;
|
||||||
err:
|
err:
|
||||||
|
@ -150,47 +149,33 @@ struct btrfs_block_group_cache *btrfs_lookup_block_group(struct
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static u64 leaf_range(struct btrfs_root *root)
|
|
||||||
{
|
|
||||||
u64 size = BTRFS_LEAF_DATA_SIZE(root);
|
|
||||||
do_div(size, sizeof(struct btrfs_extent_item) +
|
|
||||||
sizeof(struct btrfs_item));
|
|
||||||
return size;
|
|
||||||
}
|
|
||||||
|
|
||||||
static u64 find_search_start(struct btrfs_root *root,
|
static u64 find_search_start(struct btrfs_root *root,
|
||||||
struct btrfs_block_group_cache **cache_ret,
|
struct btrfs_block_group_cache **cache_ret,
|
||||||
u64 search_start, int num)
|
u64 search_start, int num, int data)
|
||||||
{
|
{
|
||||||
unsigned long gang[8];
|
|
||||||
int ret;
|
int ret;
|
||||||
struct btrfs_block_group_cache *cache = *cache_ret;
|
struct btrfs_block_group_cache *cache = *cache_ret;
|
||||||
u64 last = max(search_start, cache->key.objectid);
|
u64 last = max(search_start, cache->key.objectid);
|
||||||
|
u64 start = 0;
|
||||||
|
u64 end = 0;
|
||||||
|
|
||||||
if (cache->data)
|
|
||||||
goto out;
|
|
||||||
again:
|
again:
|
||||||
ret = cache_block_group(root, cache);
|
ret = cache_block_group(root, cache);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_radix_bit(&root->fs_info->extent_map_radix,
|
ret = find_first_extent_bit(&root->fs_info->free_space_cache,
|
||||||
gang, last, ARRAY_SIZE(gang));
|
last, &start, &end, EXTENT_DIRTY);
|
||||||
if (!ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
last = gang[ret-1] + 1;
|
|
||||||
if (num > 1) {
|
start = max(last, start);
|
||||||
if (ret != ARRAY_SIZE(gang)) {
|
last = end + 1;
|
||||||
goto new_group;
|
if (end + 1 - start < num)
|
||||||
}
|
continue;
|
||||||
if (gang[ret-1] - gang[0] > leaf_range(root)) {
|
if (start + num > cache->key.objectid + cache->key.offset)
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (gang[0] >= cache->key.objectid + cache->key.offset) {
|
|
||||||
goto new_group;
|
goto new_group;
|
||||||
}
|
return start;
|
||||||
return gang[0];
|
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
return max(cache->last_alloc, search_start);
|
return max(cache->last_alloc, search_start);
|
||||||
|
@ -202,7 +187,7 @@ new_group:
|
||||||
return max((*cache_ret)->last_alloc, search_start);
|
return max((*cache_ret)->last_alloc, search_start);
|
||||||
}
|
}
|
||||||
cache = btrfs_find_block_group(root, cache,
|
cache = btrfs_find_block_group(root, cache,
|
||||||
last + cache->key.offset - 1, 0, 0);
|
last + cache->key.offset - 1, data, 0);
|
||||||
*cache_ret = cache;
|
*cache_ret = cache;
|
||||||
goto again;
|
goto again;
|
||||||
}
|
}
|
||||||
|
@ -625,7 +610,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
|
||||||
u64 total = num;
|
u64 total = num;
|
||||||
u64 old_val;
|
u64 old_val;
|
||||||
u64 block_in_group;
|
u64 block_in_group;
|
||||||
u64 i;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
while(total) {
|
while(total) {
|
||||||
|
@ -644,12 +628,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
|
||||||
if (alloc) {
|
if (alloc) {
|
||||||
if (blocknr > cache->last_alloc)
|
if (blocknr > cache->last_alloc)
|
||||||
cache->last_alloc = blocknr;
|
cache->last_alloc = blocknr;
|
||||||
if (!cache->data) {
|
|
||||||
for (i = 0; i < num; i++) {
|
|
||||||
clear_radix_bit(&info->extent_map_radix,
|
|
||||||
blocknr + i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (cache->data != data &&
|
if (cache->data != data &&
|
||||||
old_val < (cache->key.offset >> 1)) {
|
old_val < (cache->key.offset >> 1)) {
|
||||||
cache->data = data;
|
cache->data = data;
|
||||||
|
@ -677,11 +655,10 @@ static int update_block_group(struct btrfs_trans_handle *trans,
|
||||||
old_val -= num;
|
old_val -= num;
|
||||||
if (blocknr < cache->first_free)
|
if (blocknr < cache->first_free)
|
||||||
cache->first_free = blocknr;
|
cache->first_free = blocknr;
|
||||||
if (!cache->data && mark_free) {
|
if (mark_free) {
|
||||||
for (i = 0; i < num; i++) {
|
set_extent_dirty(&info->free_space_cache,
|
||||||
set_radix_bit(&info->extent_map_radix,
|
blocknr, blocknr + num - 1,
|
||||||
blocknr + i);
|
GFP_NOFS);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (old_val < (cache->key.offset >> 1) &&
|
if (old_val < (cache->key.offset >> 1) &&
|
||||||
old_val + num >= (cache->key.offset >> 1)) {
|
old_val + num >= (cache->key.offset >> 1)) {
|
||||||
|
@ -732,7 +709,9 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
|
||||||
int ret;
|
int ret;
|
||||||
int i;
|
int i;
|
||||||
struct radix_tree_root *pinned_radix = &root->fs_info->pinned_radix;
|
struct radix_tree_root *pinned_radix = &root->fs_info->pinned_radix;
|
||||||
struct radix_tree_root *extent_radix = &root->fs_info->extent_map_radix;
|
struct extent_map_tree *free_space_cache;
|
||||||
|
|
||||||
|
free_space_cache = &root->fs_info->free_space_cache;
|
||||||
|
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_radix_bit(unpin_radix, gang, 0,
|
ret = find_first_radix_bit(unpin_radix, gang, 0,
|
||||||
|
@ -751,8 +730,11 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
|
||||||
block_group->pinned--;
|
block_group->pinned--;
|
||||||
if (gang[i] < block_group->last_alloc)
|
if (gang[i] < block_group->last_alloc)
|
||||||
block_group->last_alloc = gang[i];
|
block_group->last_alloc = gang[i];
|
||||||
if (!block_group->data)
|
if (!block_group->data) {
|
||||||
set_radix_bit(extent_radix, gang[i]);
|
set_extent_dirty(free_space_cache,
|
||||||
|
gang[i], gang[i],
|
||||||
|
GFP_NOFS);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -995,6 +977,7 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
|
||||||
struct btrfs_block_group_cache *block_group;
|
struct btrfs_block_group_cache *block_group;
|
||||||
int full_scan = 0;
|
int full_scan = 0;
|
||||||
int wrapped = 0;
|
int wrapped = 0;
|
||||||
|
u64 cached_search_start = 0;
|
||||||
|
|
||||||
WARN_ON(num_blocks < 1);
|
WARN_ON(num_blocks < 1);
|
||||||
btrfs_set_key_type(ins, BTRFS_EXTENT_ITEM_KEY);
|
btrfs_set_key_type(ins, BTRFS_EXTENT_ITEM_KEY);
|
||||||
|
@ -1017,11 +1000,9 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
|
||||||
path = btrfs_alloc_path();
|
path = btrfs_alloc_path();
|
||||||
|
|
||||||
check_failed:
|
check_failed:
|
||||||
if (!block_group->data)
|
search_start = find_search_start(root, &block_group,
|
||||||
search_start = find_search_start(root, &block_group,
|
search_start, total_needed, data);
|
||||||
search_start, total_needed);
|
cached_search_start = search_start;
|
||||||
else if (!full_scan)
|
|
||||||
search_start = max(block_group->last_alloc, search_start);
|
|
||||||
|
|
||||||
btrfs_init_path(path);
|
btrfs_init_path(path);
|
||||||
ins->objectid = search_start;
|
ins->objectid = search_start;
|
||||||
|
@ -1097,6 +1078,7 @@ check_failed:
|
||||||
|
|
||||||
start_found = 1;
|
start_found = 1;
|
||||||
last_block = key.objectid + key.offset;
|
last_block = key.objectid + key.offset;
|
||||||
|
|
||||||
if (!full_scan && last_block >= block_group->key.objectid +
|
if (!full_scan && last_block >= block_group->key.objectid +
|
||||||
block_group->key.offset) {
|
block_group->key.offset) {
|
||||||
btrfs_release_path(root, path);
|
btrfs_release_path(root, path);
|
||||||
|
@ -1138,6 +1120,9 @@ check_pending:
|
||||||
}
|
}
|
||||||
ins->offset = num_blocks;
|
ins->offset = num_blocks;
|
||||||
btrfs_free_path(path);
|
btrfs_free_path(path);
|
||||||
|
if (0 && ins->objectid != cached_search_start) {
|
||||||
|
printk("\tcached was %Lu found %Lu\n", cached_search_start, ins->objectid);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
new_group:
|
new_group:
|
||||||
|
@ -1209,6 +1194,10 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
|
||||||
btrfs_set_root_used(&root->root_item, root_blocks_used +
|
btrfs_set_root_used(&root->root_item, root_blocks_used +
|
||||||
num_blocks);
|
num_blocks);
|
||||||
|
|
||||||
|
clear_extent_dirty(&root->fs_info->free_space_cache,
|
||||||
|
ins->objectid, ins->objectid + ins->offset - 1,
|
||||||
|
GFP_NOFS);
|
||||||
|
|
||||||
if (root == extent_root) {
|
if (root == extent_root) {
|
||||||
BUG_ON(num_blocks != 1);
|
BUG_ON(num_blocks != 1);
|
||||||
set_radix_bit(&root->fs_info->extent_ins_radix, ins->objectid);
|
set_radix_bit(&root->fs_info->extent_ins_radix, ins->objectid);
|
||||||
|
@ -1227,6 +1216,7 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
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;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1265,6 +1255,7 @@ struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
btrfs_set_buffer_uptodate(buf);
|
btrfs_set_buffer_uptodate(buf);
|
||||||
|
buf->alloc_addr = (unsigned long)__builtin_return_address(0);
|
||||||
set_extent_dirty(&trans->transaction->dirty_pages, buf->start,
|
set_extent_dirty(&trans->transaction->dirty_pages, buf->start,
|
||||||
buf->start + buf->len - 1, GFP_NOFS);
|
buf->start + buf->len - 1, GFP_NOFS);
|
||||||
/*
|
/*
|
||||||
|
@ -1492,6 +1483,7 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
|
||||||
orig_level = level;
|
orig_level = level;
|
||||||
if (btrfs_disk_key_objectid(&root_item->drop_progress) == 0) {
|
if (btrfs_disk_key_objectid(&root_item->drop_progress) == 0) {
|
||||||
path->nodes[level] = root->node;
|
path->nodes[level] = root->node;
|
||||||
|
extent_buffer_get(root->node);
|
||||||
path->slots[level] = 0;
|
path->slots[level] = 0;
|
||||||
} else {
|
} else {
|
||||||
struct btrfs_key key;
|
struct btrfs_key key;
|
||||||
|
@ -1524,7 +1516,6 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
|
||||||
if (wret < 0)
|
if (wret < 0)
|
||||||
ret = wret;
|
ret = wret;
|
||||||
ret = -EAGAIN;
|
ret = -EAGAIN;
|
||||||
extent_buffer_get(root->node);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
for (i = 0; i <= orig_level; i++) {
|
for (i = 0; i <= orig_level; i++) {
|
||||||
|
@ -1562,8 +1553,8 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
int ret2;
|
int ret2;
|
||||||
unsigned long gang[16];
|
u64 start;
|
||||||
int i;
|
u64 end;
|
||||||
|
|
||||||
ret = free_block_group_radix(&info->block_group_radix);
|
ret = free_block_group_radix(&info->block_group_radix);
|
||||||
ret2 = free_block_group_radix(&info->block_group_data_radix);
|
ret2 = free_block_group_radix(&info->block_group_data_radix);
|
||||||
|
@ -1573,13 +1564,12 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
|
||||||
return ret2;
|
return ret2;
|
||||||
|
|
||||||
while(1) {
|
while(1) {
|
||||||
ret = find_first_radix_bit(&info->extent_map_radix,
|
ret = find_first_extent_bit(&info->free_space_cache, 0,
|
||||||
gang, 0, ARRAY_SIZE(gang));
|
&start, &end, EXTENT_DIRTY);
|
||||||
if (!ret)
|
if (ret)
|
||||||
break;
|
break;
|
||||||
for (i = 0; i < ret; i++) {
|
clear_extent_dirty(&info->free_space_cache, start,
|
||||||
clear_radix_bit(&info->extent_map_radix, gang[i]);
|
end, GFP_NOFS);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,8 +19,13 @@ struct kmem_cache *btrfs_cache_create(const char *name, size_t size,
|
||||||
static struct kmem_cache *extent_map_cache;
|
static struct kmem_cache *extent_map_cache;
|
||||||
static struct kmem_cache *extent_state_cache;
|
static struct kmem_cache *extent_state_cache;
|
||||||
static struct kmem_cache *extent_buffer_cache;
|
static struct kmem_cache *extent_buffer_cache;
|
||||||
|
|
||||||
static LIST_HEAD(extent_buffers);
|
static LIST_HEAD(extent_buffers);
|
||||||
|
static LIST_HEAD(buffers);
|
||||||
|
static LIST_HEAD(states);
|
||||||
|
|
||||||
static spinlock_t extent_buffers_lock;
|
static spinlock_t extent_buffers_lock;
|
||||||
|
static spinlock_t state_lock = SPIN_LOCK_UNLOCKED;
|
||||||
static int nr_extent_buffers;
|
static int nr_extent_buffers;
|
||||||
#define MAX_EXTENT_BUFFER_CACHE 128
|
#define MAX_EXTENT_BUFFER_CACHE 128
|
||||||
|
|
||||||
|
@ -48,6 +53,7 @@ void __init extent_map_init(void)
|
||||||
void __exit extent_map_exit(void)
|
void __exit extent_map_exit(void)
|
||||||
{
|
{
|
||||||
struct extent_buffer *eb;
|
struct extent_buffer *eb;
|
||||||
|
struct extent_state *state;
|
||||||
|
|
||||||
while (!list_empty(&extent_buffers)) {
|
while (!list_empty(&extent_buffers)) {
|
||||||
eb = list_entry(extent_buffers.next,
|
eb = list_entry(extent_buffers.next,
|
||||||
|
@ -55,6 +61,22 @@ void __exit extent_map_exit(void)
|
||||||
list_del(&eb->list);
|
list_del(&eb->list);
|
||||||
kmem_cache_free(extent_buffer_cache, eb);
|
kmem_cache_free(extent_buffer_cache, eb);
|
||||||
}
|
}
|
||||||
|
while (!list_empty(&states)) {
|
||||||
|
state = list_entry(states.next, struct extent_state, list);
|
||||||
|
printk("state leak: start %Lu end %Lu state %lu in tree %d refs %d\n", state->start, state->end, state->state, state->in_tree, atomic_read(&state->refs));
|
||||||
|
list_del(&state->list);
|
||||||
|
kmem_cache_free(extent_state_cache, state);
|
||||||
|
|
||||||
|
}
|
||||||
|
while (!list_empty(&buffers)) {
|
||||||
|
eb = list_entry(buffers.next,
|
||||||
|
struct extent_buffer, leak_list);
|
||||||
|
printk("buffer leak start %Lu len %lu return %lX\n", eb->start, eb->len, eb->alloc_addr);
|
||||||
|
list_del(&eb->leak_list);
|
||||||
|
kmem_cache_free(extent_buffer_cache, eb);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (extent_map_cache)
|
if (extent_map_cache)
|
||||||
kmem_cache_destroy(extent_map_cache);
|
kmem_cache_destroy(extent_map_cache);
|
||||||
if (extent_state_cache)
|
if (extent_state_cache)
|
||||||
|
@ -101,12 +123,19 @@ EXPORT_SYMBOL(free_extent_map);
|
||||||
struct extent_state *alloc_extent_state(gfp_t mask)
|
struct extent_state *alloc_extent_state(gfp_t mask)
|
||||||
{
|
{
|
||||||
struct extent_state *state;
|
struct extent_state *state;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
state = kmem_cache_alloc(extent_state_cache, mask);
|
state = kmem_cache_alloc(extent_state_cache, mask);
|
||||||
if (!state || IS_ERR(state))
|
if (!state || IS_ERR(state))
|
||||||
return state;
|
return state;
|
||||||
state->state = 0;
|
state->state = 0;
|
||||||
state->in_tree = 0;
|
state->in_tree = 0;
|
||||||
state->private = 0;
|
state->private = 0;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&state_lock, flags);
|
||||||
|
list_add(&state->list, &states);
|
||||||
|
spin_unlock_irqrestore(&state_lock, flags);
|
||||||
|
|
||||||
atomic_set(&state->refs, 1);
|
atomic_set(&state->refs, 1);
|
||||||
init_waitqueue_head(&state->wq);
|
init_waitqueue_head(&state->wq);
|
||||||
return state;
|
return state;
|
||||||
|
@ -115,10 +144,14 @@ EXPORT_SYMBOL(alloc_extent_state);
|
||||||
|
|
||||||
void free_extent_state(struct extent_state *state)
|
void free_extent_state(struct extent_state *state)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
if (!state)
|
if (!state)
|
||||||
return;
|
return;
|
||||||
if (atomic_dec_and_test(&state->refs)) {
|
if (atomic_dec_and_test(&state->refs)) {
|
||||||
WARN_ON(state->in_tree);
|
WARN_ON(state->in_tree);
|
||||||
|
spin_lock_irqsave(&state_lock, flags);
|
||||||
|
list_del(&state->list);
|
||||||
|
spin_unlock_irqrestore(&state_lock, flags);
|
||||||
kmem_cache_free(extent_state_cache, state);
|
kmem_cache_free(extent_state_cache, state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -361,10 +394,6 @@ static int insert_state(struct extent_map_tree *tree,
|
||||||
state->state |= bits;
|
state->state |= bits;
|
||||||
state->start = start;
|
state->start = start;
|
||||||
state->end = end;
|
state->end = end;
|
||||||
if ((end & 4095) == 0) {
|
|
||||||
printk("insert state %Lu %Lu strange end\n", start, end);
|
|
||||||
WARN_ON(1);
|
|
||||||
}
|
|
||||||
node = tree_insert(&tree->state, end, &state->rb_node);
|
node = tree_insert(&tree->state, end, &state->rb_node);
|
||||||
if (node) {
|
if (node) {
|
||||||
struct extent_state *found;
|
struct extent_state *found;
|
||||||
|
@ -399,11 +428,7 @@ static int split_state(struct extent_map_tree *tree, struct extent_state *orig,
|
||||||
prealloc->end = split - 1;
|
prealloc->end = split - 1;
|
||||||
prealloc->state = orig->state;
|
prealloc->state = orig->state;
|
||||||
orig->start = split;
|
orig->start = split;
|
||||||
if ((prealloc->end & 4095) == 0) {
|
|
||||||
printk("insert state %Lu %Lu strange end\n", prealloc->start,
|
|
||||||
prealloc->end);
|
|
||||||
WARN_ON(1);
|
|
||||||
}
|
|
||||||
node = tree_insert(&tree->state, prealloc->end, &prealloc->rb_node);
|
node = tree_insert(&tree->state, prealloc->end, &prealloc->rb_node);
|
||||||
if (node) {
|
if (node) {
|
||||||
struct extent_state *found;
|
struct extent_state *found;
|
||||||
|
@ -957,6 +982,7 @@ int find_first_extent_bit(struct extent_map_tree *tree, u64 start,
|
||||||
*start_ret = state->start;
|
*start_ret = state->start;
|
||||||
*end_ret = state->end;
|
*end_ret = state->end;
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
node = rb_next(node);
|
node = rb_next(node);
|
||||||
if (!node)
|
if (!node)
|
||||||
|
@ -1877,6 +1903,7 @@ sector_t extent_bmap(struct address_space *mapping, sector_t iblock,
|
||||||
static struct extent_buffer *__alloc_extent_buffer(gfp_t mask)
|
static struct extent_buffer *__alloc_extent_buffer(gfp_t mask)
|
||||||
{
|
{
|
||||||
struct extent_buffer *eb = NULL;
|
struct extent_buffer *eb = NULL;
|
||||||
|
|
||||||
spin_lock(&extent_buffers_lock);
|
spin_lock(&extent_buffers_lock);
|
||||||
if (!list_empty(&extent_buffers)) {
|
if (!list_empty(&extent_buffers)) {
|
||||||
eb = list_entry(extent_buffers.next, struct extent_buffer,
|
eb = list_entry(extent_buffers.next, struct extent_buffer,
|
||||||
|
@ -1886,15 +1913,26 @@ static struct extent_buffer *__alloc_extent_buffer(gfp_t mask)
|
||||||
nr_extent_buffers--;
|
nr_extent_buffers--;
|
||||||
}
|
}
|
||||||
spin_unlock(&extent_buffers_lock);
|
spin_unlock(&extent_buffers_lock);
|
||||||
|
|
||||||
if (eb) {
|
if (eb) {
|
||||||
memset(eb, 0, sizeof(*eb));
|
memset(eb, 0, sizeof(*eb));
|
||||||
return eb;
|
} else {
|
||||||
|
eb = kmem_cache_zalloc(extent_buffer_cache, mask);
|
||||||
}
|
}
|
||||||
return kmem_cache_zalloc(extent_buffer_cache, mask);
|
spin_lock(&extent_buffers_lock);
|
||||||
|
list_add(&eb->leak_list, &buffers);
|
||||||
|
spin_unlock(&extent_buffers_lock);
|
||||||
|
|
||||||
|
return eb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __free_extent_buffer(struct extent_buffer *eb)
|
static void __free_extent_buffer(struct extent_buffer *eb)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
spin_lock(&extent_buffers_lock);
|
||||||
|
list_del_init(&eb->leak_list);
|
||||||
|
spin_unlock(&extent_buffers_lock);
|
||||||
|
|
||||||
if (nr_extent_buffers >= MAX_EXTENT_BUFFER_CACHE) {
|
if (nr_extent_buffers >= MAX_EXTENT_BUFFER_CACHE) {
|
||||||
kmem_cache_free(extent_buffer_cache, eb);
|
kmem_cache_free(extent_buffer_cache, eb);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1933,6 +1971,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
|
||||||
if (!eb || IS_ERR(eb))
|
if (!eb || IS_ERR(eb))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
eb->alloc_addr = __builtin_return_address(0);
|
||||||
eb->start = start;
|
eb->start = start;
|
||||||
eb->len = len;
|
eb->len = len;
|
||||||
atomic_set(&eb->refs, 1);
|
atomic_set(&eb->refs, 1);
|
||||||
|
@ -1947,6 +1986,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
|
||||||
eb->start &= ~((u64)PAGE_CACHE_SIZE - 1);
|
eb->start &= ~((u64)PAGE_CACHE_SIZE - 1);
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
set_page_extent_mapped(p);
|
||||||
if (i == 0)
|
if (i == 0)
|
||||||
eb->first_page = p;
|
eb->first_page = p;
|
||||||
if (!PageUptodate(p))
|
if (!PageUptodate(p))
|
||||||
|
@ -1978,6 +2018,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
|
||||||
if (!eb || IS_ERR(eb))
|
if (!eb || IS_ERR(eb))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
eb->alloc_addr = __builtin_return_address(0);
|
||||||
eb->start = start;
|
eb->start = start;
|
||||||
eb->len = len;
|
eb->len = len;
|
||||||
atomic_set(&eb->refs, 1);
|
atomic_set(&eb->refs, 1);
|
||||||
|
@ -1992,6 +2033,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
|
||||||
eb->start &= ~((u64)PAGE_CACHE_SIZE - 1);
|
eb->start &= ~((u64)PAGE_CACHE_SIZE - 1);
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
set_page_extent_mapped(p);
|
||||||
if (i == 0)
|
if (i == 0)
|
||||||
eb->first_page = p;
|
eb->first_page = p;
|
||||||
}
|
}
|
||||||
|
|
|
@ -68,7 +68,9 @@ struct extent_buffer {
|
||||||
atomic_t refs;
|
atomic_t refs;
|
||||||
int flags;
|
int flags;
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
struct list_head leak_list;
|
||||||
struct page *first_page;
|
struct page *first_page;
|
||||||
|
unsigned long alloc_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct extent_map *(get_extent_t)(struct inode *inode,
|
typedef struct extent_map *(get_extent_t)(struct inode *inode,
|
||||||
|
|
|
@ -352,7 +352,7 @@ int btrfs_defrag_root(struct btrfs_root *root, int cacheonly)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
trans = btrfs_start_transaction(root, 1);
|
trans = btrfs_start_transaction(root, 1);
|
||||||
while (1) {
|
while (0) {
|
||||||
root->defrag_running = 1;
|
root->defrag_running = 1;
|
||||||
ret = btrfs_defrag_leaves(trans, root, cacheonly);
|
ret = btrfs_defrag_leaves(trans, root, cacheonly);
|
||||||
nr = trans->blocks_used;
|
nr = trans->blocks_used;
|
||||||
|
@ -394,7 +394,7 @@ int btrfs_defrag_dirty_roots(struct btrfs_fs_info *info)
|
||||||
for (i = 0; i < ret; i++) {
|
for (i = 0; i < ret; i++) {
|
||||||
root = gang[i];
|
root = gang[i];
|
||||||
last = root->root_key.objectid + 1;
|
last = root->root_key.objectid + 1;
|
||||||
// btrfs_defrag_root(root, 1);
|
btrfs_defrag_root(root, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// btrfs_defrag_root(info->extent_root, 1);
|
// btrfs_defrag_root(info->extent_root, 1);
|
||||||
|
@ -462,6 +462,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
|
||||||
ret = btrfs_end_transaction(trans, tree_root);
|
ret = btrfs_end_transaction(trans, tree_root);
|
||||||
BUG_ON(ret);
|
BUG_ON(ret);
|
||||||
|
|
||||||
|
free_extent_buffer(dirty->root->node);
|
||||||
kfree(dirty->root);
|
kfree(dirty->root);
|
||||||
kfree(dirty);
|
kfree(dirty);
|
||||||
mutex_unlock(&tree_root->fs_info->fs_mutex);
|
mutex_unlock(&tree_root->fs_info->fs_mutex);
|
||||||
|
|
Loading…
Reference in New Issue