2007-06-12 09:07:21 -04:00
/*
* Copyright ( C ) 2007 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 .
*/
2007-07-11 10:00:37 -04:00
# include <linux/sched.h>
2007-02-26 10:40:21 -05:00
# include "ctree.h"
# include "disk-io.h"
# include "print-tree.h"
2007-03-16 16:20:31 -04:00
# include "transaction.h"
2007-02-26 10:40:21 -05:00
2007-03-16 16:20:31 -04:00
static int finish_current_insert ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root ) ;
2007-03-22 12:13:20 -04:00
static int del_pending_extents ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root ) ;
2007-02-26 10:40:21 -05:00
2007-05-09 20:13:14 -04:00
static int cache_block_group ( struct btrfs_root * root ,
struct btrfs_block_group_cache * block_group )
{
struct btrfs_path * path ;
int ret ;
struct btrfs_key key ;
struct btrfs_leaf * leaf ;
struct radix_tree_root * extent_radix ;
int slot ;
u64 i ;
u64 last = 0 ;
u64 hole_size ;
2007-09-14 16:15:28 -04:00
u64 first_free ;
2007-05-09 20:13:14 -04:00
int found = 0 ;
root = root - > fs_info - > extent_root ;
extent_radix = & root - > fs_info - > extent_map_radix ;
if ( block_group - > cached )
return 0 ;
if ( block_group - > data )
return 0 ;
path = btrfs_alloc_path ( ) ;
if ( ! path )
return - ENOMEM ;
2007-09-14 16:15:28 -04:00
2007-08-27 16:49:44 -04:00
path - > reada = 2 ;
2007-09-14 16:15:28 -04:00
first_free = block_group - > key . objectid ;
2007-05-09 20:13:14 -04:00
key . objectid = block_group - > key . objectid ;
key . flags = 0 ;
key . offset = 0 ;
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
ret = btrfs_search_slot ( NULL , root , & key , path , 0 , 0 ) ;
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
if ( ret < 0 )
return ret ;
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
if ( ret & & path - > slots [ 0 ] > 0 )
path - > slots [ 0 ] - - ;
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
while ( 1 ) {
leaf = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
slot = path - > slots [ 0 ] ;
if ( slot > = btrfs_header_nritems ( & leaf - > header ) ) {
ret = btrfs_next_leaf ( root , path ) ;
2007-06-22 14:16:25 -04:00
if ( ret < 0 )
goto err ;
2007-05-18 13:28:27 -04:00
if ( ret = = 0 ) {
2007-05-09 20:13:14 -04:00
continue ;
2007-05-18 13:28:27 -04:00
} else {
2007-05-09 20:13:14 -04:00
break ;
}
}
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
btrfs_disk_key_to_cpu ( & key , & leaf - > items [ slot ] . key ) ;
2007-09-14 16:15:28 -04:00
if ( key . objectid < block_group - > key . objectid ) {
if ( key . objectid + key . offset > first_free )
first_free = key . objectid + key . offset ;
goto next ;
}
2007-05-09 20:13:14 -04:00
if ( key . objectid > = block_group - > key . objectid +
block_group - > key . offset ) {
break ;
}
2007-09-14 16:15:28 -04:00
2007-05-09 20:13:14 -04:00
if ( btrfs_key_type ( & key ) = = BTRFS_EXTENT_ITEM_KEY ) {
if ( ! found ) {
2007-09-14 16:15:28 -04:00
last = first_free ;
2007-05-09 20:13:14 -04:00
found = 1 ;
}
2007-09-14 16:15:28 -04:00
hole_size = key . objectid - last ;
for ( i = 0 ; i < hole_size ; i + + ) {
set_radix_bit ( extent_radix , last + i ) ;
}
last = key . objectid + key . offset ;
2007-05-09 20:13:14 -04:00
}
2007-09-14 16:15:28 -04:00
next :
2007-05-09 20:13:14 -04:00
path - > slots [ 0 ] + + ;
}
2007-09-14 16:15:28 -04:00
if ( ! found )
last = first_free ;
if ( block_group - > key . objectid +
block_group - > key . offset > last ) {
hole_size = block_group - > key . objectid +
block_group - > key . offset - last ;
for ( i = 0 ; i < hole_size ; i + + ) {
set_radix_bit ( extent_radix ,
last + i ) ;
}
}
2007-05-09 20:13:14 -04:00
block_group - > cached = 1 ;
2007-06-22 14:16:25 -04:00
err :
2007-05-09 20:13:14 -04:00
btrfs_free_path ( path ) ;
return 0 ;
}
2007-06-11 21:33:38 -04:00
struct btrfs_block_group_cache * btrfs_lookup_block_group ( struct
btrfs_fs_info * info ,
u64 blocknr )
2007-05-06 10:15:01 -04:00
{
struct btrfs_block_group_cache * block_group ;
int ret ;
ret = radix_tree_gang_lookup ( & info - > block_group_radix ,
( void * * ) & block_group ,
blocknr , 1 ) ;
if ( ret ) {
2007-05-07 20:03:49 -04:00
if ( block_group - > key . objectid < = blocknr & & blocknr < =
2007-05-06 10:15:01 -04:00
block_group - > key . objectid + block_group - > key . offset )
return block_group ;
}
ret = radix_tree_gang_lookup ( & info - > block_group_data_radix ,
( void * * ) & block_group ,
blocknr , 1 ) ;
if ( ret ) {
2007-05-07 20:03:49 -04:00
if ( block_group - > key . objectid < = blocknr & & blocknr < =
2007-05-06 10:15:01 -04:00
block_group - > key . objectid + block_group - > key . offset )
return block_group ;
}
return NULL ;
}
2007-05-09 20:13:14 -04:00
static u64 leaf_range ( struct btrfs_root * root )
{
u64 size = BTRFS_LEAF_DATA_SIZE ( root ) ;
2007-06-12 07:43:08 -04:00
do_div ( size , sizeof ( struct btrfs_extent_item ) +
sizeof ( struct btrfs_item ) ) ;
2007-05-09 20:13:14 -04:00
return size ;
}
static u64 find_search_start ( struct btrfs_root * root ,
struct btrfs_block_group_cache * * cache_ret ,
u64 search_start , int num )
{
unsigned long gang [ 8 ] ;
int ret ;
struct btrfs_block_group_cache * cache = * cache_ret ;
u64 last = max ( search_start , cache - > key . objectid ) ;
if ( cache - > data )
goto out ;
again :
2007-06-22 14:16:25 -04:00
ret = cache_block_group ( root , cache ) ;
if ( ret )
goto out ;
2007-05-09 20:13:14 -04:00
while ( 1 ) {
ret = find_first_radix_bit ( & root - > fs_info - > extent_map_radix ,
gang , last , ARRAY_SIZE ( gang ) ) ;
if ( ! ret )
goto out ;
last = gang [ ret - 1 ] + 1 ;
if ( num > 1 ) {
if ( ret ! = ARRAY_SIZE ( gang ) ) {
goto new_group ;
}
if ( gang [ ret - 1 ] - gang [ 0 ] > leaf_range ( root ) ) {
continue ;
}
}
if ( gang [ 0 ] > = cache - > key . objectid + cache - > key . offset ) {
goto new_group ;
}
return gang [ 0 ] ;
}
out :
return max ( cache - > last_alloc , search_start ) ;
new_group :
2007-06-11 21:33:38 -04:00
cache = btrfs_lookup_block_group ( root - > fs_info ,
last + cache - > key . offset - 1 ) ;
2007-05-09 20:13:14 -04:00
if ( ! cache ) {
return max ( ( * cache_ret ) - > last_alloc , search_start ) ;
}
cache = btrfs_find_block_group ( root , cache ,
2007-05-18 13:28:27 -04:00
last + cache - > key . offset - 1 , 0 , 0 ) ;
2007-05-09 20:13:14 -04:00
* cache_ret = cache ;
goto again ;
}
2007-06-12 07:43:08 -04:00
static u64 div_factor ( u64 num , int factor )
{
num * = factor ;
do_div ( num , 10 ) ;
return num ;
}
2007-04-30 15:25:45 -04:00
struct btrfs_block_group_cache * btrfs_find_block_group ( struct btrfs_root * root ,
struct btrfs_block_group_cache
2007-05-06 10:15:01 -04:00
* hint , u64 search_start ,
2007-05-18 13:28:27 -04:00
int data , int owner )
2007-04-27 10:08:34 -04:00
{
struct btrfs_block_group_cache * cache [ 8 ] ;
2007-04-30 15:25:45 -04:00
struct btrfs_block_group_cache * found_group = NULL ;
2007-04-27 10:08:34 -04:00
struct btrfs_fs_info * info = root - > fs_info ;
2007-05-06 10:15:01 -04:00
struct radix_tree_root * radix ;
2007-05-29 16:52:18 -04:00
struct radix_tree_root * swap_radix ;
2007-04-27 10:08:34 -04:00
u64 used ;
2007-04-30 15:25:45 -04:00
u64 last = 0 ;
u64 hint_last ;
2007-04-27 10:08:34 -04:00
int i ;
int ret ;
2007-04-30 15:25:45 -04:00
int full_search = 0 ;
2007-05-18 13:28:27 -04:00
int factor = 8 ;
2007-05-29 16:52:18 -04:00
int data_swap = 0 ;
2007-05-18 13:28:27 -04:00
if ( ! owner )
factor = 5 ;
2007-05-06 10:15:01 -04:00
2007-05-29 16:52:18 -04:00
if ( data ) {
2007-05-06 10:15:01 -04:00
radix = & info - > block_group_data_radix ;
2007-05-29 16:52:18 -04:00
swap_radix = & info - > block_group_radix ;
} else {
2007-05-06 10:15:01 -04:00
radix = & info - > block_group_radix ;
2007-05-29 16:52:18 -04:00
swap_radix = & info - > block_group_data_radix ;
}
2007-05-06 10:15:01 -04:00
if ( search_start ) {
struct btrfs_block_group_cache * shint ;
2007-06-11 21:33:38 -04:00
shint = btrfs_lookup_block_group ( info , search_start ) ;
2007-08-29 09:11:44 -04:00
if ( shint & & shint - > data = = data ) {
2007-05-06 10:15:01 -04:00
used = btrfs_block_group_used ( & shint - > item ) ;
if ( used + shint - > pinned <
2007-06-12 07:43:08 -04:00
div_factor ( shint - > key . offset , factor ) ) {
2007-05-06 10:15:01 -04:00
return shint ;
}
}
}
if ( hint & & hint - > data = = data ) {
2007-04-30 15:25:45 -04:00
used = btrfs_block_group_used ( & hint - > item ) ;
2007-06-12 07:43:08 -04:00
if ( used + hint - > pinned <
div_factor ( hint - > key . offset , factor ) ) {
2007-04-30 15:25:45 -04:00
return hint ;
}
2007-06-12 07:43:08 -04:00
if ( used > = div_factor ( hint - > key . offset , 8 ) ) {
2007-05-06 10:15:01 -04:00
radix_tree_tag_clear ( radix ,
hint - > key . objectid +
hint - > key . offset - 1 ,
BTRFS_BLOCK_GROUP_AVAIL ) ;
}
2007-05-10 11:24:42 -04:00
last = hint - > key . offset * 3 ;
2007-05-06 10:15:01 -04:00
if ( hint - > key . objectid > = last )
2007-05-09 20:13:14 -04:00
last = max ( search_start + hint - > key . offset - 1 ,
hint - > key . objectid - last ) ;
2007-05-06 10:15:01 -04:00
else
last = hint - > key . objectid + hint - > key . offset ;
2007-04-30 15:25:45 -04:00
hint_last = last ;
} else {
2007-05-09 20:13:14 -04:00
if ( hint )
hint_last = max ( hint - > key . objectid , search_start ) ;
else
hint_last = search_start ;
last = hint_last ;
2007-04-30 15:25:45 -04:00
}
2007-04-27 10:08:34 -04:00
while ( 1 ) {
2007-05-06 10:15:01 -04:00
ret = radix_tree_gang_lookup_tag ( radix , ( void * * ) cache ,
2007-04-27 10:08:34 -04:00
last , ARRAY_SIZE ( cache ) ,
2007-04-30 15:25:45 -04:00
BTRFS_BLOCK_GROUP_AVAIL ) ;
2007-04-27 10:08:34 -04:00
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
2007-05-03 09:06:49 -04:00
last = cache [ i ] - > key . objectid +
cache [ i ] - > key . offset ;
2007-04-27 10:08:34 -04:00
used = btrfs_block_group_used ( & cache [ i ] - > item ) ;
2007-05-06 10:15:01 -04:00
if ( used + cache [ i ] - > pinned <
2007-06-12 07:43:08 -04:00
div_factor ( cache [ i ] - > key . offset , factor ) ) {
2007-04-30 15:25:45 -04:00
found_group = cache [ i ] ;
goto found ;
2007-04-27 10:08:34 -04:00
}
2007-06-12 07:43:08 -04:00
if ( used > = div_factor ( cache [ i ] - > key . offset , 8 ) ) {
2007-05-06 10:15:01 -04:00
radix_tree_tag_clear ( radix ,
cache [ i ] - > key . objectid +
cache [ i ] - > key . offset - 1 ,
BTRFS_BLOCK_GROUP_AVAIL ) ;
}
2007-04-27 10:08:34 -04:00
}
2007-05-18 13:28:27 -04:00
cond_resched ( ) ;
2007-04-27 10:08:34 -04:00
}
2007-04-30 15:25:45 -04:00
last = hint_last ;
again :
2007-04-27 10:08:34 -04:00
while ( 1 ) {
2007-05-06 10:15:01 -04:00
ret = radix_tree_gang_lookup ( radix , ( void * * ) cache ,
last , ARRAY_SIZE ( cache ) ) ;
2007-04-27 10:08:34 -04:00
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
2007-05-03 09:06:49 -04:00
last = cache [ i ] - > key . objectid +
cache [ i ] - > key . offset ;
2007-04-27 10:08:34 -04:00
used = btrfs_block_group_used ( & cache [ i ] - > item ) ;
2007-05-06 10:15:01 -04:00
if ( used + cache [ i ] - > pinned < cache [ i ] - > key . offset ) {
2007-04-30 15:25:45 -04:00
found_group = cache [ i ] ;
goto found ;
2007-04-27 10:08:34 -04:00
}
2007-05-06 10:15:01 -04:00
if ( used > = cache [ i ] - > key . offset ) {
radix_tree_tag_clear ( radix ,
cache [ i ] - > key . objectid +
cache [ i ] - > key . offset - 1 ,
BTRFS_BLOCK_GROUP_AVAIL ) ;
}
2007-04-27 10:08:34 -04:00
}
2007-05-18 13:28:27 -04:00
cond_resched ( ) ;
2007-04-27 10:08:34 -04:00
}
2007-04-30 15:25:45 -04:00
if ( ! full_search ) {
2007-05-06 10:15:01 -04:00
last = search_start ;
2007-04-30 15:25:45 -04:00
full_search = 1 ;
goto again ;
}
2007-05-29 16:52:18 -04:00
if ( ! data_swap ) {
struct radix_tree_root * tmp = radix ;
data_swap = 1 ;
radix = swap_radix ;
swap_radix = tmp ;
last = search_start ;
goto again ;
}
2007-04-30 15:25:45 -04:00
if ( ! found_group ) {
2007-05-06 10:15:01 -04:00
ret = radix_tree_gang_lookup ( radix ,
2007-04-30 15:25:45 -04:00
( void * * ) & found_group , 0 , 1 ) ;
2007-05-29 16:52:18 -04:00
if ( ret = = 0 ) {
ret = radix_tree_gang_lookup ( swap_radix ,
( void * * ) & found_group ,
0 , 1 ) ;
}
2007-04-30 15:25:45 -04:00
BUG_ON ( ret ! = 1 ) ;
}
2007-05-06 10:15:01 -04:00
found :
2007-04-30 15:25:45 -04:00
return found_group ;
2007-04-27 10:08:34 -04:00
}
2007-04-17 13:26:50 -04:00
int btrfs_inc_extent_ref ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
u64 blocknr , u64 num_blocks )
2007-03-02 16:08:05 -05:00
{
2007-04-02 11:20:42 -04:00
struct btrfs_path * path ;
2007-03-02 16:08:05 -05:00
int ret ;
2007-03-12 16:22:34 -04:00
struct btrfs_key key ;
2007-03-13 10:46:10 -04:00
struct btrfs_leaf * l ;
struct btrfs_extent_item * item ;
2007-03-13 09:49:06 -04:00
u32 refs ;
2007-03-07 11:50:24 -05:00
2007-04-02 11:20:42 -04:00
path = btrfs_alloc_path ( ) ;
2007-06-22 14:16:25 -04:00
if ( ! path )
return - ENOMEM ;
2007-08-08 20:17:12 -04:00
2007-03-02 16:08:05 -05:00
key . objectid = blocknr ;
key . flags = 0 ;
2007-03-15 12:56:47 -04:00
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-27 06:33:00 -04:00
key . offset = num_blocks ;
2007-04-02 11:20:42 -04:00
ret = btrfs_search_slot ( trans , root - > fs_info - > extent_root , & key , path ,
2007-03-20 14:38:32 -04:00
0 , 1 ) ;
2007-06-22 14:16:25 -04:00
if ( ret < 0 )
return ret ;
2007-04-18 16:15:28 -04:00
if ( ret ! = 0 ) {
2007-03-06 20:08:01 -05:00
BUG ( ) ;
2007-04-18 16:15:28 -04:00
}
2007-03-02 16:08:05 -05:00
BUG_ON ( ret ! = 0 ) ;
2007-04-02 11:20:42 -04:00
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
item = btrfs_item_ptr ( l , path - > slots [ 0 ] , struct btrfs_extent_item ) ;
2007-03-13 09:49:06 -04:00
refs = btrfs_extent_refs ( item ) ;
btrfs_set_extent_refs ( item , refs + 1 ) ;
2007-04-02 11:20:42 -04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
2007-03-06 20:08:01 -05:00
2007-04-02 11:20:42 -04:00
btrfs_release_path ( root - > fs_info - > extent_root , path ) ;
btrfs_free_path ( path ) ;
2007-03-20 14:38:32 -04:00
finish_current_insert ( trans , root - > fs_info - > extent_root ) ;
2007-03-22 12:13:20 -04:00
del_pending_extents ( trans , root - > fs_info - > extent_root ) ;
2007-03-02 16:08:05 -05:00
return 0 ;
}
2007-08-10 14:06:19 -04:00
int btrfs_extent_post_op ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root )
{
finish_current_insert ( trans , root - > fs_info - > extent_root ) ;
del_pending_extents ( trans , root - > fs_info - > extent_root ) ;
return 0 ;
}
2007-04-17 13:26:50 -04:00
static int lookup_extent_ref ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root , u64 blocknr ,
u64 num_blocks , u32 * refs )
2007-03-06 20:08:01 -05:00
{
2007-04-02 11:20:42 -04:00
struct btrfs_path * path ;
2007-03-06 20:08:01 -05:00
int ret ;
2007-03-12 16:22:34 -04:00
struct btrfs_key key ;
2007-03-13 10:46:10 -04:00
struct btrfs_leaf * l ;
struct btrfs_extent_item * item ;
2007-04-02 11:20:42 -04:00
path = btrfs_alloc_path ( ) ;
2007-03-06 20:08:01 -05:00
key . objectid = blocknr ;
2007-03-27 06:33:00 -04:00
key . offset = num_blocks ;
2007-03-15 12:56:47 -04:00
key . flags = 0 ;
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-04-02 11:20:42 -04:00
ret = btrfs_search_slot ( trans , root - > fs_info - > extent_root , & key , path ,
2007-03-20 14:38:32 -04:00
0 , 0 ) ;
2007-06-22 14:16:25 -04:00
if ( ret < 0 )
goto out ;
2007-03-06 20:08:01 -05:00
if ( ret ! = 0 )
BUG ( ) ;
2007-04-02 11:20:42 -04:00
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
item = btrfs_item_ptr ( l , path - > slots [ 0 ] , struct btrfs_extent_item ) ;
2007-03-13 09:49:06 -04:00
* refs = btrfs_extent_refs ( item ) ;
2007-06-22 14:16:25 -04:00
out :
2007-04-02 11:20:42 -04:00
btrfs_free_path ( path ) ;
2007-03-06 20:08:01 -05:00
return 0 ;
}
2007-04-10 09:27:04 -04:00
int btrfs_inc_root_ref ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root )
{
2007-04-17 13:26:50 -04:00
return btrfs_inc_extent_ref ( trans , root , bh_blocknr ( root - > node ) , 1 ) ;
2007-04-10 09:27:04 -04:00
}
2007-03-16 16:20:31 -04:00
int btrfs_inc_ref ( struct btrfs_trans_handle * trans , struct btrfs_root * root ,
2007-03-22 12:13:20 -04:00
struct buffer_head * buf )
2007-03-02 16:08:05 -05:00
{
u64 blocknr ;
2007-03-22 12:13:20 -04:00
struct btrfs_node * buf_node ;
2007-03-27 06:33:00 -04:00
struct btrfs_leaf * buf_leaf ;
struct btrfs_disk_key * key ;
struct btrfs_file_extent_item * fi ;
2007-03-02 16:08:05 -05:00
int i ;
2007-03-27 06:33:00 -04:00
int leaf ;
int ret ;
2007-06-22 14:16:25 -04:00
int faili ;
int err ;
2007-03-06 20:08:01 -05:00
2007-03-13 16:47:54 -04:00
if ( ! root - > ref_cows )
2007-03-06 20:08:01 -05:00
return 0 ;
2007-03-22 12:13:20 -04:00
buf_node = btrfs_buffer_node ( buf ) ;
2007-03-27 06:33:00 -04:00
leaf = btrfs_is_leaf ( buf_node ) ;
buf_leaf = btrfs_buffer_leaf ( buf ) ;
2007-03-22 12:13:20 -04:00
for ( i = 0 ; i < btrfs_header_nritems ( & buf_node - > header ) ; i + + ) {
2007-03-27 06:33:00 -04:00
if ( leaf ) {
2007-05-24 13:35:57 -04:00
u64 disk_blocknr ;
2007-03-27 06:33:00 -04:00
key = & buf_leaf - > items [ i ] . key ;
if ( btrfs_disk_key_type ( key ) ! = BTRFS_EXTENT_DATA_KEY )
continue ;
fi = btrfs_item_ptr ( buf_leaf , i ,
struct btrfs_file_extent_item ) ;
2007-04-19 13:37:44 -04:00
if ( btrfs_file_extent_type ( fi ) = =
BTRFS_FILE_EXTENT_INLINE )
continue ;
2007-05-24 13:35:57 -04:00
disk_blocknr = btrfs_file_extent_disk_blocknr ( fi ) ;
if ( disk_blocknr = = 0 )
continue ;
ret = btrfs_inc_extent_ref ( trans , root , disk_blocknr ,
2007-03-27 06:33:00 -04:00
btrfs_file_extent_disk_num_blocks ( fi ) ) ;
2007-06-22 14:16:25 -04:00
if ( ret ) {
faili = i ;
goto fail ;
}
2007-03-27 06:33:00 -04:00
} else {
blocknr = btrfs_node_blockptr ( buf_node , i ) ;
2007-04-17 13:26:50 -04:00
ret = btrfs_inc_extent_ref ( trans , root , blocknr , 1 ) ;
2007-06-22 14:16:25 -04:00
if ( ret ) {
faili = i ;
goto fail ;
}
2007-03-27 06:33:00 -04:00
}
2007-03-02 16:08:05 -05:00
}
return 0 ;
2007-06-22 14:16:25 -04:00
fail :
2007-06-28 15:57:36 -04:00
WARN_ON ( 1 ) ;
2007-06-22 14:16:25 -04:00
for ( i = 0 ; i < faili ; i + + ) {
if ( leaf ) {
u64 disk_blocknr ;
key = & buf_leaf - > items [ i ] . key ;
if ( btrfs_disk_key_type ( key ) ! = BTRFS_EXTENT_DATA_KEY )
continue ;
fi = btrfs_item_ptr ( buf_leaf , i ,
struct btrfs_file_extent_item ) ;
if ( btrfs_file_extent_type ( fi ) = =
BTRFS_FILE_EXTENT_INLINE )
continue ;
disk_blocknr = btrfs_file_extent_disk_blocknr ( fi ) ;
if ( disk_blocknr = = 0 )
continue ;
err = btrfs_free_extent ( trans , root , disk_blocknr ,
btrfs_file_extent_disk_num_blocks ( fi ) , 0 ) ;
BUG_ON ( err ) ;
} else {
blocknr = btrfs_node_blockptr ( buf_node , i ) ;
err = btrfs_free_extent ( trans , root , blocknr , 1 , 0 ) ;
BUG_ON ( err ) ;
}
}
return ret ;
2007-03-02 16:08:05 -05:00
}
2007-04-26 16:46:15 -04:00
static int write_one_cache_group ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
struct btrfs_path * path ,
struct btrfs_block_group_cache * cache )
{
int ret ;
int pending_ret ;
struct btrfs_root * extent_root = root - > fs_info - > extent_root ;
struct btrfs_block_group_item * bi ;
ret = btrfs_search_slot ( trans , extent_root , & cache - > key , path , 0 , 1 ) ;
2007-06-22 14:16:25 -04:00
if ( ret < 0 )
goto fail ;
2007-04-26 16:46:15 -04:00
BUG_ON ( ret ) ;
bi = btrfs_item_ptr ( btrfs_buffer_leaf ( path - > nodes [ 0 ] ) , path - > slots [ 0 ] ,
struct btrfs_block_group_item ) ;
memcpy ( bi , & cache - > item , sizeof ( * bi ) ) ;
2007-06-28 15:57:36 -04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
2007-04-26 16:46:15 -04:00
btrfs_release_path ( extent_root , path ) ;
2007-06-22 14:16:25 -04:00
fail :
2007-04-26 16:46:15 -04:00
finish_current_insert ( trans , extent_root ) ;
pending_ret = del_pending_extents ( trans , extent_root ) ;
if ( ret )
return ret ;
if ( pending_ret )
return pending_ret ;
2007-05-06 10:15:01 -04:00
if ( cache - > data )
cache - > last_alloc = cache - > first_free ;
2007-04-26 16:46:15 -04:00
return 0 ;
}
2007-05-06 10:15:01 -04:00
static int write_dirty_block_radix ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
struct radix_tree_root * radix )
2007-04-26 16:46:15 -04:00
{
struct btrfs_block_group_cache * cache [ 8 ] ;
int ret ;
int err = 0 ;
int werr = 0 ;
int i ;
struct btrfs_path * path ;
2007-06-22 14:16:25 -04:00
unsigned long off = 0 ;
2007-04-26 16:46:15 -04:00
path = btrfs_alloc_path ( ) ;
if ( ! path )
return - ENOMEM ;
while ( 1 ) {
ret = radix_tree_gang_lookup_tag ( radix , ( void * * ) cache ,
2007-06-22 14:16:25 -04:00
off , ARRAY_SIZE ( cache ) ,
2007-04-26 16:46:15 -04:00
BTRFS_BLOCK_GROUP_DIRTY ) ;
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
err = write_one_cache_group ( trans , root ,
path , cache [ i ] ) ;
2007-06-22 14:16:25 -04:00
/*
* if we fail to write the cache group , we want
* to keep it marked dirty in hopes that a later
* write will work
*/
if ( err ) {
2007-04-26 16:46:15 -04:00
werr = err ;
2007-06-22 14:16:25 -04:00
off = cache [ i ] - > key . objectid +
cache [ i ] - > key . offset ;
continue ;
}
radix_tree_tag_clear ( radix , cache [ i ] - > key . objectid +
cache [ i ] - > key . offset - 1 ,
BTRFS_BLOCK_GROUP_DIRTY ) ;
2007-04-26 16:46:15 -04:00
}
}
btrfs_free_path ( path ) ;
return werr ;
}
2007-05-06 10:15:01 -04:00
int btrfs_write_dirty_block_groups ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root )
{
int ret ;
int ret2 ;
ret = write_dirty_block_radix ( trans , root ,
& root - > fs_info - > block_group_radix ) ;
ret2 = write_dirty_block_radix ( trans , root ,
& root - > fs_info - > block_group_data_radix ) ;
if ( ret )
return ret ;
if ( ret2 )
return ret2 ;
return 0 ;
}
2007-04-26 16:46:15 -04:00
static int update_block_group ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
2007-05-29 16:52:18 -04:00
u64 blocknr , u64 num , int alloc , int mark_free ,
int data )
2007-04-26 16:46:15 -04:00
{
struct btrfs_block_group_cache * cache ;
struct btrfs_fs_info * info = root - > fs_info ;
u64 total = num ;
u64 old_val ;
u64 block_in_group ;
2007-05-09 20:13:14 -04:00
u64 i ;
2007-05-29 16:52:18 -04:00
int ret ;
2007-05-07 20:03:49 -04:00
2007-04-26 16:46:15 -04:00
while ( total ) {
2007-06-11 21:33:38 -04:00
cache = btrfs_lookup_block_group ( info , blocknr ) ;
2007-05-07 20:03:49 -04:00
if ( ! cache ) {
2007-04-26 16:46:15 -04:00
return - 1 ;
2007-04-27 10:08:34 -04:00
}
2007-04-26 16:46:15 -04:00
block_in_group = blocknr - cache - > key . objectid ;
WARN_ON ( block_in_group > cache - > key . offset ) ;
2007-05-07 20:03:49 -04:00
radix_tree_tag_set ( cache - > radix , cache - > key . objectid +
2007-05-06 10:15:01 -04:00
cache - > key . offset - 1 ,
2007-04-26 16:46:15 -04:00
BTRFS_BLOCK_GROUP_DIRTY ) ;
old_val = btrfs_block_group_used ( & cache - > item ) ;
num = min ( total , cache - > key . offset - block_in_group ) ;
2007-04-27 10:08:34 -04:00
if ( alloc ) {
if ( blocknr > cache - > last_alloc )
cache - > last_alloc = blocknr ;
2007-05-09 20:13:14 -04:00
if ( ! cache - > data ) {
for ( i = 0 ; i < num ; i + + ) {
clear_radix_bit ( & info - > extent_map_radix ,
blocknr + i ) ;
}
}
2007-05-29 16:52:18 -04:00
if ( cache - > data ! = data & &
2007-06-12 07:43:08 -04:00
old_val < ( cache - > key . offset > > 1 ) ) {
2007-05-29 16:52:18 -04:00
cache - > data = data ;
radix_tree_delete ( cache - > radix ,
cache - > key . objectid +
cache - > key . offset - 1 ) ;
if ( data ) {
cache - > radix =
& info - > block_group_data_radix ;
cache - > item . flags | =
BTRFS_BLOCK_GROUP_DATA ;
} else {
cache - > radix = & info - > block_group_radix ;
cache - > item . flags & =
~ BTRFS_BLOCK_GROUP_DATA ;
}
ret = radix_tree_insert ( cache - > radix ,
cache - > key . objectid +
cache - > key . offset - 1 ,
( void * ) cache ) ;
}
old_val + = num ;
2007-04-27 10:08:34 -04:00
} else {
2007-04-26 16:46:15 -04:00
old_val - = num ;
2007-04-27 10:08:34 -04:00
if ( blocknr < cache - > first_free )
cache - > first_free = blocknr ;
2007-05-09 20:13:14 -04:00
if ( ! cache - > data & & mark_free ) {
for ( i = 0 ; i < num ; i + + ) {
set_radix_bit ( & info - > extent_map_radix ,
blocknr + i ) ;
}
}
2007-06-12 07:43:08 -04:00
if ( old_val < ( cache - > key . offset > > 1 ) & &
old_val + num > = ( cache - > key . offset > > 1 ) ) {
2007-05-09 20:13:14 -04:00
radix_tree_tag_set ( cache - > radix ,
cache - > key . objectid +
cache - > key . offset - 1 ,
BTRFS_BLOCK_GROUP_AVAIL ) ;
}
2007-04-27 10:08:34 -04:00
}
2007-04-26 16:46:15 -04:00
btrfs_set_block_group_used ( & cache - > item , old_val ) ;
2007-05-09 20:13:14 -04:00
total - = num ;
blocknr + = num ;
2007-04-26 16:46:15 -04:00
}
return 0 ;
}
2007-06-28 15:57:36 -04:00
int btrfs_copy_pinned ( struct btrfs_root * root , struct radix_tree_root * copy )
{
unsigned long gang [ 8 ] ;
u64 last = 0 ;
struct radix_tree_root * pinned_radix = & root - > fs_info - > pinned_radix ;
int ret ;
int i ;
while ( 1 ) {
ret = find_first_radix_bit ( pinned_radix , gang , last ,
ARRAY_SIZE ( gang ) ) ;
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
set_radix_bit ( copy , gang [ i ] ) ;
last = gang [ i ] + 1 ;
}
}
2007-08-08 20:17:12 -04:00
ret = find_first_radix_bit ( & root - > fs_info - > extent_ins_radix , gang , 0 ,
ARRAY_SIZE ( gang ) ) ;
WARN_ON ( ret ) ;
2007-06-28 15:57:36 -04:00
return 0 ;
}
int btrfs_finish_extent_commit ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
struct radix_tree_root * unpin_radix )
2007-03-06 20:08:01 -05:00
{
2007-03-26 10:15:30 -04:00
unsigned long gang [ 8 ] ;
2007-05-06 10:15:01 -04:00
struct btrfs_block_group_cache * block_group ;
2007-03-16 08:56:18 -04:00
u64 first = 0 ;
2007-03-06 20:08:01 -05:00
int ret ;
int i ;
2007-03-26 10:15:30 -04:00
struct radix_tree_root * pinned_radix = & root - > fs_info - > pinned_radix ;
2007-05-09 20:13:14 -04:00
struct radix_tree_root * extent_radix = & root - > fs_info - > extent_map_radix ;
2007-03-06 20:08:01 -05:00
while ( 1 ) {
2007-06-28 15:57:36 -04:00
ret = find_first_radix_bit ( unpin_radix , gang , 0 ,
2007-03-26 10:15:30 -04:00
ARRAY_SIZE ( gang ) ) ;
2007-03-06 20:08:01 -05:00
if ( ! ret )
break ;
2007-03-16 08:56:18 -04:00
if ( ! first )
2007-03-26 10:15:30 -04:00
first = gang [ 0 ] ;
2007-03-07 16:15:30 -05:00
for ( i = 0 ; i < ret ; i + + ) {
2007-03-26 10:15:30 -04:00
clear_radix_bit ( pinned_radix , gang [ i ] ) ;
2007-06-28 15:57:36 -04:00
clear_radix_bit ( unpin_radix , gang [ i ] ) ;
2007-06-11 21:33:38 -04:00
block_group = btrfs_lookup_block_group ( root - > fs_info ,
gang [ i ] ) ;
2007-05-06 10:15:01 -04:00
if ( block_group ) {
WARN_ON ( block_group - > pinned = = 0 ) ;
block_group - > pinned - - ;
if ( gang [ i ] < block_group - > last_alloc )
block_group - > last_alloc = gang [ i ] ;
2007-05-09 20:13:14 -04:00
if ( ! block_group - > data )
set_radix_bit ( extent_radix , gang [ i ] ) ;
2007-05-06 10:15:01 -04:00
}
2007-03-07 16:15:30 -05:00
}
2007-03-06 20:08:01 -05:00
}
return 0 ;
}
2007-03-16 16:20:31 -04:00
static int finish_current_insert ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root )
2007-03-07 11:50:24 -05:00
{
2007-03-12 16:22:34 -04:00
struct btrfs_key ins ;
2007-03-13 10:46:10 -04:00
struct btrfs_extent_item extent_item ;
2007-03-07 11:50:24 -05:00
int i ;
int ret ;
2007-08-08 20:17:12 -04:00
int err ;
unsigned long gang [ 8 ] ;
2007-03-20 20:35:03 -04:00
struct btrfs_fs_info * info = extent_root - > fs_info ;
2007-03-07 11:50:24 -05:00
2007-03-13 09:49:06 -04:00
btrfs_set_extent_refs ( & extent_item , 1 ) ;
2007-03-07 11:50:24 -05:00
ins . offset = 1 ;
ins . flags = 0 ;
2007-03-15 12:56:47 -04:00
btrfs_set_key_type ( & ins , BTRFS_EXTENT_ITEM_KEY ) ;
2007-04-23 17:01:05 -04:00
btrfs_set_extent_owner ( & extent_item , extent_root - > root_key . objectid ) ;
2007-03-07 11:50:24 -05:00
2007-08-08 20:17:12 -04:00
while ( 1 ) {
ret = find_first_radix_bit ( & info - > extent_ins_radix , gang , 0 ,
ARRAY_SIZE ( gang ) ) ;
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
ins . objectid = gang [ i ] ;
err = btrfs_insert_item ( trans , extent_root , & ins ,
& extent_item ,
sizeof ( extent_item ) ) ;
clear_radix_bit ( & info - > extent_ins_radix , gang [ i ] ) ;
WARN_ON ( err ) ;
}
2007-03-07 11:50:24 -05:00
}
return 0 ;
}
2007-03-26 10:15:30 -04:00
static int pin_down_block ( struct btrfs_root * root , u64 blocknr , int pending )
2007-03-22 12:13:20 -04:00
{
int err ;
2007-03-25 11:35:08 -04:00
struct btrfs_header * header ;
2007-03-26 10:15:30 -04:00
struct buffer_head * bh ;
2007-03-27 11:05:53 -04:00
if ( ! pending ) {
2007-03-28 13:57:48 -04:00
bh = btrfs_find_tree_block ( root , blocknr ) ;
2007-04-02 10:50:19 -04:00
if ( bh ) {
if ( buffer_uptodate ( bh ) ) {
u64 transid =
root - > fs_info - > running_transaction - > transid ;
header = btrfs_buffer_header ( bh ) ;
if ( btrfs_header_generation ( header ) = =
transid ) {
btrfs_block_release ( root , bh ) ;
return 0 ;
}
2007-03-27 11:05:53 -04:00
}
2007-03-30 14:27:56 -04:00
btrfs_block_release ( root , bh ) ;
2007-03-26 10:15:30 -04:00
}
err = set_radix_bit ( & root - > fs_info - > pinned_radix , blocknr ) ;
2007-05-06 10:15:01 -04:00
if ( ! err ) {
struct btrfs_block_group_cache * cache ;
2007-06-11 21:33:38 -04:00
cache = btrfs_lookup_block_group ( root - > fs_info ,
blocknr ) ;
2007-05-06 10:15:01 -04:00
if ( cache )
cache - > pinned + + ;
}
2007-03-27 11:05:53 -04:00
} else {
err = set_radix_bit ( & root - > fs_info - > pending_del_radix , blocknr ) ;
}
2007-05-06 10:15:01 -04:00
BUG_ON ( err < 0 ) ;
2007-03-22 12:13:20 -04:00
return 0 ;
}
2007-02-26 10:40:21 -05:00
/*
2007-03-06 20:08:01 -05:00
* remove an extent from the root , returns 0 on success
2007-02-26 10:40:21 -05:00
*/
2007-03-16 16:20:31 -04:00
static int __free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-05-09 20:13:14 -04:00
* root , u64 blocknr , u64 num_blocks , int pin ,
int mark_free )
2007-03-06 20:08:01 -05:00
{
2007-04-02 11:20:42 -04:00
struct btrfs_path * path ;
2007-03-12 16:22:34 -04:00
struct btrfs_key key ;
2007-03-20 20:35:03 -04:00
struct btrfs_fs_info * info = root - > fs_info ;
struct btrfs_root * extent_root = info - > extent_root ;
2007-03-06 20:08:01 -05:00
int ret ;
2007-03-13 10:46:10 -04:00
struct btrfs_extent_item * ei ;
2007-03-13 09:49:06 -04:00
u32 refs ;
2007-03-07 11:50:24 -05:00
2007-03-06 20:08:01 -05:00
key . objectid = blocknr ;
key . flags = 0 ;
2007-03-15 12:56:47 -04:00
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-06 20:08:01 -05:00
key . offset = num_blocks ;
2007-04-02 11:20:42 -04:00
path = btrfs_alloc_path ( ) ;
2007-06-22 14:16:25 -04:00
if ( ! path )
return - ENOMEM ;
2007-04-05 10:38:44 -04:00
2007-06-22 14:16:25 -04:00
ret = btrfs_search_slot ( trans , extent_root , & key , path , - 1 , 1 ) ;
if ( ret < 0 )
return ret ;
BUG_ON ( ret ) ;
2007-04-02 11:20:42 -04:00
ei = btrfs_item_ptr ( btrfs_buffer_leaf ( path - > nodes [ 0 ] ) , path - > slots [ 0 ] ,
2007-03-14 14:14:43 -04:00
struct btrfs_extent_item ) ;
2007-03-06 20:08:01 -05:00
BUG_ON ( ei - > refs = = 0 ) ;
2007-03-13 09:49:06 -04:00
refs = btrfs_extent_refs ( ei ) - 1 ;
btrfs_set_extent_refs ( ei , refs ) ;
2007-04-02 11:20:42 -04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
2007-03-13 09:49:06 -04:00
if ( refs = = 0 ) {
2007-08-29 15:47:34 -04:00
u64 super_blocks_used , root_blocks_used ;
2007-03-25 11:35:08 -04:00
if ( pin ) {
2007-03-26 10:15:30 -04:00
ret = pin_down_block ( root , blocknr , 0 ) ;
2007-03-25 11:35:08 -04:00
BUG_ON ( ret ) ;
}
2007-08-29 15:47:34 -04:00
/* block accounting for super block */
2007-06-26 10:06:50 -04:00
super_blocks_used = btrfs_super_blocks_used ( & info - > super_copy ) ;
btrfs_set_super_blocks_used ( & info - > super_copy ,
2007-03-20 20:35:03 -04:00
super_blocks_used - num_blocks ) ;
2007-08-29 15:47:34 -04:00
/* block accounting for root item */
root_blocks_used = btrfs_root_blocks_used ( & root - > root_item ) ;
btrfs_set_root_blocks_used ( & root - > root_item ,
root_blocks_used - num_blocks ) ;
2007-04-02 11:20:42 -04:00
ret = btrfs_del_item ( trans , extent_root , path ) ;
2007-06-22 14:16:25 -04:00
if ( ret ) {
return ret ;
}
2007-05-09 20:13:14 -04:00
ret = update_block_group ( trans , root , blocknr , num_blocks , 0 ,
2007-05-29 16:52:18 -04:00
mark_free , 0 ) ;
2007-04-26 16:46:15 -04:00
BUG_ON ( ret ) ;
2007-03-06 20:08:01 -05:00
}
2007-04-02 11:20:42 -04:00
btrfs_free_path ( path ) ;
2007-03-16 16:20:31 -04:00
finish_current_insert ( trans , extent_root ) ;
2007-03-06 20:08:01 -05:00
return ret ;
}
/*
* find all the blocks marked as pending in the radix tree and remove
* them from the extent map
*/
2007-03-16 16:20:31 -04:00
static int del_pending_extents ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root )
2007-03-06 20:08:01 -05:00
{
int ret ;
2007-03-22 12:13:20 -04:00
int wret ;
int err = 0 ;
2007-03-26 10:15:30 -04:00
unsigned long gang [ 4 ] ;
2007-03-06 20:08:01 -05:00
int i ;
2007-03-26 10:15:30 -04:00
struct radix_tree_root * pending_radix ;
struct radix_tree_root * pinned_radix ;
2007-05-06 10:15:01 -04:00
struct btrfs_block_group_cache * cache ;
2007-03-26 10:15:30 -04:00
pending_radix = & extent_root - > fs_info - > pending_del_radix ;
pinned_radix = & extent_root - > fs_info - > pinned_radix ;
2007-03-06 20:08:01 -05:00
while ( 1 ) {
2007-05-09 20:13:14 -04:00
ret = find_first_radix_bit ( pending_radix , gang , 0 ,
2007-03-26 10:15:30 -04:00
ARRAY_SIZE ( gang ) ) ;
2007-03-06 20:08:01 -05:00
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
2007-03-26 10:15:30 -04:00
wret = set_radix_bit ( pinned_radix , gang [ i ] ) ;
2007-05-06 10:15:01 -04:00
if ( wret = = 0 ) {
2007-06-11 21:33:38 -04:00
cache =
btrfs_lookup_block_group ( extent_root - > fs_info ,
2007-05-06 10:15:01 -04:00
gang [ i ] ) ;
if ( cache )
cache - > pinned + + ;
}
if ( wret < 0 ) {
printk ( KERN_CRIT " set_radix_bit, err %d \n " ,
wret ) ;
BUG_ON ( wret < 0 ) ;
}
2007-03-26 10:15:30 -04:00
wret = clear_radix_bit ( pending_radix , gang [ i ] ) ;
BUG_ON ( wret ) ;
2007-03-23 10:01:08 -04:00
wret = __free_extent ( trans , extent_root ,
2007-05-09 20:13:14 -04:00
gang [ i ] , 1 , 0 , 0 ) ;
2007-03-22 12:13:20 -04:00
if ( wret )
err = wret ;
2007-02-26 10:40:21 -05:00
}
}
2007-03-22 12:13:20 -04:00
return err ;
2007-02-26 10:40:21 -05:00
}
/*
* remove an extent from the root , returns 0 on success
*/
2007-03-16 16:20:31 -04:00
int btrfs_free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , u64 blocknr , u64 num_blocks , int pin )
2007-02-26 10:40:21 -05:00
{
2007-03-20 14:38:32 -04:00
struct btrfs_root * extent_root = root - > fs_info - > extent_root ;
2007-02-26 10:40:21 -05:00
int pending_ret ;
int ret ;
2007-03-06 20:08:01 -05:00
2007-02-26 10:40:21 -05:00
if ( root = = extent_root ) {
2007-03-26 10:15:30 -04:00
pin_down_block ( root , blocknr , 1 ) ;
2007-02-26 10:40:21 -05:00
return 0 ;
}
2007-05-09 20:13:14 -04:00
ret = __free_extent ( trans , root , blocknr , num_blocks , pin , pin = = 0 ) ;
2007-03-22 12:13:20 -04:00
pending_ret = del_pending_extents ( trans , root - > fs_info - > extent_root ) ;
2007-02-26 10:40:21 -05:00
return ret ? ret : pending_ret ;
}
/*
* walks the btree of allocated extents and find a hole of a given size .
* The key ins is changed to record the hole :
* ins - > objectid = = block start
2007-03-15 12:56:47 -04:00
* ins - > flags = BTRFS_EXTENT_ITEM_KEY
2007-02-26 10:40:21 -05:00
* ins - > offset = = number of blocks
* Any available blocks before search_start are skipped .
*/
2007-03-16 16:20:31 -04:00
static int find_free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-08-07 16:15:09 -04:00
* orig_root , u64 num_blocks , u64 empty_size ,
u64 search_start , u64 search_end , u64 hint_block ,
2007-06-26 12:20:46 -04:00
struct btrfs_key * ins , u64 exclude_start ,
u64 exclude_nr , int data )
2007-02-26 10:40:21 -05:00
{
2007-04-02 11:20:42 -04:00
struct btrfs_path * path ;
2007-03-12 16:22:34 -04:00
struct btrfs_key key ;
2007-02-26 10:40:21 -05:00
int ret ;
u64 hole_size = 0 ;
int slot = 0 ;
2007-03-22 12:13:20 -04:00
u64 last_block = 0 ;
2007-03-07 11:50:24 -05:00
u64 test_block ;
2007-05-06 10:15:01 -04:00
u64 orig_search_start = search_start ;
2007-02-26 10:40:21 -05:00
int start_found ;
2007-03-13 10:46:10 -04:00
struct btrfs_leaf * l ;
2007-03-20 14:38:32 -04:00
struct btrfs_root * root = orig_root - > fs_info - > extent_root ;
2007-04-25 15:52:25 -04:00
struct btrfs_fs_info * info = root - > fs_info ;
2007-03-07 16:15:30 -05:00
int total_needed = num_blocks ;
2007-03-22 12:13:20 -04:00
int level ;
2007-05-03 09:06:49 -04:00
struct btrfs_block_group_cache * block_group ;
2007-05-06 10:15:01 -04:00
int full_scan = 0 ;
2007-05-30 10:22:12 -04:00
int wrapped = 0 ;
2007-02-26 10:40:21 -05:00
2007-08-08 20:17:12 -04:00
WARN_ON ( num_blocks < 1 ) ;
2007-04-04 15:27:52 -04:00
ins - > flags = 0 ;
btrfs_set_key_type ( ins , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-22 12:13:20 -04:00
level = btrfs_header_level ( btrfs_buffer_header ( root - > node ) ) ;
2007-05-07 20:03:49 -04:00
if ( search_end = = ( u64 ) - 1 )
2007-06-26 10:06:50 -04:00
search_end = btrfs_super_total_blocks ( & info - > super_copy ) ;
2007-05-30 10:22:12 -04:00
if ( hint_block ) {
2007-06-11 21:33:38 -04:00
block_group = btrfs_lookup_block_group ( info , hint_block ) ;
2007-05-06 10:15:01 -04:00
block_group = btrfs_find_block_group ( root , block_group ,
2007-05-30 10:22:12 -04:00
hint_block , data , 1 ) ;
2007-05-06 10:15:01 -04:00
} else {
block_group = btrfs_find_block_group ( root ,
trans - > block_group , 0 ,
2007-05-18 13:28:27 -04:00
data , 1 ) ;
2007-05-06 10:15:01 -04:00
}
2007-08-07 16:15:09 -04:00
total_needed + = empty_size ;
2007-06-19 16:23:05 -04:00
path = btrfs_alloc_path ( ) ;
2007-05-06 10:15:01 -04:00
check_failed :
2007-05-29 16:52:18 -04:00
if ( ! block_group - > data )
2007-05-09 20:13:14 -04:00
search_start = find_search_start ( root , & block_group ,
search_start , total_needed ) ;
2007-05-30 10:22:12 -04:00
else if ( ! full_scan )
2007-05-09 20:13:14 -04:00
search_start = max ( block_group - > last_alloc , search_start ) ;
2007-04-02 11:20:42 -04:00
btrfs_init_path ( path ) ;
2007-02-26 10:40:21 -05:00
ins - > objectid = search_start ;
ins - > offset = 0 ;
start_found = 0 ;
2007-08-27 16:49:44 -04:00
path - > reada = 2 ;
2007-05-09 20:13:14 -04:00
2007-04-02 11:20:42 -04:00
ret = btrfs_search_slot ( trans , root , ins , path , 0 , 0 ) ;
2007-02-28 16:46:22 -05:00
if ( ret < 0 )
goto error ;
2007-02-28 16:35:06 -05:00
2007-05-09 20:13:14 -04:00
if ( path - > slots [ 0 ] > 0 ) {
2007-04-02 11:20:42 -04:00
path - > slots [ 0 ] - - ;
2007-05-09 20:13:14 -04:00
}
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
btrfs_disk_key_to_cpu ( & key , & l - > items [ path - > slots [ 0 ] ] . key ) ;
/*
* a rare case , go back one key if we hit a block group item
* instead of an extent item
*/
if ( btrfs_key_type ( & key ) ! = BTRFS_EXTENT_ITEM_KEY & &
key . objectid + key . offset > = search_start ) {
ins - > objectid = key . objectid ;
ins - > offset = key . offset - 1 ;
btrfs_release_path ( root , path ) ;
ret = btrfs_search_slot ( trans , root , ins , path , 0 , 0 ) ;
if ( ret < 0 )
goto error ;
if ( path - > slots [ 0 ] > 0 ) {
path - > slots [ 0 ] - - ;
}
}
2007-03-07 16:15:30 -05:00
2007-02-26 10:40:21 -05:00
while ( 1 ) {
2007-04-02 11:20:42 -04:00
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
slot = path - > slots [ 0 ] ;
2007-03-12 12:01:18 -04:00
if ( slot > = btrfs_header_nritems ( & l - > header ) ) {
2007-04-02 11:20:42 -04:00
ret = btrfs_next_leaf ( root , path ) ;
2007-02-26 10:40:21 -05:00
if ( ret = = 0 )
continue ;
2007-02-28 16:46:22 -05:00
if ( ret < 0 )
goto error ;
2007-02-26 10:40:21 -05:00
if ( ! start_found ) {
ins - > objectid = search_start ;
2007-05-07 20:03:49 -04:00
ins - > offset = search_end - search_start ;
2007-02-26 10:40:21 -05:00
start_found = 1 ;
goto check_pending ;
}
ins - > objectid = last_block > search_start ?
last_block : search_start ;
2007-05-07 20:03:49 -04:00
ins - > offset = search_end - ins - > objectid ;
2007-02-26 10:40:21 -05:00
goto check_pending ;
}
2007-05-09 20:13:14 -04:00
2007-03-12 16:22:34 -04:00
btrfs_disk_key_to_cpu ( & key , & l - > items [ slot ] . key ) ;
2007-05-09 20:13:14 -04:00
if ( key . objectid > = search_start & & key . objectid > last_block & &
start_found ) {
if ( last_block < search_start )
last_block = search_start ;
hole_size = key . objectid - last_block ;
if ( hole_size > = num_blocks ) {
ins - > objectid = last_block ;
ins - > offset = hole_size ;
goto check_pending ;
2007-03-07 16:15:30 -05:00
}
2007-02-26 10:40:21 -05:00
}
2007-05-09 20:13:14 -04:00
if ( btrfs_key_type ( & key ) ! = BTRFS_EXTENT_ITEM_KEY )
goto next ;
2007-03-07 16:15:30 -05:00
start_found = 1 ;
2007-03-12 16:22:34 -04:00
last_block = key . objectid + key . offset ;
2007-05-30 10:22:12 -04:00
if ( ! full_scan & & last_block > = block_group - > key . objectid +
2007-05-06 10:15:01 -04:00
block_group - > key . offset ) {
btrfs_release_path ( root , path ) ;
search_start = block_group - > key . objectid +
block_group - > key . offset * 2 ;
goto new_group ;
}
2007-04-26 16:46:15 -04:00
next :
2007-04-02 11:20:42 -04:00
path - > slots [ 0 ] + + ;
2007-05-18 13:28:27 -04:00
cond_resched ( ) ;
2007-02-26 10:40:21 -05:00
}
check_pending :
/* we have to make sure we didn't find an extent that has already
* been allocated by the map tree or the original allocation
*/
2007-04-02 11:20:42 -04:00
btrfs_release_path ( root , path ) ;
2007-02-26 10:40:21 -05:00
BUG_ON ( ins - > objectid < search_start ) ;
2007-05-09 20:13:14 -04:00
2007-05-07 20:03:49 -04:00
if ( ins - > objectid + num_blocks > = search_end ) {
2007-05-30 10:22:12 -04:00
if ( full_scan ) {
ret = - ENOSPC ;
goto error ;
}
2007-05-06 10:15:01 -04:00
search_start = orig_search_start ;
2007-08-07 16:15:09 -04:00
if ( wrapped ) {
if ( ! full_scan )
total_needed - = empty_size ;
2007-05-30 10:22:12 -04:00
full_scan = 1 ;
2007-08-07 16:15:09 -04:00
} else
2007-05-30 10:22:12 -04:00
wrapped = 1 ;
2007-05-06 10:15:01 -04:00
goto new_group ;
2007-04-28 08:48:10 -04:00
}
2007-03-07 11:50:24 -05:00
for ( test_block = ins - > objectid ;
2007-04-25 15:52:25 -04:00
test_block < ins - > objectid + num_blocks ; test_block + + ) {
2007-08-08 20:17:12 -04:00
if ( test_radix_bit ( & info - > pinned_radix , test_block ) | |
test_radix_bit ( & info - > extent_ins_radix , test_block ) ) {
2007-03-07 11:50:24 -05:00
search_start = test_block + 1 ;
2007-05-06 10:15:01 -04:00
goto new_group ;
2007-02-26 10:40:21 -05:00
}
}
2007-06-26 12:20:46 -04:00
if ( exclude_nr > 0 & & ( ins - > objectid + num_blocks > exclude_start & &
ins - > objectid < exclude_start + exclude_nr ) ) {
search_start = exclude_start + exclude_nr ;
goto new_group ;
}
2007-05-09 20:13:14 -04:00
if ( ! data ) {
2007-06-11 21:33:38 -04:00
block_group = btrfs_lookup_block_group ( info , ins - > objectid ) ;
2007-08-08 20:17:12 -04:00
if ( block_group )
trans - > block_group = block_group ;
2007-04-25 15:52:25 -04:00
}
2007-03-07 11:50:24 -05:00
ins - > offset = num_blocks ;
2007-04-02 11:20:42 -04:00
btrfs_free_path ( path ) ;
2007-02-26 10:40:21 -05:00
return 0 ;
2007-05-06 10:15:01 -04:00
new_group :
2007-05-07 20:03:49 -04:00
if ( search_start + num_blocks > = search_end ) {
2007-05-06 10:15:01 -04:00
search_start = orig_search_start ;
2007-05-30 10:22:12 -04:00
if ( full_scan ) {
ret = - ENOSPC ;
goto error ;
}
2007-08-07 16:15:09 -04:00
if ( wrapped ) {
if ( ! full_scan )
total_needed - = empty_size ;
2007-05-30 10:22:12 -04:00
full_scan = 1 ;
2007-08-07 16:15:09 -04:00
} else
2007-05-30 10:22:12 -04:00
wrapped = 1 ;
2007-05-06 10:15:01 -04:00
}
2007-06-11 21:33:38 -04:00
block_group = btrfs_lookup_block_group ( info , search_start ) ;
2007-05-30 10:22:12 -04:00
cond_resched ( ) ;
2007-05-06 10:15:01 -04:00
if ( ! full_scan )
block_group = btrfs_find_block_group ( root , block_group ,
2007-05-18 13:28:27 -04:00
search_start , data , 0 ) ;
2007-05-06 10:15:01 -04:00
goto check_failed ;
2007-02-28 16:46:22 -05:00
error :
2007-04-02 11:20:42 -04:00
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
2007-02-28 16:46:22 -05:00
return ret ;
2007-02-26 10:40:21 -05:00
}
/*
* finds a free extent and does all the dirty work required for allocation
* returns the key for the extent through ins , and a tree buffer for
* the first block of the extent through buf .
*
* returns 0 if everything worked , non - zero otherwise .
*/
2007-04-20 20:23:12 -04:00
int btrfs_alloc_extent ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root , u64 owner ,
2007-08-07 16:15:09 -04:00
u64 num_blocks , u64 empty_size , u64 hint_block ,
2007-05-03 09:06:49 -04:00
u64 search_end , struct btrfs_key * ins , int data )
2007-02-26 10:40:21 -05:00
{
int ret ;
int pending_ret ;
2007-08-29 15:47:34 -04:00
u64 super_blocks_used , root_blocks_used ;
2007-05-30 10:22:12 -04:00
u64 search_start = 0 ;
2007-03-20 20:35:03 -04:00
struct btrfs_fs_info * info = root - > fs_info ;
struct btrfs_root * extent_root = info - > extent_root ;
2007-03-13 10:46:10 -04:00
struct btrfs_extent_item extent_item ;
2007-03-07 11:50:24 -05:00
2007-03-13 09:49:06 -04:00
btrfs_set_extent_refs ( & extent_item , 1 ) ;
2007-04-20 20:23:12 -04:00
btrfs_set_extent_owner ( & extent_item , owner ) ;
2007-02-26 10:40:21 -05:00
2007-08-08 20:17:12 -04:00
WARN_ON ( num_blocks < 1 ) ;
2007-08-07 16:15:09 -04:00
ret = find_free_extent ( trans , root , num_blocks , empty_size ,
search_start , search_end , hint_block , ins ,
2007-08-08 20:17:12 -04:00
trans - > alloc_exclude_start ,
trans - > alloc_exclude_nr , data ) ;
2007-06-28 15:57:36 -04:00
BUG_ON ( ret ) ;
2007-06-26 12:20:46 -04:00
if ( ret )
return ret ;
2007-02-26 10:40:21 -05:00
2007-08-29 15:47:34 -04:00
/* block accounting for super block */
2007-06-26 10:06:50 -04:00
super_blocks_used = btrfs_super_blocks_used ( & info - > super_copy ) ;
btrfs_set_super_blocks_used ( & info - > super_copy , super_blocks_used +
2007-03-20 20:35:03 -04:00
num_blocks ) ;
2007-08-08 20:17:12 -04:00
2007-08-29 15:47:34 -04:00
/* block accounting for root item */
root_blocks_used = btrfs_root_blocks_used ( & root - > root_item ) ;
btrfs_set_root_blocks_used ( & root - > root_item , root_blocks_used +
num_blocks ) ;
2007-08-08 20:17:12 -04:00
if ( root = = extent_root ) {
BUG_ON ( num_blocks ! = 1 ) ;
set_radix_bit ( & root - > fs_info - > extent_ins_radix , ins - > objectid ) ;
goto update_block ;
}
WARN_ON ( trans - > alloc_exclude_nr ) ;
trans - > alloc_exclude_start = ins - > objectid ;
trans - > alloc_exclude_nr = ins - > offset ;
2007-03-16 16:20:31 -04:00
ret = btrfs_insert_item ( trans , extent_root , ins , & extent_item ,
sizeof ( extent_item ) ) ;
2007-03-07 11:50:24 -05:00
2007-08-08 20:17:12 -04:00
trans - > alloc_exclude_start = 0 ;
trans - > alloc_exclude_nr = 0 ;
2007-06-28 15:57:36 -04:00
BUG_ON ( ret ) ;
2007-03-16 16:20:31 -04:00
finish_current_insert ( trans , extent_root ) ;
2007-03-22 12:13:20 -04:00
pending_ret = del_pending_extents ( trans , extent_root ) ;
2007-05-09 20:13:14 -04:00
if ( ret ) {
2007-03-07 11:50:24 -05:00
return ret ;
2007-05-09 20:13:14 -04:00
}
if ( pending_ret ) {
2007-03-07 11:50:24 -05:00
return pending_ret ;
2007-05-09 20:13:14 -04:00
}
2007-08-08 20:17:12 -04:00
update_block :
2007-05-29 16:52:18 -04:00
ret = update_block_group ( trans , root , ins - > objectid , ins - > offset , 1 , 0 ,
data ) ;
2007-06-07 22:13:21 -04:00
BUG_ON ( ret ) ;
2007-03-07 11:50:24 -05:00
return 0 ;
2007-02-26 10:40:21 -05:00
}
/*
* helper function to allocate a block for a given tree
* returns the tree buffer or NULL .
*/
2007-03-22 12:13:20 -04:00
struct buffer_head * btrfs_alloc_free_block ( struct btrfs_trans_handle * trans ,
2007-08-07 16:15:09 -04:00
struct btrfs_root * root , u64 hint ,
u64 empty_size )
2007-02-26 10:40:21 -05:00
{
2007-03-12 16:22:34 -04:00
struct btrfs_key ins ;
2007-02-26 10:40:21 -05:00
int ret ;
2007-03-22 12:13:20 -04:00
struct buffer_head * buf ;
2007-02-26 10:40:21 -05:00
2007-04-20 20:23:12 -04:00
ret = btrfs_alloc_extent ( trans , root , root - > root_key . objectid ,
2007-08-29 09:11:44 -04:00
1 , empty_size , hint , ( u64 ) - 1 , & ins , 0 ) ;
2007-02-26 10:40:21 -05:00
if ( ret ) {
2007-06-22 14:16:25 -04:00
BUG_ON ( ret > 0 ) ;
return ERR_PTR ( ret ) ;
2007-02-26 10:40:21 -05:00
}
2007-03-28 13:57:48 -04:00
buf = btrfs_find_create_tree_block ( root , ins . objectid ) ;
2007-06-22 14:16:25 -04:00
if ( ! buf ) {
btrfs_free_extent ( trans , root , ins . objectid , 1 , 0 ) ;
return ERR_PTR ( - ENOMEM ) ;
}
2007-08-07 16:15:09 -04:00
WARN_ON ( buffer_dirty ( buf ) ) ;
2007-03-23 11:00:45 -04:00
set_buffer_uptodate ( buf ) ;
2007-05-01 08:53:32 -04:00
set_buffer_checked ( buf ) ;
2007-08-10 14:42:37 -04:00
set_buffer_defrag ( buf ) ;
2007-04-28 09:29:35 -04:00
set_radix_bit ( & trans - > transaction - > dirty_pages , buf - > b_page - > index ) ;
2007-09-17 10:58:06 -04:00
trans - > blocks_used + + ;
2007-02-26 10:40:21 -05:00
return buf ;
}
2007-03-06 20:08:01 -05:00
2007-03-27 06:33:00 -04:00
static int drop_leaf_ref ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root , struct buffer_head * cur )
{
struct btrfs_disk_key * key ;
struct btrfs_leaf * leaf ;
struct btrfs_file_extent_item * fi ;
int i ;
int nritems ;
int ret ;
BUG_ON ( ! btrfs_is_leaf ( btrfs_buffer_node ( cur ) ) ) ;
leaf = btrfs_buffer_leaf ( cur ) ;
nritems = btrfs_header_nritems ( & leaf - > header ) ;
for ( i = 0 ; i < nritems ; i + + ) {
2007-05-24 13:35:57 -04:00
u64 disk_blocknr ;
2007-03-27 06:33:00 -04:00
key = & leaf - > items [ i ] . key ;
if ( btrfs_disk_key_type ( key ) ! = BTRFS_EXTENT_DATA_KEY )
continue ;
fi = btrfs_item_ptr ( leaf , i , struct btrfs_file_extent_item ) ;
2007-04-19 13:37:44 -04:00
if ( btrfs_file_extent_type ( fi ) = = BTRFS_FILE_EXTENT_INLINE )
continue ;
2007-03-27 06:33:00 -04:00
/*
* FIXME make sure to insert a trans record that
* repeats the snapshot del on crash
*/
2007-05-24 13:35:57 -04:00
disk_blocknr = btrfs_file_extent_disk_blocknr ( fi ) ;
if ( disk_blocknr = = 0 )
continue ;
ret = btrfs_free_extent ( trans , root , disk_blocknr ,
2007-03-27 06:33:00 -04:00
btrfs_file_extent_disk_num_blocks ( fi ) ,
0 ) ;
BUG_ON ( ret ) ;
}
return 0 ;
}
2007-06-19 16:23:05 -04:00
static void reada_walk_down ( struct btrfs_root * root ,
struct btrfs_node * node )
{
int i ;
u32 nritems ;
u64 blocknr ;
int ret ;
u32 refs ;
nritems = btrfs_header_nritems ( & node - > header ) ;
for ( i = 0 ; i < nritems ; i + + ) {
blocknr = btrfs_node_blockptr ( node , i ) ;
ret = lookup_extent_ref ( NULL , root , blocknr , 1 , & refs ) ;
BUG_ON ( ret ) ;
if ( refs ! = 1 )
continue ;
2007-08-08 20:17:12 -04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-06-19 16:23:05 -04:00
ret = readahead_tree_block ( root , blocknr ) ;
2007-08-08 20:17:12 -04:00
cond_resched ( ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-06-19 16:23:05 -04:00
if ( ret )
break ;
}
}
2007-03-13 11:09:37 -04:00
/*
* helper function for drop_snapshot , this walks down the tree dropping ref
* counts as it goes .
*/
2007-03-16 16:20:31 -04:00
static int walk_down_tree ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , struct btrfs_path * path , int * level )
2007-03-10 06:35:47 -05:00
{
2007-03-22 12:13:20 -04:00
struct buffer_head * next ;
struct buffer_head * cur ;
2007-03-10 06:35:47 -05:00
u64 blocknr ;
int ret ;
u32 refs ;
2007-04-02 11:20:42 -04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-04-17 13:26:50 -04:00
ret = lookup_extent_ref ( trans , root , bh_blocknr ( path - > nodes [ * level ] ) ,
2007-03-27 06:33:00 -04:00
1 , & refs ) ;
2007-03-10 06:35:47 -05:00
BUG_ON ( ret ) ;
if ( refs > 1 )
goto out ;
2007-06-19 16:23:05 -04:00
2007-03-13 11:09:37 -04:00
/*
* walk down to the last node level and free all the leaves
*/
2007-03-27 06:33:00 -04:00
while ( * level > = 0 ) {
2007-04-02 11:20:42 -04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-03-10 06:35:47 -05:00
cur = path - > nodes [ * level ] ;
2007-06-19 16:23:05 -04:00
if ( * level > 0 & & path - > slots [ * level ] = = 0 )
reada_walk_down ( root , btrfs_buffer_node ( cur ) ) ;
2007-04-02 10:50:19 -04:00
if ( btrfs_header_level ( btrfs_buffer_header ( cur ) ) ! = * level )
WARN_ON ( 1 ) ;
2007-06-19 16:23:05 -04:00
2007-03-12 12:01:18 -04:00
if ( path - > slots [ * level ] > =
2007-03-22 12:13:20 -04:00
btrfs_header_nritems ( btrfs_buffer_header ( cur ) ) )
2007-03-10 06:35:47 -05:00
break ;
2007-03-27 06:33:00 -04:00
if ( * level = = 0 ) {
ret = drop_leaf_ref ( trans , root , cur ) ;
BUG_ON ( ret ) ;
break ;
}
2007-03-22 12:13:20 -04:00
blocknr = btrfs_node_blockptr ( btrfs_buffer_node ( cur ) ,
path - > slots [ * level ] ) ;
2007-04-17 13:26:50 -04:00
ret = lookup_extent_ref ( trans , root , blocknr , 1 , & refs ) ;
2007-03-27 06:33:00 -04:00
BUG_ON ( ret ) ;
if ( refs ! = 1 ) {
2007-03-10 06:35:47 -05:00
path - > slots [ * level ] + + ;
2007-03-16 16:20:31 -04:00
ret = btrfs_free_extent ( trans , root , blocknr , 1 , 1 ) ;
2007-03-10 06:35:47 -05:00
BUG_ON ( ret ) ;
continue ;
}
2007-08-10 14:06:19 -04:00
next = btrfs_find_tree_block ( root , blocknr ) ;
if ( ! next | | ! buffer_uptodate ( next ) ) {
brelse ( next ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
next = read_tree_block ( root , blocknr ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
/* we dropped the lock, check one more time */
ret = lookup_extent_ref ( trans , root , blocknr , 1 , & refs ) ;
BUG_ON ( ret ) ;
if ( refs ! = 1 ) {
path - > slots [ * level ] + + ;
brelse ( next ) ;
ret = btrfs_free_extent ( trans , root ,
blocknr , 1 , 1 ) ;
BUG_ON ( ret ) ;
continue ;
}
}
2007-04-02 11:20:42 -04:00
WARN_ON ( * level < = 0 ) ;
2007-03-12 09:03:27 -04:00
if ( path - > nodes [ * level - 1 ] )
2007-03-13 10:46:10 -04:00
btrfs_block_release ( root , path - > nodes [ * level - 1 ] ) ;
2007-03-10 06:35:47 -05:00
path - > nodes [ * level - 1 ] = next ;
2007-03-22 12:13:20 -04:00
* level = btrfs_header_level ( btrfs_buffer_header ( next ) ) ;
2007-03-10 06:35:47 -05:00
path - > slots [ * level ] = 0 ;
}
out :
2007-04-02 11:20:42 -04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-03-27 06:33:00 -04:00
ret = btrfs_free_extent ( trans , root ,
2007-04-11 15:53:25 -04:00
bh_blocknr ( path - > nodes [ * level ] ) , 1 , 1 ) ;
2007-03-13 10:46:10 -04:00
btrfs_block_release ( root , path - > nodes [ * level ] ) ;
2007-03-10 06:35:47 -05:00
path - > nodes [ * level ] = NULL ;
* level + = 1 ;
BUG_ON ( ret ) ;
return 0 ;
}
2007-03-13 11:09:37 -04:00
/*
* helper for dropping snapshots . This walks back up the tree in the path
* to find the first node higher up where we haven ' t yet gone through
* all the slots
*/
2007-03-16 16:20:31 -04:00
static int walk_up_tree ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , struct btrfs_path * path , int * level )
2007-03-10 06:35:47 -05:00
{
int i ;
int slot ;
int ret ;
2007-08-07 15:52:19 -04:00
struct btrfs_root_item * root_item = & root - > root_item ;
2007-03-13 10:46:10 -04:00
for ( i = * level ; i < BTRFS_MAX_LEVEL - 1 & & path - > nodes [ i ] ; i + + ) {
2007-03-10 06:35:47 -05:00
slot = path - > slots [ i ] ;
2007-03-22 12:13:20 -04:00
if ( slot < btrfs_header_nritems (
btrfs_buffer_header ( path - > nodes [ i ] ) ) - 1 ) {
2007-08-07 15:52:19 -04:00
struct btrfs_node * node ;
node = btrfs_buffer_node ( path - > nodes [ i ] ) ;
2007-03-10 06:35:47 -05:00
path - > slots [ i ] + + ;
* level = i ;
2007-08-07 15:52:19 -04:00
WARN_ON ( * level = = 0 ) ;
memcpy ( & root_item - > drop_progress ,
& node - > ptrs [ path - > slots [ i ] ] . key ,
sizeof ( root_item - > drop_progress ) ) ;
root_item - > drop_level = i ;
2007-03-10 06:35:47 -05:00
return 0 ;
} else {
2007-03-16 16:20:31 -04:00
ret = btrfs_free_extent ( trans , root ,
2007-04-11 15:53:25 -04:00
bh_blocknr ( path - > nodes [ * level ] ) ,
2007-03-16 16:20:31 -04:00
1 , 1 ) ;
2007-03-27 06:33:00 -04:00
BUG_ON ( ret ) ;
2007-03-13 10:46:10 -04:00
btrfs_block_release ( root , path - > nodes [ * level ] ) ;
2007-03-12 09:03:27 -04:00
path - > nodes [ * level ] = NULL ;
2007-03-10 06:35:47 -05:00
* level = i + 1 ;
}
}
return 1 ;
}
2007-03-13 11:09:37 -04:00
/*
* drop the reference count on the tree rooted at ' snap ' . This traverses
* the tree freeing any blocks that have a ref count of zero after being
* decremented .
*/
2007-03-16 16:20:31 -04:00
int btrfs_drop_snapshot ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-08-07 15:52:19 -04:00
* root )
2007-03-10 06:35:47 -05:00
{
2007-03-13 16:47:54 -04:00
int ret = 0 ;
2007-03-13 11:09:37 -04:00
int wret ;
2007-03-10 06:35:47 -05:00
int level ;
2007-04-02 11:20:42 -04:00
struct btrfs_path * path ;
2007-03-10 06:35:47 -05:00
int i ;
int orig_level ;
2007-08-07 15:52:19 -04:00
struct btrfs_root_item * root_item = & root - > root_item ;
2007-03-10 06:35:47 -05:00
2007-04-02 11:20:42 -04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
2007-03-10 06:35:47 -05:00
2007-08-07 15:52:19 -04:00
level = btrfs_header_level ( btrfs_buffer_header ( root - > node ) ) ;
2007-03-10 06:35:47 -05:00
orig_level = level ;
2007-08-07 15:52:19 -04:00
if ( btrfs_disk_key_objectid ( & root_item - > drop_progress ) = = 0 ) {
path - > nodes [ level ] = root - > node ;
path - > slots [ level ] = 0 ;
} else {
struct btrfs_key key ;
struct btrfs_disk_key * found_key ;
struct btrfs_node * node ;
2007-08-07 16:15:09 -04:00
2007-08-07 15:52:19 -04:00
btrfs_disk_key_to_cpu ( & key , & root_item - > drop_progress ) ;
2007-08-07 16:15:09 -04:00
level = root_item - > drop_level ;
path - > lowest_level = level ;
2007-08-07 15:52:19 -04:00
wret = btrfs_search_slot ( NULL , root , & key , path , 0 , 0 ) ;
2007-08-07 16:15:09 -04:00
if ( wret < 0 ) {
2007-08-07 15:52:19 -04:00
ret = wret ;
goto out ;
}
node = btrfs_buffer_node ( path - > nodes [ level ] ) ;
found_key = & node - > ptrs [ path - > slots [ level ] ] . key ;
WARN_ON ( memcmp ( found_key , & root_item - > drop_progress ,
sizeof ( * found_key ) ) ) ;
}
2007-03-10 06:35:47 -05:00
while ( 1 ) {
2007-04-02 11:20:42 -04:00
wret = walk_down_tree ( trans , root , path , & level ) ;
2007-03-13 11:09:37 -04:00
if ( wret > 0 )
2007-03-10 06:35:47 -05:00
break ;
2007-03-13 11:09:37 -04:00
if ( wret < 0 )
ret = wret ;
2007-04-02 11:20:42 -04:00
wret = walk_up_tree ( trans , root , path , & level ) ;
2007-03-13 11:09:37 -04:00
if ( wret > 0 )
2007-03-10 06:35:47 -05:00
break ;
2007-03-13 11:09:37 -04:00
if ( wret < 0 )
ret = wret ;
2007-08-08 20:17:12 -04:00
ret = - EAGAIN ;
get_bh ( root - > node ) ;
break ;
2007-03-10 06:35:47 -05:00
}
2007-03-12 09:03:27 -04:00
for ( i = 0 ; i < = orig_level ; i + + ) {
2007-04-02 11:20:42 -04:00
if ( path - > nodes [ i ] ) {
btrfs_block_release ( root , path - > nodes [ i ] ) ;
2007-08-07 16:15:09 -04:00
path - > nodes [ i ] = 0 ;
2007-03-12 09:03:27 -04:00
}
2007-03-10 06:35:47 -05:00
}
2007-08-07 15:52:19 -04:00
out :
2007-04-02 11:20:42 -04:00
btrfs_free_path ( path ) ;
2007-03-13 11:09:37 -04:00
return ret ;
2007-03-10 06:35:47 -05:00
}
2007-04-26 16:46:15 -04:00
2007-05-06 10:15:01 -04:00
static int free_block_group_radix ( struct radix_tree_root * radix )
2007-04-26 16:46:15 -04:00
{
int ret ;
struct btrfs_block_group_cache * cache [ 8 ] ;
int i ;
while ( 1 ) {
2007-05-06 10:15:01 -04:00
ret = radix_tree_gang_lookup ( radix , ( void * * ) cache , 0 ,
2007-04-26 16:46:15 -04:00
ARRAY_SIZE ( cache ) ) ;
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
2007-05-06 10:15:01 -04:00
radix_tree_delete ( radix , cache [ i ] - > key . objectid +
2007-04-26 16:46:15 -04:00
cache [ i ] - > key . offset - 1 ) ;
kfree ( cache [ i ] ) ;
}
}
return 0 ;
}
2007-05-06 10:15:01 -04:00
int btrfs_free_block_groups ( struct btrfs_fs_info * info )
{
int ret ;
int ret2 ;
2007-05-09 20:13:14 -04:00
unsigned long gang [ 16 ] ;
int i ;
2007-05-06 10:15:01 -04:00
ret = free_block_group_radix ( & info - > block_group_radix ) ;
ret2 = free_block_group_radix ( & info - > block_group_data_radix ) ;
if ( ret )
return ret ;
if ( ret2 )
return ret2 ;
2007-05-09 20:13:14 -04:00
while ( 1 ) {
ret = find_first_radix_bit ( & info - > extent_map_radix ,
gang , 0 , ARRAY_SIZE ( gang ) ) ;
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
clear_radix_bit ( & info - > extent_map_radix , gang [ i ] ) ;
}
}
2007-05-06 10:15:01 -04:00
return 0 ;
}
2007-04-26 16:46:15 -04:00
int btrfs_read_block_groups ( struct btrfs_root * root )
{
struct btrfs_path * path ;
int ret ;
int err = 0 ;
struct btrfs_block_group_item * bi ;
struct btrfs_block_group_cache * cache ;
2007-05-06 10:15:01 -04:00
struct btrfs_fs_info * info = root - > fs_info ;
struct radix_tree_root * radix ;
2007-04-26 16:46:15 -04:00
struct btrfs_key key ;
struct btrfs_key found_key ;
struct btrfs_leaf * leaf ;
2007-06-12 07:43:08 -04:00
u64 group_size_blocks ;
2007-04-30 15:25:45 -04:00
u64 used ;
2007-04-26 16:46:15 -04:00
2007-06-12 07:43:08 -04:00
group_size_blocks = BTRFS_BLOCK_GROUP_SIZE > >
root - > fs_info - > sb - > s_blocksize_bits ;
2007-05-06 10:15:01 -04:00
root = info - > extent_root ;
2007-04-26 16:46:15 -04:00
key . objectid = 0 ;
key . offset = group_size_blocks ;
key . flags = 0 ;
btrfs_set_key_type ( & key , BTRFS_BLOCK_GROUP_ITEM_KEY ) ;
path = btrfs_alloc_path ( ) ;
if ( ! path )
return - ENOMEM ;
while ( 1 ) {
2007-05-06 10:15:01 -04:00
ret = btrfs_search_slot ( NULL , info - > extent_root ,
2007-04-26 16:46:15 -04:00
& key , path , 0 , 0 ) ;
if ( ret ! = 0 ) {
err = ret ;
break ;
}
leaf = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
btrfs_disk_key_to_cpu ( & found_key ,
& leaf - > items [ path - > slots [ 0 ] ] . key ) ;
cache = kmalloc ( sizeof ( * cache ) , GFP_NOFS ) ;
if ( ! cache ) {
err = - 1 ;
break ;
}
2007-05-07 20:03:49 -04:00
2007-05-29 16:52:18 -04:00
bi = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
struct btrfs_block_group_item ) ;
if ( bi - > flags & BTRFS_BLOCK_GROUP_DATA ) {
2007-05-07 20:03:49 -04:00
radix = & info - > block_group_data_radix ;
2007-05-29 16:52:18 -04:00
cache - > data = 1 ;
} else {
2007-05-07 20:03:49 -04:00
radix = & info - > block_group_radix ;
2007-05-29 16:52:18 -04:00
cache - > data = 0 ;
}
2007-05-07 20:03:49 -04:00
2007-04-26 16:46:15 -04:00
memcpy ( & cache - > item , bi , sizeof ( * bi ) ) ;
memcpy ( & cache - > key , & found_key , sizeof ( found_key ) ) ;
2007-04-30 15:25:45 -04:00
cache - > last_alloc = cache - > key . objectid ;
cache - > first_free = cache - > key . objectid ;
2007-05-06 10:15:01 -04:00
cache - > pinned = 0 ;
2007-05-09 20:13:14 -04:00
cache - > cached = 0 ;
2007-05-07 20:03:49 -04:00
cache - > radix = radix ;
2007-04-26 16:46:15 -04:00
key . objectid = found_key . objectid + found_key . offset ;
btrfs_release_path ( root , path ) ;
2007-05-06 10:15:01 -04:00
ret = radix_tree_insert ( radix , found_key . objectid +
2007-04-26 16:46:15 -04:00
found_key . offset - 1 ,
( void * ) cache ) ;
BUG_ON ( ret ) ;
2007-04-30 15:25:45 -04:00
used = btrfs_block_group_used ( bi ) ;
2007-06-12 07:43:08 -04:00
if ( used < div_factor ( key . offset , 8 ) ) {
2007-05-06 10:15:01 -04:00
radix_tree_tag_set ( radix , found_key . objectid +
2007-04-30 15:25:45 -04:00
found_key . offset - 1 ,
BTRFS_BLOCK_GROUP_AVAIL ) ;
}
2007-04-26 16:46:15 -04:00
if ( key . objectid > =
2007-06-26 10:06:50 -04:00
btrfs_super_total_blocks ( & info - > super_copy ) )
2007-04-26 16:46:15 -04:00
break ;
}
btrfs_free_path ( path ) ;
return 0 ;
}