2007-03-21 18:12:56 +03:00
# include <linux/module.h>
2007-02-26 18:40:21 +03:00
# include "ctree.h"
# include "disk-io.h"
# include "print-tree.h"
2007-03-16 23:20:31 +03:00
# include "transaction.h"
2007-02-26 18:40:21 +03:00
2007-03-16 23:20:31 +03:00
static int find_free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
* orig_root , u64 num_blocks , u64 search_start , u64
search_end , struct btrfs_key * ins ) ;
static int finish_current_insert ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root ) ;
2007-03-22 19:13:20 +03:00
static int del_pending_extents ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root ) ;
2007-02-26 18:40:21 +03:00
2007-03-16 23:20:31 +03:00
static int inc_block_ref ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-27 14:33:00 +04:00
* root , u64 blocknr , u64 num_blocks )
2007-03-03 00:08:05 +03:00
{
2007-04-02 19:20:42 +04:00
struct btrfs_path * path ;
2007-03-03 00:08:05 +03:00
int ret ;
2007-03-12 23:22:34 +03:00
struct btrfs_key key ;
2007-03-13 17:46:10 +03:00
struct btrfs_leaf * l ;
struct btrfs_extent_item * item ;
2007-03-12 23:22:34 +03:00
struct btrfs_key ins ;
2007-03-13 16:49:06 +03:00
u32 refs ;
2007-03-07 19:50:24 +03:00
2007-03-20 21:38:32 +03:00
find_free_extent ( trans , root - > fs_info - > extent_root , 0 , 0 , ( u64 ) - 1 ,
& ins ) ;
2007-04-02 19:20:42 +04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
btrfs_init_path ( path ) ;
2007-03-03 00:08:05 +03:00
key . objectid = blocknr ;
key . flags = 0 ;
2007-03-15 19:56:47 +03:00
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-27 14:33:00 +04:00
key . offset = num_blocks ;
2007-04-02 19:20:42 +04:00
ret = btrfs_search_slot ( trans , root - > fs_info - > extent_root , & key , path ,
2007-03-20 21:38:32 +03:00
0 , 1 ) ;
2007-03-07 04:08:01 +03:00
if ( ret ! = 0 )
BUG ( ) ;
2007-03-03 00:08:05 +03:00
BUG_ON ( ret ! = 0 ) ;
2007-04-02 19: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 16:49:06 +03:00
refs = btrfs_extent_refs ( item ) ;
btrfs_set_extent_refs ( item , refs + 1 ) ;
2007-04-02 19:20:42 +04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
2007-03-07 04:08:01 +03:00
2007-04-02 19:20:42 +04:00
btrfs_release_path ( root - > fs_info - > extent_root , path ) ;
btrfs_free_path ( path ) ;
2007-03-20 21:38:32 +03:00
finish_current_insert ( trans , root - > fs_info - > extent_root ) ;
2007-03-22 19:13:20 +03:00
del_pending_extents ( trans , root - > fs_info - > extent_root ) ;
2007-03-03 00:08:05 +03:00
return 0 ;
}
2007-03-16 23:20:31 +03:00
static int lookup_block_ref ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-27 14:33:00 +04:00
* root , u64 blocknr , u64 num_blocks , u32 * refs )
2007-03-07 04:08:01 +03:00
{
2007-04-02 19:20:42 +04:00
struct btrfs_path * path ;
2007-03-07 04:08:01 +03:00
int ret ;
2007-03-12 23:22:34 +03:00
struct btrfs_key key ;
2007-03-13 17:46:10 +03:00
struct btrfs_leaf * l ;
struct btrfs_extent_item * item ;
2007-04-02 19:20:42 +04:00
path = btrfs_alloc_path ( ) ;
btrfs_init_path ( path ) ;
2007-03-07 04:08:01 +03:00
key . objectid = blocknr ;
2007-03-27 14:33:00 +04:00
key . offset = num_blocks ;
2007-03-15 19:56:47 +03:00
key . flags = 0 ;
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-04-02 19:20:42 +04:00
ret = btrfs_search_slot ( trans , root - > fs_info - > extent_root , & key , path ,
2007-03-20 21:38:32 +03:00
0 , 0 ) ;
2007-03-07 04:08:01 +03:00
if ( ret ! = 0 )
BUG ( ) ;
2007-04-02 19: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 16:49:06 +03:00
* refs = btrfs_extent_refs ( item ) ;
2007-04-02 19:20:42 +04:00
btrfs_release_path ( root - > fs_info - > extent_root , path ) ;
btrfs_free_path ( path ) ;
2007-03-07 04:08:01 +03:00
return 0 ;
}
2007-04-10 17:27:04 +04:00
int btrfs_inc_root_ref ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root )
{
return inc_block_ref ( trans , root , root - > node - > b_blocknr , 1 ) ;
}
2007-03-16 23:20:31 +03:00
int btrfs_inc_ref ( struct btrfs_trans_handle * trans , struct btrfs_root * root ,
2007-03-22 19:13:20 +03:00
struct buffer_head * buf )
2007-03-03 00:08:05 +03:00
{
u64 blocknr ;
2007-03-22 19:13:20 +03:00
struct btrfs_node * buf_node ;
2007-03-27 14:33:00 +04:00
struct btrfs_leaf * buf_leaf ;
struct btrfs_disk_key * key ;
struct btrfs_file_extent_item * fi ;
2007-03-03 00:08:05 +03:00
int i ;
2007-03-27 14:33:00 +04:00
int leaf ;
int ret ;
2007-03-07 04:08:01 +03:00
2007-03-13 23:47:54 +03:00
if ( ! root - > ref_cows )
2007-03-07 04:08:01 +03:00
return 0 ;
2007-03-22 19:13:20 +03:00
buf_node = btrfs_buffer_node ( buf ) ;
2007-03-27 14:33:00 +04:00
leaf = btrfs_is_leaf ( buf_node ) ;
buf_leaf = btrfs_buffer_leaf ( buf ) ;
2007-03-22 19:13:20 +03:00
for ( i = 0 ; i < btrfs_header_nritems ( & buf_node - > header ) ; i + + ) {
2007-03-27 14:33:00 +04:00
if ( leaf ) {
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 ) ;
ret = inc_block_ref ( trans , root ,
btrfs_file_extent_disk_blocknr ( fi ) ,
btrfs_file_extent_disk_num_blocks ( fi ) ) ;
BUG_ON ( ret ) ;
} else {
blocknr = btrfs_node_blockptr ( buf_node , i ) ;
ret = inc_block_ref ( trans , root , blocknr , 1 ) ;
BUG_ON ( ret ) ;
}
2007-03-03 00:08:05 +03:00
}
return 0 ;
}
2007-03-16 23:20:31 +03:00
int btrfs_finish_extent_commit ( struct btrfs_trans_handle * trans , struct
btrfs_root * root )
2007-03-07 04:08:01 +03:00
{
2007-03-26 18:15:30 +04:00
unsigned long gang [ 8 ] ;
2007-03-16 15:56:18 +03:00
u64 first = 0 ;
2007-03-07 04:08:01 +03:00
int ret ;
int i ;
2007-03-26 18:15:30 +04:00
struct radix_tree_root * pinned_radix = & root - > fs_info - > pinned_radix ;
2007-03-07 04:08:01 +03:00
while ( 1 ) {
2007-03-26 18:15:30 +04:00
ret = find_first_radix_bit ( pinned_radix , gang ,
ARRAY_SIZE ( gang ) ) ;
2007-03-07 04:08:01 +03:00
if ( ! ret )
break ;
2007-03-16 15:56:18 +03:00
if ( ! first )
2007-03-26 18:15:30 +04:00
first = gang [ 0 ] ;
2007-03-08 00:15:30 +03:00
for ( i = 0 ; i < ret ; i + + ) {
2007-03-26 18:15:30 +04:00
clear_radix_bit ( pinned_radix , gang [ i ] ) ;
2007-03-08 00:15:30 +03:00
}
2007-03-07 04:08:01 +03:00
}
2007-03-23 17:01:08 +03:00
if ( root - > fs_info - > last_insert . objectid > first )
root - > fs_info - > last_insert . objectid = first ;
2007-03-20 21:38:32 +03:00
root - > fs_info - > last_insert . offset = 0 ;
2007-03-07 04:08:01 +03:00
return 0 ;
}
2007-03-16 23:20:31 +03:00
static int finish_current_insert ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root )
2007-03-07 19:50:24 +03:00
{
2007-03-12 23:22:34 +03:00
struct btrfs_key ins ;
2007-03-13 17:46:10 +03:00
struct btrfs_extent_item extent_item ;
2007-03-07 19:50:24 +03:00
int i ;
int ret ;
2007-03-21 03:35:03 +03:00
u64 super_blocks_used ;
struct btrfs_fs_info * info = extent_root - > fs_info ;
2007-03-07 19:50:24 +03:00
2007-03-13 16:49:06 +03:00
btrfs_set_extent_refs ( & extent_item , 1 ) ;
2007-03-07 19:50:24 +03:00
ins . offset = 1 ;
ins . flags = 0 ;
2007-03-15 19:56:47 +03:00
btrfs_set_key_type ( & ins , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-07 19:50:24 +03:00
2007-03-20 21:38:32 +03:00
for ( i = 0 ; i < extent_root - > fs_info - > current_insert . flags ; i + + ) {
ins . objectid = extent_root - > fs_info - > current_insert . objectid +
i ;
2007-03-21 03:35:03 +03:00
super_blocks_used = btrfs_super_blocks_used ( info - > disk_super ) ;
btrfs_set_super_blocks_used ( info - > disk_super ,
super_blocks_used + 1 ) ;
2007-03-16 23:20:31 +03:00
ret = btrfs_insert_item ( trans , extent_root , & ins , & extent_item ,
sizeof ( extent_item ) ) ;
2007-03-07 19:50:24 +03:00
BUG_ON ( ret ) ;
}
2007-03-20 21:38:32 +03:00
extent_root - > fs_info - > current_insert . offset = 0 ;
2007-03-07 19:50:24 +03:00
return 0 ;
}
2007-03-26 18:15:30 +04:00
static int pin_down_block ( struct btrfs_root * root , u64 blocknr , int pending )
2007-03-22 19:13:20 +03:00
{
int err ;
2007-03-25 19:35:08 +04:00
struct btrfs_header * header ;
2007-03-26 18:15:30 +04:00
struct buffer_head * bh ;
2007-03-27 19:05:53 +04:00
if ( ! pending ) {
2007-03-28 21:57:48 +04:00
bh = btrfs_find_tree_block ( root , blocknr ) ;
2007-04-02 18: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 19:05:53 +04:00
}
2007-03-30 22:27:56 +04:00
btrfs_block_release ( root , bh ) ;
2007-03-26 18:15:30 +04:00
}
err = set_radix_bit ( & root - > fs_info - > pinned_radix , blocknr ) ;
2007-03-27 19:05:53 +04:00
} else {
err = set_radix_bit ( & root - > fs_info - > pending_del_radix , blocknr ) ;
}
2007-03-26 18:15:30 +04:00
BUG_ON ( err ) ;
2007-03-22 19:13:20 +03:00
return 0 ;
}
2007-02-26 18:40:21 +03:00
/*
2007-03-07 04:08:01 +03:00
* remove an extent from the root , returns 0 on success
2007-02-26 18:40:21 +03:00
*/
2007-03-16 23:20:31 +03:00
static int __free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-25 19:35:08 +04:00
* root , u64 blocknr , u64 num_blocks , int pin )
2007-03-07 04:08:01 +03:00
{
2007-04-02 19:20:42 +04:00
struct btrfs_path * path ;
2007-03-12 23:22:34 +03:00
struct btrfs_key key ;
2007-03-21 03:35:03 +03:00
struct btrfs_fs_info * info = root - > fs_info ;
struct btrfs_root * extent_root = info - > extent_root ;
2007-03-07 04:08:01 +03:00
int ret ;
2007-03-13 17:46:10 +03:00
struct btrfs_extent_item * ei ;
2007-03-12 23:22:34 +03:00
struct btrfs_key ins ;
2007-03-13 16:49:06 +03:00
u32 refs ;
2007-03-07 19:50:24 +03:00
2007-03-07 04:08:01 +03:00
key . objectid = blocknr ;
key . flags = 0 ;
2007-03-15 19:56:47 +03:00
btrfs_set_key_type ( & key , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-07 04:08:01 +03:00
key . offset = num_blocks ;
2007-03-16 23:20:31 +03:00
find_free_extent ( trans , root , 0 , 0 , ( u64 ) - 1 , & ins ) ;
2007-04-02 19:20:42 +04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
btrfs_init_path ( path ) ;
2007-04-05 18:38:44 +04:00
2007-04-02 19:20:42 +04:00
ret = btrfs_search_slot ( trans , extent_root , & key , path , - 1 , 1 ) ;
2007-03-07 04:08:01 +03:00
if ( ret ) {
2007-03-21 18:12:56 +03:00
printk ( " failed to find %Lu \n " , key . objectid ) ;
2007-03-13 17:46:10 +03:00
btrfs_print_tree ( extent_root , extent_root - > node ) ;
2007-03-21 18:12:56 +03:00
printk ( " failed to find %Lu \n " , key . objectid ) ;
2007-03-07 04:08:01 +03:00
BUG ( ) ;
}
2007-04-02 19:20:42 +04:00
ei = btrfs_item_ptr ( btrfs_buffer_leaf ( path - > nodes [ 0 ] ) , path - > slots [ 0 ] ,
2007-03-14 21:14:43 +03:00
struct btrfs_extent_item ) ;
2007-03-07 04:08:01 +03:00
BUG_ON ( ei - > refs = = 0 ) ;
2007-03-13 16:49:06 +03:00
refs = btrfs_extent_refs ( ei ) - 1 ;
btrfs_set_extent_refs ( ei , refs ) ;
2007-04-02 19:20:42 +04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
2007-03-13 16:49:06 +03:00
if ( refs = = 0 ) {
2007-03-21 03:35:03 +03:00
u64 super_blocks_used ;
2007-03-25 19:35:08 +04:00
if ( pin ) {
2007-03-26 18:15:30 +04:00
ret = pin_down_block ( root , blocknr , 0 ) ;
2007-03-25 19:35:08 +04:00
BUG_ON ( ret ) ;
}
2007-03-21 03:35:03 +03:00
super_blocks_used = btrfs_super_blocks_used ( info - > disk_super ) ;
btrfs_set_super_blocks_used ( info - > disk_super ,
super_blocks_used - num_blocks ) ;
2007-04-02 19:20:42 +04:00
ret = btrfs_del_item ( trans , extent_root , path ) ;
2007-03-07 04:08:01 +03:00
if ( ret )
BUG ( ) ;
}
2007-04-02 19:20:42 +04:00
btrfs_release_path ( extent_root , path ) ;
btrfs_free_path ( path ) ;
2007-03-16 23:20:31 +03:00
finish_current_insert ( trans , extent_root ) ;
2007-03-07 04:08:01 +03:00
return ret ;
}
/*
* find all the blocks marked as pending in the radix tree and remove
* them from the extent map
*/
2007-03-16 23:20:31 +03:00
static int del_pending_extents ( struct btrfs_trans_handle * trans , struct
btrfs_root * extent_root )
2007-03-07 04:08:01 +03:00
{
int ret ;
2007-03-22 19:13:20 +03:00
int wret ;
int err = 0 ;
2007-03-26 18:15:30 +04:00
unsigned long gang [ 4 ] ;
2007-03-07 04:08:01 +03:00
int i ;
2007-03-26 18:15:30 +04:00
struct radix_tree_root * pending_radix ;
struct radix_tree_root * pinned_radix ;
pending_radix = & extent_root - > fs_info - > pending_del_radix ;
pinned_radix = & extent_root - > fs_info - > pinned_radix ;
2007-03-07 04:08:01 +03:00
while ( 1 ) {
2007-03-26 18:15:30 +04:00
ret = find_first_radix_bit ( pending_radix , gang ,
ARRAY_SIZE ( gang ) ) ;
2007-03-07 04:08:01 +03:00
if ( ! ret )
break ;
for ( i = 0 ; i < ret ; i + + ) {
2007-03-26 18:15:30 +04:00
wret = set_radix_bit ( pinned_radix , gang [ i ] ) ;
BUG_ON ( wret ) ;
wret = clear_radix_bit ( pending_radix , gang [ i ] ) ;
BUG_ON ( wret ) ;
2007-03-23 17:01:08 +03:00
wret = __free_extent ( trans , extent_root ,
2007-03-26 18:15:30 +04:00
gang [ i ] , 1 , 0 ) ;
2007-03-22 19:13:20 +03:00
if ( wret )
err = wret ;
2007-02-26 18:40:21 +03:00
}
}
2007-03-22 19:13:20 +03:00
return err ;
2007-02-26 18:40:21 +03:00
}
/*
* remove an extent from the root , returns 0 on success
*/
2007-03-16 23:20:31 +03:00
int btrfs_free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , u64 blocknr , u64 num_blocks , int pin )
2007-02-26 18:40:21 +03:00
{
2007-03-20 21:38:32 +03:00
struct btrfs_root * extent_root = root - > fs_info - > extent_root ;
2007-02-26 18:40:21 +03:00
int pending_ret ;
int ret ;
2007-03-07 04:08:01 +03:00
2007-02-26 18:40:21 +03:00
if ( root = = extent_root ) {
2007-03-26 18:15:30 +04:00
pin_down_block ( root , blocknr , 1 ) ;
2007-02-26 18:40:21 +03:00
return 0 ;
}
2007-03-25 19:35:08 +04:00
ret = __free_extent ( trans , root , blocknr , num_blocks , pin ) ;
2007-03-22 19:13:20 +03:00
pending_ret = del_pending_extents ( trans , root - > fs_info - > extent_root ) ;
2007-02-26 18:40:21 +03: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 19:56:47 +03:00
* ins - > flags = BTRFS_EXTENT_ITEM_KEY
2007-02-26 18:40:21 +03:00
* ins - > offset = = number of blocks
* Any available blocks before search_start are skipped .
*/
2007-03-16 23:20:31 +03:00
static int find_free_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
* orig_root , u64 num_blocks , u64 search_start , u64
search_end , struct btrfs_key * ins )
2007-02-26 18:40:21 +03:00
{
2007-04-02 19:20:42 +04:00
struct btrfs_path * path ;
2007-03-12 23:22:34 +03:00
struct btrfs_key key ;
2007-02-26 18:40:21 +03:00
int ret ;
u64 hole_size = 0 ;
int slot = 0 ;
2007-03-22 19:13:20 +03:00
u64 last_block = 0 ;
2007-03-07 19:50:24 +03:00
u64 test_block ;
2007-02-26 18:40:21 +03:00
int start_found ;
2007-03-13 17:46:10 +03:00
struct btrfs_leaf * l ;
2007-03-20 21:38:32 +03:00
struct btrfs_root * root = orig_root - > fs_info - > extent_root ;
2007-03-08 00:15:30 +03:00
int total_needed = num_blocks ;
2007-03-22 19:13:20 +03:00
int level ;
2007-02-26 18:40:21 +03:00
2007-04-04 23:27:52 +04:00
path = btrfs_alloc_path ( ) ;
ins - > flags = 0 ;
btrfs_set_key_type ( ins , BTRFS_EXTENT_ITEM_KEY ) ;
2007-03-22 19:13:20 +03:00
level = btrfs_header_level ( btrfs_buffer_header ( root - > node ) ) ;
total_needed + = ( level + 1 ) * 3 ;
2007-04-04 23:27:52 +04:00
if ( root - > fs_info - > last_insert . objectid = = 0 & & search_end = = ( u64 ) - 1 ) {
struct btrfs_disk_key * last_key ;
btrfs_init_path ( path ) ;
ins - > objectid = ( u64 ) - 1 ;
ins - > offset = ( u64 ) - 1 ;
ret = btrfs_search_slot ( trans , root , ins , path , 0 , 0 ) ;
if ( ret < 0 )
goto error ;
BUG_ON ( ret = = 0 ) ;
if ( path - > slots [ 0 ] > 0 )
path - > slots [ 0 ] - - ;
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
last_key = & l - > items [ path - > slots [ 0 ] ] . key ;
search_start = btrfs_disk_key_objectid ( last_key ) ;
}
2007-03-20 21:38:32 +03:00
if ( root - > fs_info - > last_insert . objectid > search_start )
search_start = root - > fs_info - > last_insert . objectid ;
2007-03-15 19:56:47 +03:00
2007-02-26 18:40:21 +03:00
check_failed :
2007-04-02 19:20:42 +04:00
btrfs_init_path ( path ) ;
2007-02-26 18:40:21 +03:00
ins - > objectid = search_start ;
ins - > offset = 0 ;
start_found = 0 ;
2007-04-02 19:20:42 +04:00
ret = btrfs_search_slot ( trans , root , ins , path , 0 , 0 ) ;
2007-03-01 00:46:22 +03:00
if ( ret < 0 )
goto error ;
2007-03-01 00:35:06 +03:00
2007-04-02 19:20:42 +04:00
if ( path - > slots [ 0 ] > 0 )
path - > slots [ 0 ] - - ;
2007-03-08 00:15:30 +03:00
2007-02-26 18:40:21 +03:00
while ( 1 ) {
2007-04-02 19:20:42 +04:00
l = btrfs_buffer_leaf ( path - > nodes [ 0 ] ) ;
slot = path - > slots [ 0 ] ;
2007-03-12 19:01:18 +03:00
if ( slot > = btrfs_header_nritems ( & l - > header ) ) {
2007-04-02 19:20:42 +04:00
ret = btrfs_next_leaf ( root , path ) ;
2007-02-26 18:40:21 +03:00
if ( ret = = 0 )
continue ;
2007-03-01 00:46:22 +03:00
if ( ret < 0 )
goto error ;
2007-02-26 18:40:21 +03:00
if ( ! start_found ) {
ins - > objectid = search_start ;
2007-03-07 19:50:24 +03:00
ins - > offset = ( u64 ) - 1 ;
2007-02-26 18:40:21 +03:00
start_found = 1 ;
goto check_pending ;
}
ins - > objectid = last_block > search_start ?
last_block : search_start ;
2007-03-07 19:50:24 +03:00
ins - > offset = ( u64 ) - 1 ;
2007-02-26 18:40:21 +03:00
goto check_pending ;
}
2007-03-12 23:22:34 +03:00
btrfs_disk_key_to_cpu ( & key , & l - > items [ slot ] . key ) ;
if ( key . objectid > = search_start ) {
2007-02-26 18:40:21 +03:00
if ( start_found ) {
2007-03-08 00:15:30 +03:00
if ( last_block < search_start )
last_block = search_start ;
2007-03-12 23:22:34 +03:00
hole_size = key . objectid - last_block ;
2007-03-07 19:50:24 +03:00
if ( hole_size > total_needed ) {
2007-02-26 18:40:21 +03:00
ins - > objectid = last_block ;
2007-03-07 19:50:24 +03:00
ins - > offset = hole_size ;
2007-02-26 18:40:21 +03:00
goto check_pending ;
}
2007-03-08 00:15:30 +03:00
}
2007-02-26 18:40:21 +03:00
}
2007-03-08 00:15:30 +03:00
start_found = 1 ;
2007-03-12 23:22:34 +03:00
last_block = key . objectid + key . offset ;
2007-04-02 19:20:42 +04:00
path - > slots [ 0 ] + + ;
2007-02-26 18:40:21 +03:00
}
// FIXME -ENOSPC
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 19:20:42 +04:00
btrfs_release_path ( root , path ) ;
2007-02-26 18:40:21 +03:00
BUG_ON ( ins - > objectid < search_start ) ;
2007-03-07 19:50:24 +03:00
for ( test_block = ins - > objectid ;
test_block < ins - > objectid + total_needed ; test_block + + ) {
2007-03-26 18:15:30 +04:00
if ( test_radix_bit ( & root - > fs_info - > pinned_radix ,
2007-03-20 21:38:32 +03:00
test_block ) ) {
2007-03-07 19:50:24 +03:00
search_start = test_block + 1 ;
2007-02-26 18:40:21 +03:00
goto check_failed ;
}
}
2007-03-20 21:38:32 +03:00
BUG_ON ( root - > fs_info - > current_insert . offset ) ;
root - > fs_info - > current_insert . offset = total_needed - num_blocks ;
root - > fs_info - > current_insert . objectid = ins - > objectid + num_blocks ;
root - > fs_info - > current_insert . flags = 0 ;
root - > fs_info - > last_insert . objectid = ins - > objectid ;
2007-03-07 19:50:24 +03:00
ins - > offset = num_blocks ;
2007-04-02 19:20:42 +04:00
btrfs_free_path ( path ) ;
2007-02-26 18:40:21 +03:00
return 0 ;
2007-03-01 00:46:22 +03:00
error :
2007-04-02 19:20:42 +04:00
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
2007-03-01 00:46:22 +03:00
return ret ;
2007-02-26 18:40:21 +03: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-03-27 00:00:06 +04:00
int btrfs_alloc_extent ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-16 23:20:31 +03:00
* root , u64 num_blocks , u64 search_start , u64
2007-04-10 20:36:36 +04:00
search_end , struct btrfs_key * ins )
2007-02-26 18:40:21 +03:00
{
int ret ;
int pending_ret ;
2007-03-21 03:35:03 +03:00
u64 super_blocks_used ;
struct btrfs_fs_info * info = root - > fs_info ;
struct btrfs_root * extent_root = info - > extent_root ;
2007-03-13 17:46:10 +03:00
struct btrfs_extent_item extent_item ;
2007-03-07 19:50:24 +03:00
2007-03-13 16:49:06 +03:00
btrfs_set_extent_refs ( & extent_item , 1 ) ;
2007-02-26 18:40:21 +03:00
2007-03-07 19:50:24 +03:00
if ( root = = extent_root ) {
2007-03-20 21:38:32 +03:00
BUG_ON ( extent_root - > fs_info - > current_insert . offset = = 0 ) ;
2007-03-07 19:50:24 +03:00
BUG_ON ( num_blocks ! = 1 ) ;
2007-03-20 21:38:32 +03:00
BUG_ON ( extent_root - > fs_info - > current_insert . flags = =
extent_root - > fs_info - > current_insert . offset ) ;
2007-03-07 19:50:24 +03:00
ins - > offset = 1 ;
2007-03-20 21:38:32 +03:00
ins - > objectid = extent_root - > fs_info - > current_insert . objectid +
extent_root - > fs_info - > current_insert . flags + + ;
2007-02-26 18:40:21 +03:00
return 0 ;
}
2007-03-16 23:20:31 +03:00
ret = find_free_extent ( trans , root , num_blocks , search_start ,
2007-03-07 19:50:24 +03:00
search_end , ins ) ;
if ( ret )
return ret ;
2007-02-26 18:40:21 +03:00
2007-03-21 03:35:03 +03:00
super_blocks_used = btrfs_super_blocks_used ( info - > disk_super ) ;
btrfs_set_super_blocks_used ( info - > disk_super , super_blocks_used +
num_blocks ) ;
2007-03-16 23:20:31 +03:00
ret = btrfs_insert_item ( trans , extent_root , ins , & extent_item ,
sizeof ( extent_item ) ) ;
2007-03-07 19:50:24 +03:00
2007-03-16 23:20:31 +03:00
finish_current_insert ( trans , extent_root ) ;
2007-03-22 19:13:20 +03:00
pending_ret = del_pending_extents ( trans , extent_root ) ;
2007-03-07 19:50:24 +03:00
if ( ret )
return ret ;
if ( pending_ret )
return pending_ret ;
return 0 ;
2007-02-26 18:40:21 +03:00
}
/*
* helper function to allocate a block for a given tree
* returns the tree buffer or NULL .
*/
2007-03-22 19:13:20 +03:00
struct buffer_head * btrfs_alloc_free_block ( struct btrfs_trans_handle * trans ,
2007-03-16 23:20:31 +03:00
struct btrfs_root * root )
2007-02-26 18:40:21 +03:00
{
2007-03-12 23:22:34 +03:00
struct btrfs_key ins ;
2007-02-26 18:40:21 +03:00
int ret ;
2007-03-22 19:13:20 +03:00
struct buffer_head * buf ;
2007-02-26 18:40:21 +03:00
2007-04-10 20:36:36 +04:00
ret = btrfs_alloc_extent ( trans , root , 1 , 0 , ( unsigned long ) - 1 , & ins ) ;
2007-02-26 18:40:21 +03:00
if ( ret ) {
BUG ( ) ;
return NULL ;
}
2007-03-28 21:57:48 +04:00
buf = btrfs_find_create_tree_block ( root , ins . objectid ) ;
2007-03-23 18:00:45 +03:00
set_buffer_uptodate ( buf ) ;
2007-02-26 18:40:21 +03:00
return buf ;
}
2007-03-07 04:08:01 +03:00
2007-03-27 14: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 + + ) {
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 ) ;
/*
* FIXME make sure to insert a trans record that
* repeats the snapshot del on crash
*/
ret = btrfs_free_extent ( trans , root ,
btrfs_file_extent_disk_blocknr ( fi ) ,
btrfs_file_extent_disk_num_blocks ( fi ) ,
0 ) ;
BUG_ON ( ret ) ;
}
return 0 ;
}
2007-03-13 18:09:37 +03:00
/*
* helper function for drop_snapshot , this walks down the tree dropping ref
* counts as it goes .
*/
2007-03-16 23:20:31 +03:00
static int walk_down_tree ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , struct btrfs_path * path , int * level )
2007-03-10 14:35:47 +03:00
{
2007-03-22 19:13:20 +03:00
struct buffer_head * next ;
struct buffer_head * cur ;
2007-03-10 14:35:47 +03:00
u64 blocknr ;
int ret ;
u32 refs ;
2007-04-02 19:20:42 +04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-03-22 19:13:20 +03:00
ret = lookup_block_ref ( trans , root , path - > nodes [ * level ] - > b_blocknr ,
2007-03-27 14:33:00 +04:00
1 , & refs ) ;
2007-03-10 14:35:47 +03:00
BUG_ON ( ret ) ;
if ( refs > 1 )
goto out ;
2007-03-13 18:09:37 +03:00
/*
* walk down to the last node level and free all the leaves
*/
2007-03-27 14:33:00 +04:00
while ( * level > = 0 ) {
2007-04-02 19:20:42 +04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-03-10 14:35:47 +03:00
cur = path - > nodes [ * level ] ;
2007-04-02 18:50:19 +04:00
if ( btrfs_header_level ( btrfs_buffer_header ( cur ) ) ! = * level )
WARN_ON ( 1 ) ;
2007-03-12 19:01:18 +03:00
if ( path - > slots [ * level ] > =
2007-03-22 19:13:20 +03:00
btrfs_header_nritems ( btrfs_buffer_header ( cur ) ) )
2007-03-10 14:35:47 +03:00
break ;
2007-03-27 14:33:00 +04:00
if ( * level = = 0 ) {
ret = drop_leaf_ref ( trans , root , cur ) ;
BUG_ON ( ret ) ;
break ;
}
2007-03-22 19:13:20 +03:00
blocknr = btrfs_node_blockptr ( btrfs_buffer_node ( cur ) ,
path - > slots [ * level ] ) ;
2007-03-27 14:33:00 +04:00
ret = lookup_block_ref ( trans , root , blocknr , 1 , & refs ) ;
BUG_ON ( ret ) ;
if ( refs ! = 1 ) {
2007-03-10 14:35:47 +03:00
path - > slots [ * level ] + + ;
2007-03-16 23:20:31 +03:00
ret = btrfs_free_extent ( trans , root , blocknr , 1 , 1 ) ;
2007-03-10 14:35:47 +03:00
BUG_ON ( ret ) ;
continue ;
}
next = read_tree_block ( root , blocknr ) ;
2007-04-02 19:20:42 +04:00
WARN_ON ( * level < = 0 ) ;
2007-03-12 16:03:27 +03:00
if ( path - > nodes [ * level - 1 ] )
2007-03-13 17:46:10 +03:00
btrfs_block_release ( root , path - > nodes [ * level - 1 ] ) ;
2007-03-10 14:35:47 +03:00
path - > nodes [ * level - 1 ] = next ;
2007-03-22 19:13:20 +03:00
* level = btrfs_header_level ( btrfs_buffer_header ( next ) ) ;
2007-03-10 14:35:47 +03:00
path - > slots [ * level ] = 0 ;
}
out :
2007-04-02 19:20:42 +04:00
WARN_ON ( * level < 0 ) ;
WARN_ON ( * level > = BTRFS_MAX_LEVEL ) ;
2007-03-27 14:33:00 +04:00
ret = btrfs_free_extent ( trans , root ,
path - > nodes [ * level ] - > b_blocknr , 1 , 1 ) ;
2007-03-13 17:46:10 +03:00
btrfs_block_release ( root , path - > nodes [ * level ] ) ;
2007-03-10 14:35:47 +03:00
path - > nodes [ * level ] = NULL ;
* level + = 1 ;
BUG_ON ( ret ) ;
return 0 ;
}
2007-03-13 18:09:37 +03: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 23:20:31 +03:00
static int walk_up_tree ( struct btrfs_trans_handle * trans , struct btrfs_root
* root , struct btrfs_path * path , int * level )
2007-03-10 14:35:47 +03:00
{
int i ;
int slot ;
int ret ;
2007-03-13 17:46:10 +03:00
for ( i = * level ; i < BTRFS_MAX_LEVEL - 1 & & path - > nodes [ i ] ; i + + ) {
2007-03-10 14:35:47 +03:00
slot = path - > slots [ i ] ;
2007-03-22 19:13:20 +03:00
if ( slot < btrfs_header_nritems (
btrfs_buffer_header ( path - > nodes [ i ] ) ) - 1 ) {
2007-03-10 14:35:47 +03:00
path - > slots [ i ] + + ;
* level = i ;
return 0 ;
} else {
2007-03-16 23:20:31 +03:00
ret = btrfs_free_extent ( trans , root ,
2007-03-22 19:13:20 +03:00
path - > nodes [ * level ] - > b_blocknr ,
2007-03-16 23:20:31 +03:00
1 , 1 ) ;
2007-03-27 14:33:00 +04:00
BUG_ON ( ret ) ;
2007-03-13 17:46:10 +03:00
btrfs_block_release ( root , path - > nodes [ * level ] ) ;
2007-03-12 16:03:27 +03:00
path - > nodes [ * level ] = NULL ;
2007-03-10 14:35:47 +03:00
* level = i + 1 ;
}
}
return 1 ;
}
2007-03-13 18:09:37 +03: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 23:20:31 +03:00
int btrfs_drop_snapshot ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-22 19:13:20 +03:00
* root , struct buffer_head * snap )
2007-03-10 14:35:47 +03:00
{
2007-03-13 23:47:54 +03:00
int ret = 0 ;
2007-03-13 18:09:37 +03:00
int wret ;
2007-03-10 14:35:47 +03:00
int level ;
2007-04-02 19:20:42 +04:00
struct btrfs_path * path ;
2007-03-10 14:35:47 +03:00
int i ;
int orig_level ;
2007-04-02 19:20:42 +04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
btrfs_init_path ( path ) ;
2007-03-10 14:35:47 +03:00
2007-03-22 19:13:20 +03:00
level = btrfs_header_level ( btrfs_buffer_header ( snap ) ) ;
2007-03-10 14:35:47 +03:00
orig_level = level ;
2007-04-02 19:20:42 +04:00
path - > nodes [ level ] = snap ;
path - > slots [ level ] = 0 ;
2007-03-10 14:35:47 +03:00
while ( 1 ) {
2007-04-02 19:20:42 +04:00
wret = walk_down_tree ( trans , root , path , & level ) ;
2007-03-13 18:09:37 +03:00
if ( wret > 0 )
2007-03-10 14:35:47 +03:00
break ;
2007-03-13 18:09:37 +03:00
if ( wret < 0 )
ret = wret ;
2007-04-02 19:20:42 +04:00
wret = walk_up_tree ( trans , root , path , & level ) ;
2007-03-13 18:09:37 +03:00
if ( wret > 0 )
2007-03-10 14:35:47 +03:00
break ;
2007-03-13 18:09:37 +03:00
if ( wret < 0 )
ret = wret ;
2007-03-10 14:35:47 +03:00
}
2007-03-12 16:03:27 +03:00
for ( i = 0 ; i < = orig_level ; i + + ) {
2007-04-02 19:20:42 +04:00
if ( path - > nodes [ i ] ) {
btrfs_block_release ( root , path - > nodes [ i ] ) ;
2007-03-12 16:03:27 +03:00
}
2007-03-10 14:35:47 +03:00
}
2007-04-02 19:20:42 +04:00
btrfs_free_path ( path ) ;
2007-03-13 18:09:37 +03:00
return ret ;
2007-03-10 14:35:47 +03:00
}