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-03-22 12:13:20 -04:00
# include <linux/fs.h>
2007-03-28 13:57:48 -04:00
# include <linux/blkdev.h>
2007-06-22 14:16:24 -04:00
# include <linux/crc32c.h>
2007-03-28 19:44:27 -04:00
# include <linux/scatterlist.h>
2007-03-30 08:47:31 -04:00
# include <linux/swap.h>
2007-04-09 10:42:37 -04:00
# include <linux/radix-tree.h>
2007-05-02 15:53:43 -04:00
# include <linux/writeback.h>
2007-10-15 16:14:19 -04:00
# include <linux/buffer_head.h> // for block_sync_page
2007-02-02 09:18:22 -05:00
# include "ctree.h"
# include "disk-io.h"
2007-03-16 16:20:31 -04:00
# include "transaction.h"
2007-04-09 10:42:37 -04:00
# include "btrfs_inode.h"
2007-10-15 16:15:53 -04:00
# include "print-tree.h"
2007-02-02 09:18:22 -05:00
2007-10-15 16:14:19 -04:00
#if 0
static int check_tree_block ( struct btrfs_root * root , struct extent_buffer * buf )
2007-04-11 15:53:25 -04:00
{
2007-10-15 16:14:19 -04:00
if ( extent_buffer_blocknr ( buf ) ! = btrfs_header_blocknr ( buf ) ) {
printk ( KERN_CRIT " buf blocknr(buf) is %llu, header is %llu \n " ,
( unsigned long long ) extent_buffer_blocknr ( buf ) ,
( unsigned long long ) btrfs_header_blocknr ( buf ) ) ;
2007-06-12 06:35:45 -04:00
return 1 ;
2007-03-28 13:57:48 -04:00
}
2007-02-23 08:38:36 -05:00
return 0 ;
2007-02-02 09:18:22 -05:00
}
2007-10-15 16:14:19 -04:00
# endif
2007-02-02 09:18:22 -05:00
2007-10-15 16:14:19 -04:00
struct extent_buffer * btrfs_find_tree_block ( struct btrfs_root * root ,
2007-10-15 16:15:53 -04:00
u64 bytenr , u32 blocksize )
2007-03-28 13:57:48 -04:00
{
2007-10-15 16:14:19 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
2007-10-15 16:14:48 -04:00
struct extent_buffer * eb ;
eb = find_extent_buffer ( & BTRFS_I ( btree_inode ) - > extent_tree ,
2007-10-15 16:15:53 -04:00
bytenr , blocksize , GFP_NOFS ) ;
2007-10-15 16:14:48 -04:00
return eb ;
2007-10-15 16:14:19 -04:00
}
2007-03-28 13:57:48 -04:00
2007-10-15 16:14:19 -04:00
struct extent_buffer * btrfs_find_create_tree_block ( struct btrfs_root * root ,
2007-10-15 16:15:53 -04:00
u64 bytenr , u32 blocksize )
2007-10-15 16:14:19 -04:00
{
struct inode * btree_inode = root - > fs_info - > btree_inode ;
2007-10-15 16:14:48 -04:00
struct extent_buffer * eb ;
2007-10-15 16:15:53 -04:00
2007-10-15 16:14:48 -04:00
eb = alloc_extent_buffer ( & BTRFS_I ( btree_inode ) - > extent_tree ,
2007-10-15 16:19:22 -04:00
bytenr , blocksize , NULL , GFP_NOFS ) ;
2007-10-15 16:14:48 -04:00
return eb ;
2007-03-28 13:57:48 -04:00
}
2007-10-15 16:14:19 -04:00
struct extent_map * btree_get_extent ( struct inode * inode , struct page * page ,
size_t page_offset , u64 start , u64 end ,
int create )
2007-04-11 15:53:25 -04:00
{
2007-10-15 16:14:19 -04:00
struct extent_map_tree * em_tree = & BTRFS_I ( inode ) - > extent_tree ;
struct extent_map * em ;
int ret ;
again :
em = lookup_extent_mapping ( em_tree , start , end ) ;
if ( em ) {
goto out ;
2007-04-11 15:53:25 -04:00
}
2007-10-15 16:14:19 -04:00
em = alloc_extent_map ( GFP_NOFS ) ;
if ( ! em ) {
em = ERR_PTR ( - ENOMEM ) ;
goto out ;
}
em - > start = 0 ;
em - > end = ( i_size_read ( inode ) & ~ ( ( u64 ) PAGE_CACHE_SIZE - 1 ) ) - 1 ;
em - > block_start = 0 ;
em - > block_end = em - > end ;
em - > bdev = inode - > i_sb - > s_bdev ;
ret = add_extent_mapping ( em_tree , em ) ;
if ( ret = = - EEXIST ) {
free_extent_map ( em ) ;
em = NULL ;
goto again ;
} else if ( ret ) {
em = ERR_PTR ( ret ) ;
}
out :
return em ;
2007-04-11 15:53:25 -04:00
}
2007-10-15 16:19:22 -04:00
u32 btrfs_csum_data ( struct btrfs_root * root , char * data , u32 seed , size_t len )
{
return crc32c ( seed , data , len ) ;
}
void btrfs_csum_final ( u32 crc , char * result )
{
* ( __le32 * ) result = ~ cpu_to_le32 ( crc ) ;
}
static int csum_tree_block ( struct btrfs_root * root , struct extent_buffer * buf ,
int verify )
{
char result [ BTRFS_CRC32_SIZE ] ;
unsigned long len ;
unsigned long cur_len ;
unsigned long offset = BTRFS_CSUM_SIZE ;
char * map_token = NULL ;
char * kaddr ;
unsigned long map_start ;
unsigned long map_len ;
int err ;
u32 crc = ~ ( u32 ) 0 ;
len = buf - > len - offset ;
while ( len > 0 ) {
err = map_private_extent_buffer ( buf , offset , 32 ,
& map_token , & kaddr ,
& map_start , & map_len , KM_USER0 ) ;
if ( err ) {
printk ( " failed to map extent buffer! %lu \n " ,
offset ) ;
return 1 ;
}
cur_len = min ( len , map_len - ( offset - map_start ) ) ;
crc = btrfs_csum_data ( root , kaddr + offset - map_start ,
crc , cur_len ) ;
len - = cur_len ;
offset + = cur_len ;
unmap_extent_buffer ( buf , map_token , KM_USER0 ) ;
}
btrfs_csum_final ( crc , result ) ;
if ( verify ) {
if ( memcmp_extent_buffer ( buf , result , 0 , BTRFS_CRC32_SIZE ) ) {
printk ( " btrfs: %s checksum verify failed on %llu \n " ,
root - > fs_info - > sb - > s_id ,
buf - > start ) ;
return 1 ;
}
} else {
write_extent_buffer ( buf , result , 0 , BTRFS_CRC32_SIZE ) ;
}
return 0 ;
}
int csum_dirty_buffer ( struct btrfs_root * root , struct page * page )
{
struct extent_map_tree * tree ;
u64 start = page - > index < < PAGE_CACHE_SHIFT ;
u64 found_start ;
int found_level ;
unsigned long len ;
struct extent_buffer * eb ;
tree = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
if ( page - > private = = EXTENT_PAGE_PRIVATE )
goto out ;
if ( ! page - > private )
goto out ;
len = page - > private > > 2 ;
if ( len = = 0 ) {
WARN_ON ( 1 ) ;
}
eb = alloc_extent_buffer ( tree , start , len , page , GFP_NOFS ) ;
read_extent_buffer_pages ( tree , eb , start + PAGE_CACHE_SIZE , 1 ) ;
found_start = btrfs_header_bytenr ( eb ) ;
if ( found_start ! = start ) {
printk ( " warning: eb start incorrect %Lu buffer %Lu len %lu \n " ,
start , found_start , len ) ;
}
found_level = btrfs_header_level ( eb ) ;
csum_tree_block ( root , eb , 0 ) ;
free_extent_buffer ( eb ) ;
out :
return 0 ;
}
2007-10-15 16:14:19 -04:00
static int btree_writepage ( struct page * page , struct writeback_control * wbc )
2007-03-28 13:57:48 -04:00
{
2007-10-15 16:14:19 -04:00
struct extent_map_tree * tree ;
2007-10-15 16:19:22 -04:00
struct btrfs_root * root = BTRFS_I ( page - > mapping - > host ) - > root ;
2007-10-15 16:14:19 -04:00
tree = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
2007-10-15 16:19:22 -04:00
csum_dirty_buffer ( root , page ) ;
2007-10-15 16:14:19 -04:00
return extent_write_full_page ( tree , page , btree_get_extent , wbc ) ;
}
int btree_readpage ( struct file * file , struct page * page )
{
struct extent_map_tree * tree ;
tree = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
return extent_read_full_page ( tree , page , btree_get_extent ) ;
}
2007-03-30 08:47:31 -04:00
2007-10-15 16:14:19 -04:00
static int btree_releasepage ( struct page * page , gfp_t unused_gfp_flags )
{
struct extent_map_tree * tree ;
int ret ;
2007-03-28 13:57:48 -04:00
2007-10-15 16:14:19 -04:00
tree = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
ret = try_release_extent_mapping ( tree , page ) ;
if ( ret = = 1 ) {
ClearPagePrivate ( page ) ;
set_page_private ( page , 0 ) ;
page_cache_release ( page ) ;
}
2007-03-28 13:57:48 -04:00
return ret ;
}
2007-10-15 16:14:19 -04:00
static void btree_invalidatepage ( struct page * page , unsigned long offset )
2007-03-28 13:57:48 -04:00
{
2007-10-15 16:14:19 -04:00
struct extent_map_tree * tree ;
tree = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
extent_invalidatepage ( tree , page , offset ) ;
btree_releasepage ( page , GFP_NOFS ) ;
2007-03-28 13:57:48 -04:00
}
2007-10-15 16:14:19 -04:00
#if 0
2007-03-28 13:57:48 -04:00
static int btree_writepage ( struct page * page , struct writeback_control * wbc )
2007-03-01 18:59:40 -05:00
{
2007-03-28 19:44:27 -04:00
struct buffer_head * bh ;
2007-04-09 10:42:37 -04:00
struct btrfs_root * root = BTRFS_I ( page - > mapping - > host ) - > root ;
2007-03-28 19:44:27 -04:00
struct buffer_head * head ;
if ( ! page_has_buffers ( page ) ) {
create_empty_buffers ( page , root - > fs_info - > sb - > s_blocksize ,
( 1 < < BH_Dirty ) | ( 1 < < BH_Uptodate ) ) ;
}
head = page_buffers ( page ) ;
bh = head ;
do {
if ( buffer_dirty ( bh ) )
csum_tree_block ( root , bh , 0 ) ;
bh = bh - > b_this_page ;
} while ( bh ! = head ) ;
2007-03-28 13:57:48 -04:00
return block_write_full_page ( page , btree_get_block , wbc ) ;
2007-03-01 18:59:40 -05:00
}
2007-10-15 16:14:19 -04:00
# endif
2007-02-02 09:18:22 -05:00
2007-03-28 13:57:48 -04:00
static struct address_space_operations btree_aops = {
. readpage = btree_readpage ,
. writepage = btree_writepage ,
2007-10-15 16:14:19 -04:00
. releasepage = btree_releasepage ,
. invalidatepage = btree_invalidatepage ,
2007-03-28 13:57:48 -04:00
. sync_page = block_sync_page ,
} ;
2007-10-15 16:15:53 -04:00
int readahead_tree_block ( struct btrfs_root * root , u64 bytenr , u32 blocksize )
2007-05-01 08:53:32 -04:00
{
2007-10-15 16:14:19 -04:00
struct extent_buffer * buf = NULL ;
struct inode * btree_inode = root - > fs_info - > btree_inode ;
2007-05-18 13:28:27 -04:00
int ret = 0 ;
2007-05-01 08:53:32 -04:00
2007-10-15 16:15:53 -04:00
buf = btrfs_find_create_tree_block ( root , bytenr , blocksize ) ;
2007-10-15 16:14:19 -04:00
if ( ! buf )
2007-05-01 08:53:32 -04:00
return 0 ;
2007-10-15 16:14:19 -04:00
read_extent_buffer_pages ( & BTRFS_I ( btree_inode ) - > extent_tree ,
2007-10-15 16:19:22 -04:00
buf , 0 , 0 ) ;
2007-10-15 16:14:19 -04:00
free_extent_buffer ( buf ) ;
2007-05-18 13:28:27 -04:00
return ret ;
2007-05-01 08:53:32 -04:00
}
2007-10-15 16:15:53 -04:00
struct extent_buffer * read_tree_block ( struct btrfs_root * root , u64 bytenr ,
u32 blocksize )
2007-02-02 09:18:22 -05:00
{
2007-10-15 16:14:19 -04:00
struct extent_buffer * buf = NULL ;
struct inode * btree_inode = root - > fs_info - > btree_inode ;
2007-10-15 16:19:22 -04:00
struct extent_map_tree * extent_tree ;
int ret ;
extent_tree = & BTRFS_I ( btree_inode ) - > extent_tree ;
2007-10-15 16:14:19 -04:00
2007-10-15 16:15:53 -04:00
buf = btrfs_find_create_tree_block ( root , bytenr , blocksize ) ;
2007-10-15 16:14:19 -04:00
if ( ! buf )
return NULL ;
read_extent_buffer_pages ( & BTRFS_I ( btree_inode ) - > extent_tree ,
2007-10-15 16:19:22 -04:00
buf , 0 , 1 ) ;
if ( buf - > flags & EXTENT_CSUM ) {
return buf ;
}
if ( test_range_bit ( extent_tree , buf - > start , buf - > start + buf - > len - 1 ,
EXTENT_CSUM , 1 ) ) {
buf - > flags | = EXTENT_CSUM ;
return buf ;
}
ret = csum_tree_block ( root , buf , 1 ) ;
set_extent_bits ( extent_tree , buf - > start ,
buf - > start + buf - > len - 1 ,
EXTENT_CSUM , GFP_NOFS ) ;
buf - > flags | = EXTENT_CSUM ;
2007-10-15 16:14:19 -04:00
return buf ;
2007-02-02 09:18:22 -05:00
}
2007-03-16 16:20:31 -04:00
int clean_tree_block ( struct btrfs_trans_handle * trans , struct btrfs_root * root ,
2007-10-15 16:14:19 -04:00
struct extent_buffer * buf )
2007-03-01 18:59:40 -05:00
{
2007-10-15 16:14:19 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
clear_extent_buffer_dirty ( & BTRFS_I ( btree_inode ) - > extent_tree , buf ) ;
return 0 ;
}
int wait_on_tree_block_writeback ( struct btrfs_root * root ,
struct extent_buffer * buf )
{
struct inode * btree_inode = root - > fs_info - > btree_inode ;
wait_on_extent_buffer_writeback ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf ) ;
return 0 ;
}
2007-10-15 16:15:53 -04:00
static int __setup_root ( u32 nodesize , u32 leafsize , u32 sectorsize ,
2007-03-20 14:38:32 -04:00
struct btrfs_root * root ,
struct btrfs_fs_info * fs_info ,
2007-03-22 12:13:20 -04:00
u64 objectid )
2007-02-20 16:40:44 -05:00
{
2007-02-21 17:04:57 -05:00
root - > node = NULL ;
2007-04-09 10:42:37 -04:00
root - > inode = NULL ;
2007-03-06 20:08:01 -05:00
root - > commit_root = NULL ;
2007-10-15 16:15:53 -04:00
root - > sectorsize = sectorsize ;
root - > nodesize = nodesize ;
root - > leafsize = leafsize ;
2007-03-14 14:14:43 -04:00
root - > ref_cows = 0 ;
2007-03-20 14:38:32 -04:00
root - > fs_info = fs_info ;
2007-04-09 10:42:37 -04:00
root - > objectid = objectid ;
root - > last_trans = 0 ;
2007-04-10 12:13:09 -04:00
root - > highest_inode = 0 ;
root - > last_inode_alloc = 0 ;
2007-08-29 15:47:34 -04:00
root - > name = NULL ;
2007-03-13 16:47:54 -04:00
memset ( & root - > root_key , 0 , sizeof ( root - > root_key ) ) ;
memset ( & root - > root_item , 0 , sizeof ( root - > root_item ) ) ;
2007-08-07 16:15:09 -04:00
memset ( & root - > defrag_progress , 0 , sizeof ( root - > defrag_progress ) ) ;
2007-08-29 15:47:34 -04:00
memset ( & root - > root_kobj , 0 , sizeof ( root - > root_kobj ) ) ;
init_completion ( & root - > kobj_unregister ) ;
2007-09-10 19:58:36 -04:00
init_rwsem ( & root - > snap_sem ) ;
2007-08-07 16:15:09 -04:00
root - > defrag_running = 0 ;
root - > defrag_level = 0 ;
2007-04-20 20:23:12 -04:00
root - > root_key . objectid = objectid ;
2007-03-13 16:47:54 -04:00
return 0 ;
}
2007-10-15 16:15:53 -04:00
static int find_and_setup_root ( struct btrfs_root * tree_root ,
2007-03-20 14:38:32 -04:00
struct btrfs_fs_info * fs_info ,
u64 objectid ,
2007-03-22 12:13:20 -04:00
struct btrfs_root * root )
2007-03-13 16:47:54 -04:00
{
int ret ;
2007-10-15 16:15:53 -04:00
u32 blocksize ;
2007-03-13 16:47:54 -04:00
2007-10-15 16:15:53 -04:00
__setup_root ( tree_root - > nodesize , tree_root - > leafsize ,
tree_root - > sectorsize , root , fs_info , objectid ) ;
2007-03-13 16:47:54 -04:00
ret = btrfs_find_last_root ( tree_root , objectid ,
& root - > root_item , & root - > root_key ) ;
BUG_ON ( ret ) ;
2007-10-15 16:15:53 -04:00
blocksize = btrfs_level_size ( root , btrfs_root_level ( & root - > root_item ) ) ;
root - > node = read_tree_block ( root , btrfs_root_bytenr ( & root - > root_item ) ,
blocksize ) ;
2007-03-13 16:47:54 -04:00
BUG_ON ( ! root - > node ) ;
2007-02-20 16:40:44 -05:00
return 0 ;
}
2007-06-22 14:16:25 -04:00
struct btrfs_root * btrfs_read_fs_root_no_radix ( struct btrfs_fs_info * fs_info ,
struct btrfs_key * location )
2007-04-09 10:42:37 -04:00
{
struct btrfs_root * root ;
struct btrfs_root * tree_root = fs_info - > tree_root ;
struct btrfs_path * path ;
2007-10-15 16:14:19 -04:00
struct extent_buffer * l ;
2007-04-10 12:13:09 -04:00
u64 highest_inode ;
2007-10-15 16:15:53 -04:00
u32 blocksize ;
2007-04-09 10:42:37 -04:00
int ret = 0 ;
2007-06-22 14:16:25 -04:00
root = kzalloc ( sizeof ( * root ) , GFP_NOFS ) ;
2007-06-09 09:22:25 -04:00
if ( ! root )
2007-04-09 10:42:37 -04:00
return ERR_PTR ( - ENOMEM ) ;
if ( location - > offset = = ( u64 ) - 1 ) {
2007-10-15 16:15:53 -04:00
ret = find_and_setup_root ( tree_root , fs_info ,
2007-04-09 10:42:37 -04:00
location - > objectid , root ) ;
if ( ret ) {
kfree ( root ) ;
return ERR_PTR ( ret ) ;
}
goto insert ;
}
2007-10-15 16:15:53 -04:00
__setup_root ( tree_root - > nodesize , tree_root - > leafsize ,
tree_root - > sectorsize , root , fs_info ,
2007-04-09 10:42:37 -04:00
location - > objectid ) ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
ret = btrfs_search_slot ( NULL , tree_root , location , path , 0 , 0 ) ;
if ( ret ! = 0 ) {
if ( ret > 0 )
ret = - ENOENT ;
goto out ;
}
2007-10-15 16:14:19 -04:00
l = path - > nodes [ 0 ] ;
read_extent_buffer ( l , & root - > root_item ,
btrfs_item_ptr_offset ( l , path - > slots [ 0 ] ) ,
2007-04-09 10:42:37 -04:00
sizeof ( root - > root_item ) ) ;
ret = 0 ;
out :
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
if ( ret ) {
kfree ( root ) ;
return ERR_PTR ( ret ) ;
}
2007-10-15 16:15:53 -04:00
blocksize = btrfs_level_size ( root , btrfs_root_level ( & root - > root_item ) ) ;
root - > node = read_tree_block ( root , btrfs_root_bytenr ( & root - > root_item ) ,
blocksize ) ;
2007-04-09 10:42:37 -04:00
BUG_ON ( ! root - > node ) ;
insert :
root - > ref_cows = 1 ;
2007-06-22 14:16:25 -04:00
ret = btrfs_find_highest_inode ( root , & highest_inode ) ;
if ( ret = = 0 ) {
root - > highest_inode = highest_inode ;
root - > last_inode_alloc = highest_inode ;
}
return root ;
}
struct btrfs_root * btrfs_read_fs_root ( struct btrfs_fs_info * fs_info ,
2007-08-29 15:47:34 -04:00
struct btrfs_key * location ,
const char * name , int namelen )
2007-06-22 14:16:25 -04:00
{
struct btrfs_root * root ;
int ret ;
root = radix_tree_lookup ( & fs_info - > fs_roots_radix ,
( unsigned long ) location - > objectid ) ;
if ( root )
return root ;
root = btrfs_read_fs_root_no_radix ( fs_info , location ) ;
if ( IS_ERR ( root ) )
return root ;
2007-04-10 16:58:11 -04:00
ret = radix_tree_insert ( & fs_info - > fs_roots_radix ,
( unsigned long ) root - > root_key . objectid ,
2007-04-09 10:42:37 -04:00
root ) ;
if ( ret ) {
2007-10-15 16:14:19 -04:00
free_extent_buffer ( root - > node ) ;
2007-04-09 10:42:37 -04:00
kfree ( root ) ;
return ERR_PTR ( ret ) ;
}
2007-08-29 15:47:34 -04:00
ret = btrfs_set_root_name ( root , name , namelen ) ;
if ( ret ) {
2007-10-15 16:14:19 -04:00
free_extent_buffer ( root - > node ) ;
2007-08-29 15:47:34 -04:00
kfree ( root ) ;
return ERR_PTR ( ret ) ;
}
ret = btrfs_sysfs_add_root ( root ) ;
if ( ret ) {
2007-10-15 16:14:19 -04:00
free_extent_buffer ( root - > node ) ;
2007-08-29 15:47:34 -04:00
kfree ( root - > name ) ;
kfree ( root ) ;
return ERR_PTR ( ret ) ;
}
2007-09-11 11:15:39 -04:00
ret = btrfs_find_dead_roots ( fs_info - > tree_root ,
root - > root_key . objectid , root ) ;
BUG_ON ( ret ) ;
2007-04-09 10:42:37 -04:00
return root ;
}
2007-10-15 16:19:22 -04:00
#if 0
static int add_hasher ( struct btrfs_fs_info * info , char * type ) {
struct btrfs_hasher * hasher ;
hasher = kmalloc ( sizeof ( * hasher ) , GFP_NOFS ) ;
if ( ! hasher )
return - ENOMEM ;
hasher - > hash_tfm = crypto_alloc_hash ( type , 0 , CRYPTO_ALG_ASYNC ) ;
if ( ! hasher - > hash_tfm ) {
kfree ( hasher ) ;
return - EINVAL ;
}
spin_lock ( & info - > hash_lock ) ;
list_add ( & hasher - > list , & info - > hashers ) ;
spin_unlock ( & info - > hash_lock ) ;
return 0 ;
}
# endif
2007-04-02 10:50:19 -04:00
struct btrfs_root * open_ctree ( struct super_block * sb )
2007-03-21 11:12:56 -04:00
{
2007-10-15 16:15:53 -04:00
u32 sectorsize ;
u32 nodesize ;
u32 leafsize ;
u32 blocksize ;
2007-03-22 12:13:20 -04:00
struct btrfs_root * extent_root = kmalloc ( sizeof ( struct btrfs_root ) ,
GFP_NOFS ) ;
struct btrfs_root * tree_root = kmalloc ( sizeof ( struct btrfs_root ) ,
GFP_NOFS ) ;
struct btrfs_fs_info * fs_info = kmalloc ( sizeof ( * fs_info ) ,
GFP_NOFS ) ;
2007-02-02 09:18:22 -05:00
int ret ;
2007-06-12 06:35:45 -04:00
int err = - EIO ;
2007-04-02 10:50:19 -04:00
struct btrfs_super_block * disk_super ;
2007-02-02 09:18:22 -05:00
2007-06-12 06:35:45 -04:00
if ( ! extent_root | | ! tree_root | | ! fs_info ) {
err = - ENOMEM ;
goto fail ;
}
2007-04-09 10:42:37 -04:00
INIT_RADIX_TREE ( & fs_info - > fs_roots_radix , GFP_NOFS ) ;
2007-04-19 21:01:03 -04:00
INIT_LIST_HEAD ( & fs_info - > trans_list ) ;
2007-06-08 18:11:48 -04:00
INIT_LIST_HEAD ( & fs_info - > dead_roots ) ;
2007-10-15 16:19:22 -04:00
INIT_LIST_HEAD ( & fs_info - > hashers ) ;
spin_lock_init ( & fs_info - > hash_lock ) ;
2007-08-29 15:47:34 -04:00
memset ( & fs_info - > super_kobj , 0 , sizeof ( fs_info - > super_kobj ) ) ;
init_completion ( & fs_info - > kobj_unregister ) ;
2007-04-02 10:50:19 -04:00
sb_set_blocksize ( sb , 4096 ) ;
2007-03-20 14:38:32 -04:00
fs_info - > running_transaction = NULL ;
2007-08-10 16:22:09 -04:00
fs_info - > last_trans_committed = 0 ;
2007-03-20 14:38:32 -04:00
fs_info - > tree_root = tree_root ;
fs_info - > extent_root = extent_root ;
2007-03-22 12:13:20 -04:00
fs_info - > sb = sb ;
2007-03-28 13:57:48 -04:00
fs_info - > btree_inode = new_inode ( sb ) ;
fs_info - > btree_inode - > i_ino = 1 ;
2007-04-02 10:50:19 -04:00
fs_info - > btree_inode - > i_nlink = 1 ;
2007-03-28 13:57:48 -04:00
fs_info - > btree_inode - > i_size = sb - > s_bdev - > bd_inode - > i_size ;
fs_info - > btree_inode - > i_mapping - > a_ops = & btree_aops ;
2007-10-15 16:14:19 -04:00
extent_map_tree_init ( & BTRFS_I ( fs_info - > btree_inode ) - > extent_tree ,
fs_info - > btree_inode - > i_mapping ,
GFP_NOFS ) ;
2007-10-15 16:14:48 -04:00
extent_map_tree_init ( & fs_info - > free_space_cache ,
fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
2007-10-15 16:15:19 -04:00
extent_map_tree_init ( & fs_info - > block_group_cache ,
fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
2007-10-15 16:15:26 -04:00
extent_map_tree_init ( & fs_info - > pinned_extents ,
fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
extent_map_tree_init ( & fs_info - > pending_del ,
fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
extent_map_tree_init ( & fs_info - > extent_ins ,
fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
2007-04-20 13:16:02 -04:00
fs_info - > do_barriers = 1 ;
2007-06-08 18:11:48 -04:00
fs_info - > closing = 0 ;
2007-06-08 15:33:54 -04:00
INIT_DELAYED_WORK ( & fs_info - > trans_work , btrfs_transaction_cleaner ) ;
2007-04-09 10:42:37 -04:00
BTRFS_I ( fs_info - > btree_inode ) - > root = tree_root ;
memset ( & BTRFS_I ( fs_info - > btree_inode ) - > location , 0 ,
sizeof ( struct btrfs_key ) ) ;
2007-03-30 08:47:31 -04:00
insert_inode_hash ( fs_info - > btree_inode ) ;
2007-03-28 13:57:48 -04:00
mapping_set_gfp_mask ( fs_info - > btree_inode - > i_mapping , GFP_NOFS ) ;
2007-06-12 06:35:45 -04:00
2007-03-22 15:59:16 -04:00
mutex_init ( & fs_info - > trans_mutex ) ;
2007-03-23 19:47:49 -04:00
mutex_init ( & fs_info - > fs_mutex ) ;
2007-03-13 16:47:54 -04:00
2007-10-15 16:19:22 -04:00
#if 0
ret = add_hasher ( fs_info , " crc32c " ) ;
if ( ret ) {
printk ( " btrfs: failed hash setup, modprobe cryptomgr? \n " ) ;
err = - ENOMEM ;
goto fail_iput ;
}
# endif
2007-10-15 16:15:53 -04:00
__setup_root ( 512 , 512 , 512 , tree_root ,
2007-04-02 10:50:19 -04:00
fs_info , BTRFS_ROOT_TREE_OBJECTID ) ;
2007-04-11 15:53:25 -04:00
2007-04-02 10:50:19 -04:00
fs_info - > sb_buffer = read_tree_block ( tree_root ,
2007-10-15 16:15:53 -04:00
BTRFS_SUPER_INFO_OFFSET ,
512 ) ;
2007-03-28 13:57:48 -04:00
2007-04-09 10:42:37 -04:00
if ( ! fs_info - > sb_buffer )
2007-06-12 06:35:45 -04:00
goto fail_iput ;
2007-10-15 16:14:19 -04:00
read_extent_buffer ( fs_info - > sb_buffer , & fs_info - > super_copy , 0 ,
sizeof ( fs_info - > super_copy ) ) ;
read_extent_buffer ( fs_info - > sb_buffer , fs_info - > fsid ,
( unsigned long ) btrfs_super_fsid ( fs_info - > sb_buffer ) ,
BTRFS_FSID_SIZE ) ;
disk_super = & fs_info - > super_copy ;
2007-04-09 10:42:37 -04:00
if ( ! btrfs_super_root ( disk_super ) )
2007-06-12 06:35:45 -04:00
goto fail_sb_buffer ;
2007-04-09 10:42:37 -04:00
2007-10-15 16:15:53 -04:00
nodesize = btrfs_super_nodesize ( disk_super ) ;
leafsize = btrfs_super_leafsize ( disk_super ) ;
sectorsize = btrfs_super_sectorsize ( disk_super ) ;
tree_root - > nodesize = nodesize ;
tree_root - > leafsize = leafsize ;
tree_root - > sectorsize = sectorsize ;
2007-04-12 10:43:05 -04:00
i_size_write ( fs_info - > btree_inode ,
2007-10-15 16:15:53 -04:00
btrfs_super_total_bytes ( disk_super ) ) ;
2007-04-12 10:43:05 -04:00
2007-06-12 06:35:45 -04:00
if ( strncmp ( ( char * ) ( & disk_super - > magic ) , BTRFS_MAGIC ,
sizeof ( disk_super - > magic ) ) ) {
printk ( " btrfs: valid FS not found on %s \n " , sb - > s_id ) ;
goto fail_sb_buffer ;
}
2007-10-15 16:19:22 -04:00
2007-10-15 16:15:53 -04:00
blocksize = btrfs_level_size ( tree_root ,
btrfs_super_root_level ( disk_super ) ) ;
2007-10-15 16:19:22 -04:00
2007-03-22 12:13:20 -04:00
tree_root - > node = read_tree_block ( tree_root ,
2007-10-15 16:15:53 -04:00
btrfs_super_root ( disk_super ) ,
blocksize ) ;
2007-06-12 06:35:45 -04:00
if ( ! tree_root - > node )
goto fail_sb_buffer ;
2007-03-13 16:47:54 -04:00
2007-04-02 10:50:19 -04:00
mutex_lock ( & fs_info - > fs_mutex ) ;
2007-10-15 16:15:53 -04:00
ret = find_and_setup_root ( tree_root , fs_info ,
2007-03-22 12:13:20 -04:00
BTRFS_EXTENT_TREE_OBJECTID , extent_root ) ;
2007-06-12 06:35:45 -04:00
if ( ret ) {
mutex_unlock ( & fs_info - > fs_mutex ) ;
goto fail_tree_root ;
}
2007-03-13 16:47:54 -04:00
2007-04-26 16:46:15 -04:00
btrfs_read_block_groups ( extent_root ) ;
2007-04-09 10:42:37 -04:00
fs_info - > generation = btrfs_super_generation ( disk_super ) + 1 ;
2007-04-05 13:35:25 -04:00
mutex_unlock ( & fs_info - > fs_mutex ) ;
2007-04-09 10:42:37 -04:00
return tree_root ;
2007-06-12 06:35:45 -04:00
fail_tree_root :
2007-10-15 16:14:19 -04:00
free_extent_buffer ( tree_root - > node ) ;
2007-06-12 06:35:45 -04:00
fail_sb_buffer :
2007-10-15 16:14:19 -04:00
free_extent_buffer ( fs_info - > sb_buffer ) ;
2007-06-12 06:35:45 -04:00
fail_iput :
iput ( fs_info - > btree_inode ) ;
fail :
kfree ( extent_root ) ;
kfree ( tree_root ) ;
kfree ( fs_info ) ;
return ERR_PTR ( err ) ;
2007-02-02 09:18:22 -05:00
}
2007-03-16 16:20:31 -04:00
int write_ctree_super ( struct btrfs_trans_handle * trans , struct btrfs_root
2007-03-22 15:59:16 -04:00
* root )
2007-02-02 09:18:22 -05:00
{
2007-04-20 13:16:02 -04:00
int ret ;
2007-10-15 16:14:19 -04:00
struct extent_buffer * super = root - > fs_info - > sb_buffer ;
struct inode * btree_inode = root - > fs_info - > btree_inode ;
set_extent_buffer_dirty ( & BTRFS_I ( btree_inode ) - > extent_tree , super ) ;
ret = sync_page_range_nolock ( btree_inode , btree_inode - > i_mapping ,
super - > start , super - > len ) ;
return ret ;
2007-02-21 17:04:57 -05:00
}
2007-06-22 14:16:25 -04:00
int btrfs_free_fs_root ( struct btrfs_fs_info * fs_info , struct btrfs_root * root )
2007-04-10 16:58:11 -04:00
{
radix_tree_delete ( & fs_info - > fs_roots_radix ,
( unsigned long ) root - > root_key . objectid ) ;
2007-08-29 15:47:34 -04:00
btrfs_sysfs_del_root ( root ) ;
2007-04-10 16:58:11 -04:00
if ( root - > inode )
iput ( root - > inode ) ;
if ( root - > node )
2007-10-15 16:14:19 -04:00
free_extent_buffer ( root - > node ) ;
2007-04-10 16:58:11 -04:00
if ( root - > commit_root )
2007-10-15 16:14:19 -04:00
free_extent_buffer ( root - > commit_root ) ;
2007-08-29 15:47:34 -04:00
if ( root - > name )
kfree ( root - > name ) ;
2007-04-10 16:58:11 -04:00
kfree ( root ) ;
return 0 ;
}
2007-05-02 15:53:43 -04:00
static int del_fs_roots ( struct btrfs_fs_info * fs_info )
2007-04-09 10:42:37 -04:00
{
int ret ;
struct btrfs_root * gang [ 8 ] ;
int i ;
while ( 1 ) {
ret = radix_tree_gang_lookup ( & fs_info - > fs_roots_radix ,
( void * * ) gang , 0 ,
ARRAY_SIZE ( gang ) ) ;
if ( ! ret )
break ;
2007-04-10 16:58:11 -04:00
for ( i = 0 ; i < ret ; i + + )
2007-06-22 14:16:25 -04:00
btrfs_free_fs_root ( fs_info , gang [ i ] ) ;
2007-04-09 10:42:37 -04:00
}
return 0 ;
}
2007-04-12 12:14:00 -04:00
2007-03-22 12:13:20 -04:00
int close_ctree ( struct btrfs_root * root )
2007-02-21 17:04:57 -05:00
{
2007-03-13 16:47:54 -04:00
int ret ;
2007-03-16 16:20:31 -04:00
struct btrfs_trans_handle * trans ;
2007-04-09 10:42:37 -04:00
struct btrfs_fs_info * fs_info = root - > fs_info ;
2007-03-16 16:20:31 -04:00
2007-06-08 18:11:48 -04:00
fs_info - > closing = 1 ;
2007-06-08 15:33:54 -04:00
btrfs_transaction_flush_work ( root ) ;
2007-04-09 10:42:37 -04:00
mutex_lock ( & fs_info - > fs_mutex ) ;
2007-08-07 16:15:09 -04:00
btrfs_defrag_dirty_roots ( root - > fs_info ) ;
2007-03-22 15:59:16 -04:00
trans = btrfs_start_transaction ( root , 1 ) ;
2007-06-22 14:16:25 -04:00
ret = btrfs_commit_transaction ( trans , root ) ;
2007-03-22 15:59:16 -04:00
/* run commit again to drop the original snapshot */
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_commit_transaction ( trans , root ) ;
ret = btrfs_write_and_wait_transaction ( NULL , root ) ;
2007-03-13 16:47:54 -04:00
BUG_ON ( ret ) ;
2007-03-22 15:59:16 -04:00
write_ctree_super ( NULL , root ) ;
2007-04-09 10:42:37 -04:00
mutex_unlock ( & fs_info - > fs_mutex ) ;
if ( fs_info - > extent_root - > node )
2007-10-15 16:14:19 -04:00
free_extent_buffer ( fs_info - > extent_root - > node ) ;
2007-10-15 16:14:48 -04:00
2007-04-09 10:42:37 -04:00
if ( fs_info - > tree_root - > node )
2007-10-15 16:14:19 -04:00
free_extent_buffer ( fs_info - > tree_root - > node ) ;
2007-10-15 16:14:48 -04:00
2007-10-15 16:14:19 -04:00
free_extent_buffer ( fs_info - > sb_buffer ) ;
2007-04-11 15:53:25 -04:00
2007-04-26 16:46:15 -04:00
btrfs_free_block_groups ( root - > fs_info ) ;
2007-04-09 10:42:37 -04:00
del_fs_roots ( fs_info ) ;
2007-10-15 16:19:22 -04:00
extent_map_tree_empty_lru ( & BTRFS_I ( fs_info - > btree_inode ) - > extent_tree ) ;
2007-10-15 16:15:53 -04:00
truncate_inode_pages ( fs_info - > btree_inode - > i_mapping , 0 ) ;
iput ( fs_info - > btree_inode ) ;
2007-10-15 16:19:22 -04:00
#if 0
while ( ! list_empty ( & fs_info - > hashers ) ) {
struct btrfs_hasher * hasher ;
hasher = list_entry ( fs_info - > hashers . next , struct btrfs_hasher ,
hashers ) ;
list_del ( & hasher - > hashers ) ;
crypto_free_hash ( & fs_info - > hash_tfm ) ;
kfree ( hasher ) ;
}
# endif
2007-04-09 10:42:37 -04:00
kfree ( fs_info - > extent_root ) ;
kfree ( fs_info - > tree_root ) ;
2007-02-02 09:18:22 -05:00
return 0 ;
}
2007-10-15 16:14:19 -04:00
int btrfs_buffer_uptodate ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct inode * btree_inode = buf - > first_page - > mapping - > host ;
2007-10-15 16:14:19 -04:00
return extent_buffer_uptodate ( & BTRFS_I ( btree_inode ) - > extent_tree , buf ) ;
}
int btrfs_set_buffer_uptodate ( struct extent_buffer * buf )
2007-06-28 15:57:36 -04:00
{
2007-10-15 16:18:55 -04:00
struct inode * btree_inode = buf - > first_page - > mapping - > host ;
2007-10-15 16:14:19 -04:00
return set_extent_buffer_uptodate ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf ) ;
}
2007-08-07 16:15:09 -04:00
2007-10-15 16:14:19 -04:00
void btrfs_mark_buffer_dirty ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:14:19 -04:00
u64 transid = btrfs_header_generation ( buf ) ;
struct inode * btree_inode = root - > fs_info - > btree_inode ;
2007-08-07 16:15:09 -04:00
2007-06-28 15:57:36 -04:00
if ( transid ! = root - > fs_info - > generation ) {
printk ( KERN_CRIT " transid mismatch buffer %llu, found %Lu running %Lu \n " ,
2007-10-15 16:15:53 -04:00
( unsigned long long ) buf - > start ,
2007-06-28 15:57:36 -04:00
transid , root - > fs_info - > generation ) ;
WARN_ON ( 1 ) ;
}
2007-10-15 16:14:19 -04:00
set_extent_buffer_dirty ( & BTRFS_I ( btree_inode ) - > extent_tree , buf ) ;
2007-02-02 09:18:22 -05:00
}
2007-09-17 10:58:06 -04:00
void btrfs_btree_balance_dirty ( struct btrfs_root * root , unsigned long nr )
2007-05-02 15:53:43 -04:00
{
2007-09-17 10:58:06 -04:00
balance_dirty_pages_ratelimited_nr (
2007-10-15 16:21:17 -04:00
root - > fs_info - > btree_inode - > i_mapping , 1 ) ;
2007-05-02 15:53:43 -04:00
}
2007-10-15 16:17:34 -04:00
void btrfs_set_buffer_defrag ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
set_extent_bits ( & BTRFS_I ( btree_inode ) - > extent_tree , buf - > start ,
buf - > start + buf - > len - 1 , EXTENT_DEFRAG , GFP_NOFS ) ;
}
void btrfs_set_buffer_defrag_done ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
set_extent_bits ( & BTRFS_I ( btree_inode ) - > extent_tree , buf - > start ,
buf - > start + buf - > len - 1 , EXTENT_DEFRAG_DONE ,
GFP_NOFS ) ;
}
int btrfs_buffer_defrag ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
return test_range_bit ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf - > start , buf - > start + buf - > len - 1 , EXTENT_DEFRAG , 0 ) ;
}
int btrfs_buffer_defrag_done ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
return test_range_bit ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf - > start , buf - > start + buf - > len - 1 ,
EXTENT_DEFRAG_DONE , 0 ) ;
}
int btrfs_clear_buffer_defrag_done ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
return clear_extent_bits ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf - > start , buf - > start + buf - > len - 1 ,
EXTENT_DEFRAG_DONE , GFP_NOFS ) ;
}
int btrfs_clear_buffer_defrag ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
return clear_extent_bits ( & BTRFS_I ( btree_inode ) - > extent_tree ,
buf - > start , buf - > start + buf - > len - 1 ,
EXTENT_DEFRAG , GFP_NOFS ) ;
}
int btrfs_read_buffer ( struct extent_buffer * buf )
{
2007-10-15 16:18:55 -04:00
struct btrfs_root * root = BTRFS_I ( buf - > first_page - > mapping - > host ) - > root ;
2007-10-15 16:17:34 -04:00
struct inode * btree_inode = root - > fs_info - > btree_inode ;
return read_extent_buffer_pages ( & BTRFS_I ( btree_inode ) - > extent_tree ,
2007-10-15 16:19:22 -04:00
buf , 0 , 1 ) ;
2007-10-15 16:17:34 -04:00
}