2007-06-12 17: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 .
*/
2008-04-26 00:53:30 +04:00
# include <linux/kernel.h>
2008-02-20 20:07:25 +03:00
# include <linux/bio.h>
2007-06-12 14:35:45 +04:00
# include <linux/buffer_head.h>
2008-05-02 22:43:14 +04:00
# include <linux/file.h>
2007-06-12 14:35:45 +04:00
# include <linux/fs.h>
# include <linux/pagemap.h>
# include <linux/highmem.h>
# include <linux/time.h>
# include <linux/init.h>
# include <linux/string.h>
# include <linux/smp_lock.h>
# include <linux/backing-dev.h>
# include <linux/mpage.h>
# include <linux/swap.h>
# include <linux/writeback.h>
# include <linux/statfs.h>
# include <linux/compat.h>
2007-06-15 21:50:00 +04:00
# include <linux/bit_spinlock.h>
2007-07-25 20:31:35 +04:00
# include <linux/version.h>
2007-11-16 19:45:54 +03:00
# include <linux/xattr.h>
2007-06-12 14:35:45 +04:00
# include "ctree.h"
# include "disk-io.h"
# include "transaction.h"
# include "btrfs_inode.h"
# include "ioctl.h"
# include "print-tree.h"
2008-03-24 22:01:56 +03:00
# include "volumes.h"
2007-06-12 14:35:45 +04:00
struct btrfs_iget_args {
u64 ino ;
struct btrfs_root * root ;
} ;
static struct inode_operations btrfs_dir_inode_operations ;
static struct inode_operations btrfs_symlink_inode_operations ;
static struct inode_operations btrfs_dir_ro_inode_operations ;
2007-07-11 18:18:17 +04:00
static struct inode_operations btrfs_special_inode_operations ;
2007-06-12 14:35:45 +04:00
static struct inode_operations btrfs_file_inode_operations ;
static struct address_space_operations btrfs_aops ;
static struct address_space_operations btrfs_symlink_aops ;
static struct file_operations btrfs_dir_file_operations ;
2008-01-25 00:13:08 +03:00
static struct extent_io_ops btrfs_extent_io_ops ;
2007-06-12 14:35:45 +04:00
static struct kmem_cache * btrfs_inode_cachep ;
struct kmem_cache * btrfs_trans_handle_cachep ;
struct kmem_cache * btrfs_transaction_cachep ;
struct kmem_cache * btrfs_bit_radix_cachep ;
struct kmem_cache * btrfs_path_cachep ;
# define S_SHIFT 12
static unsigned char btrfs_type_by_mode [ S_IFMT > > S_SHIFT ] = {
[ S_IFREG > > S_SHIFT ] = BTRFS_FT_REG_FILE ,
[ S_IFDIR > > S_SHIFT ] = BTRFS_FT_DIR ,
[ S_IFCHR > > S_SHIFT ] = BTRFS_FT_CHRDEV ,
[ S_IFBLK > > S_SHIFT ] = BTRFS_FT_BLKDEV ,
[ S_IFIFO > > S_SHIFT ] = BTRFS_FT_FIFO ,
[ S_IFSOCK > > S_SHIFT ] = BTRFS_FT_SOCK ,
[ S_IFLNK > > S_SHIFT ] = BTRFS_FT_SYMLINK ,
} ;
2007-12-22 00:27:21 +03:00
int btrfs_check_free_space ( struct btrfs_root * root , u64 num_required ,
int for_del )
{
u64 total = btrfs_super_total_bytes ( & root - > fs_info - > super_copy ) ;
u64 used = btrfs_super_bytes_used ( & root - > fs_info - > super_copy ) ;
u64 thresh ;
2008-04-22 21:26:47 +04:00
unsigned long flags ;
2007-12-22 00:27:21 +03:00
int ret = 0 ;
if ( for_del )
2008-01-03 17:22:38 +03:00
thresh = total * 90 ;
2007-12-22 00:27:21 +03:00
else
2008-01-03 17:22:38 +03:00
thresh = total * 85 ;
do_div ( thresh , 100 ) ;
2007-12-22 00:27:21 +03:00
2008-04-22 21:26:47 +04:00
spin_lock_irqsave ( & root - > fs_info - > delalloc_lock , flags ) ;
2007-12-22 00:27:21 +03:00
if ( used + root - > fs_info - > delalloc_bytes + num_required > thresh )
ret = - ENOSPC ;
2008-04-22 21:26:47 +04:00
spin_unlock_irqrestore ( & root - > fs_info - > delalloc_lock , flags ) ;
2007-12-22 00:27:21 +03:00
return ret ;
}
2007-12-18 04:14:01 +03:00
static int cow_file_range ( struct inode * inode , u64 start , u64 end )
2007-08-28 00:49:44 +04:00
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
u64 alloc_hint = 0 ;
2007-10-16 00:15:53 +04:00
u64 num_bytes ;
2007-12-18 04:14:04 +03:00
u64 cur_alloc_size ;
2007-10-16 00:15:53 +04:00
u64 blocksize = root - > sectorsize ;
2008-01-25 00:13:08 +03:00
u64 orig_start = start ;
u64 orig_num_bytes ;
2007-12-18 04:14:01 +03:00
struct btrfs_key ins ;
int ret ;
2007-08-28 00:49:44 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
BUG_ON ( ! trans ) ;
2007-12-18 04:14:01 +03:00
btrfs_set_trans_block_group ( trans , inode ) ;
2007-10-16 00:15:53 +04:00
num_bytes = ( end - start + blocksize ) & ~ ( blocksize - 1 ) ;
2007-12-18 04:14:01 +03:00
num_bytes = max ( blocksize , num_bytes ) ;
2007-08-28 00:49:44 +04:00
ret = btrfs_drop_extents ( trans , root , inode ,
2007-10-16 00:18:25 +04:00
start , start + num_bytes , start , & alloc_hint ) ;
2008-01-25 00:13:08 +03:00
orig_num_bytes = num_bytes ;
2007-10-16 00:15:53 +04:00
2007-11-01 18:28:41 +03:00
if ( alloc_hint = = EXTENT_MAP_INLINE )
goto out ;
2008-04-17 19:29:12 +04:00
BUG_ON ( num_bytes > btrfs_super_total_bytes ( & root - > fs_info - > super_copy ) ) ;
2007-12-18 04:14:04 +03:00
while ( num_bytes > 0 ) {
cur_alloc_size = min ( num_bytes , root - > fs_info - > max_extent ) ;
ret = btrfs_alloc_extent ( trans , root , cur_alloc_size ,
2008-04-14 17:46:10 +04:00
root - > sectorsize ,
2007-12-18 04:14:04 +03:00
root - > root_key . objectid ,
trans - > transid ,
inode - > i_ino , start , 0 ,
alloc_hint , ( u64 ) - 1 , & ins , 1 ) ;
if ( ret ) {
WARN_ON ( 1 ) ;
goto out ;
}
2008-04-14 17:46:10 +04:00
cur_alloc_size = ins . offset ;
2007-12-18 04:14:04 +03:00
ret = btrfs_insert_file_extent ( trans , root , inode - > i_ino ,
start , ins . objectid , ins . offset ,
2008-05-02 22:43:14 +04:00
ins . offset , 0 ) ;
2008-02-08 21:49:28 +03:00
inode - > i_blocks + = ins . offset > > 9 ;
2008-01-23 00:47:59 +03:00
btrfs_check_file ( root , inode ) ;
2008-04-17 19:29:12 +04:00
if ( num_bytes < cur_alloc_size ) {
printk ( " num_bytes %Lu cur_alloc %Lu \n " , num_bytes ,
cur_alloc_size ) ;
break ;
}
2007-12-18 04:14:04 +03:00
num_bytes - = cur_alloc_size ;
alloc_hint = ins . objectid + ins . offset ;
start + = cur_alloc_size ;
2007-08-28 00:49:44 +04:00
}
2008-01-25 00:13:08 +03:00
btrfs_drop_extent_cache ( inode , orig_start ,
orig_start + orig_num_bytes - 1 ) ;
2008-01-08 23:46:30 +03:00
btrfs_add_ordered_inode ( inode ) ;
2008-02-08 21:49:28 +03:00
btrfs_update_inode ( trans , root , inode ) ;
2007-08-28 00:49:44 +04:00
out :
btrfs_end_transaction ( trans , root ) ;
2007-12-18 04:14:01 +03:00
return ret ;
}
static int run_delalloc_nocow ( struct inode * inode , u64 start , u64 end )
{
u64 extent_start ;
u64 extent_end ;
u64 bytenr ;
u64 cow_end ;
2007-12-22 00:27:21 +03:00
u64 loops = 0 ;
2008-01-08 23:46:31 +03:00
u64 total_fs_bytes ;
2007-12-18 04:14:01 +03:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-05-08 22:11:56 +04:00
struct btrfs_block_group_cache * block_group ;
2007-12-18 04:14:01 +03:00
struct extent_buffer * leaf ;
int found_type ;
struct btrfs_path * path ;
struct btrfs_file_extent_item * item ;
int ret ;
int err ;
struct btrfs_key found_key ;
2008-01-08 23:46:31 +03:00
total_fs_bytes = btrfs_super_total_bytes ( & root - > fs_info - > super_copy ) ;
2007-12-18 04:14:01 +03:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
again :
ret = btrfs_lookup_file_extent ( NULL , root , path ,
inode - > i_ino , start , 0 ) ;
if ( ret < 0 ) {
btrfs_free_path ( path ) ;
return ret ;
}
cow_end = end ;
if ( ret ! = 0 ) {
if ( path - > slots [ 0 ] = = 0 )
goto not_found ;
path - > slots [ 0 ] - - ;
}
leaf = path - > nodes [ 0 ] ;
item = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
struct btrfs_file_extent_item ) ;
/* are we inside the extent that was found? */
btrfs_item_key_to_cpu ( leaf , & found_key , path - > slots [ 0 ] ) ;
found_type = btrfs_key_type ( & found_key ) ;
if ( found_key . objectid ! = inode - > i_ino | |
2008-05-09 00:31:21 +04:00
found_type ! = BTRFS_EXTENT_DATA_KEY )
2007-12-18 04:14:01 +03:00
goto not_found ;
found_type = btrfs_file_extent_type ( leaf , item ) ;
extent_start = found_key . offset ;
if ( found_type = = BTRFS_FILE_EXTENT_REG ) {
2008-01-08 23:46:31 +03:00
u64 extent_num_bytes ;
extent_num_bytes = btrfs_file_extent_num_bytes ( leaf , item ) ;
extent_end = extent_start + extent_num_bytes ;
2007-12-18 04:14:01 +03:00
err = 0 ;
2007-12-22 00:27:21 +03:00
if ( loops & & start ! = extent_start )
goto not_found ;
2007-12-18 04:14:01 +03:00
if ( start < extent_start | | start > = extent_end )
goto not_found ;
cow_end = min ( end , extent_end - 1 ) ;
bytenr = btrfs_file_extent_disk_bytenr ( leaf , item ) ;
if ( bytenr = = 0 )
goto not_found ;
2008-05-08 22:11:56 +04:00
if ( btrfs_count_snapshots_in_path ( root , path , inode - > i_ino ,
bytenr ) ! = 1 ) {
goto not_found ;
}
2008-01-08 23:46:31 +03:00
/*
* we may be called by the resizer , make sure we ' re inside
* the limits of the FS
*/
2008-05-08 22:11:56 +04:00
block_group = btrfs_lookup_block_group ( root - > fs_info ,
bytenr ) ;
if ( ! block_group | | block_group - > ro )
2008-01-08 23:46:31 +03:00
goto not_found ;
2007-12-18 04:14:01 +03:00
start = extent_end ;
2008-01-03 21:23:19 +03:00
} else {
2007-12-18 04:14:01 +03:00
goto not_found ;
}
loop :
if ( start > end ) {
btrfs_free_path ( path ) ;
return 0 ;
}
btrfs_release_path ( root , path ) ;
2007-12-22 00:27:21 +03:00
loops + + ;
2007-12-18 04:14:01 +03:00
goto again ;
not_found :
2008-05-09 00:31:21 +04:00
cow_file_range ( inode , start , end ) ;
start = end + 1 ;
2007-12-18 04:14:01 +03:00
goto loop ;
}
static int run_delalloc_range ( struct inode * inode , u64 start , u64 end )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
int ret ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2008-01-08 23:54:37 +03:00
if ( btrfs_test_opt ( root , NODATACOW ) | |
btrfs_test_flag ( inode , NODATACOW ) )
2007-12-18 04:14:01 +03:00
ret = run_delalloc_nocow ( inode , start , end ) ;
else
ret = cow_file_range ( inode , start , end ) ;
2007-12-22 00:27:21 +03:00
2007-08-28 00:49:44 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
2008-01-29 23:55:23 +03:00
int btrfs_set_bit_hook ( struct inode * inode , u64 start , u64 end ,
2008-01-31 19:05:37 +03:00
unsigned long old , unsigned long bits )
2008-01-29 23:55:23 +03:00
{
2008-04-22 21:26:47 +04:00
unsigned long flags ;
2008-01-31 19:05:37 +03:00
if ( ! ( old & EXTENT_DELALLOC ) & & ( bits & EXTENT_DELALLOC ) ) {
2008-01-29 23:55:23 +03:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-04-22 21:26:47 +04:00
spin_lock_irqsave ( & root - > fs_info - > delalloc_lock , flags ) ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes + = end - start + 1 ;
2008-01-29 23:55:23 +03:00
root - > fs_info - > delalloc_bytes + = end - start + 1 ;
2008-04-22 21:26:47 +04:00
spin_unlock_irqrestore ( & root - > fs_info - > delalloc_lock , flags ) ;
2008-01-29 23:55:23 +03:00
}
return 0 ;
}
int btrfs_clear_bit_hook ( struct inode * inode , u64 start , u64 end ,
2008-01-31 19:05:37 +03:00
unsigned long old , unsigned long bits )
2008-01-29 23:55:23 +03:00
{
2008-01-31 19:05:37 +03:00
if ( ( old & EXTENT_DELALLOC ) & & ( bits & EXTENT_DELALLOC ) ) {
2008-01-29 23:55:23 +03:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-04-22 21:26:47 +04:00
unsigned long flags ;
spin_lock_irqsave ( & root - > fs_info - > delalloc_lock , flags ) ;
2008-01-31 19:05:37 +03:00
if ( end - start + 1 > root - > fs_info - > delalloc_bytes ) {
printk ( " warning: delalloc account %Lu %Lu \n " ,
end - start + 1 , root - > fs_info - > delalloc_bytes ) ;
root - > fs_info - > delalloc_bytes = 0 ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes = 0 ;
2008-01-31 19:05:37 +03:00
} else {
root - > fs_info - > delalloc_bytes - = end - start + 1 ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes - = end - start + 1 ;
2008-01-31 19:05:37 +03:00
}
2008-04-22 21:26:47 +04:00
spin_unlock_irqrestore ( & root - > fs_info - > delalloc_lock , flags ) ;
2008-01-29 23:55:23 +03:00
}
return 0 ;
}
2008-03-24 22:02:07 +03:00
int btrfs_merge_bio_hook ( struct page * page , unsigned long offset ,
size_t size , struct bio * bio )
{
struct btrfs_root * root = BTRFS_I ( page - > mapping - > host ) - > root ;
struct btrfs_mapping_tree * map_tree ;
u64 logical = bio - > bi_sector < < 9 ;
u64 length = 0 ;
u64 map_length ;
int ret ;
2008-04-21 18:03:05 +04:00
length = bio - > bi_size ;
2008-03-24 22:02:07 +03:00
map_tree = & root - > fs_info - > mapping_tree ;
map_length = length ;
2008-04-10 00:28:12 +04:00
ret = btrfs_map_block ( map_tree , READ , logical ,
2008-04-10 00:28:12 +04:00
& map_length , NULL , 0 ) ;
2008-04-10 00:28:12 +04:00
2008-03-24 22:02:07 +03:00
if ( map_length < length + size ) {
return 1 ;
}
return 0 ;
}
2008-04-16 19:14:51 +04:00
int __btrfs_submit_bio_hook ( struct inode * inode , int rw , struct bio * bio ,
2008-04-10 00:28:12 +04:00
int mirror_num )
2008-02-20 20:07:25 +03:00
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
int ret = 0 ;
2008-04-16 19:15:20 +04:00
char * sums = NULL ;
ret = btrfs_csum_one_bio ( root , bio , & sums ) ;
BUG_ON ( ret ) ;
2008-02-20 20:07:25 +03:00
2008-04-16 19:14:51 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
2008-04-16 19:15:20 +04:00
2008-04-16 19:14:51 +04:00
btrfs_set_trans_block_group ( trans , inode ) ;
2008-04-16 19:15:20 +04:00
btrfs_csum_file_blocks ( trans , root , inode , bio , sums ) ;
2008-04-16 19:14:51 +04:00
ret = btrfs_end_transaction ( trans , root ) ;
BUG_ON ( ret ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2008-04-16 19:15:20 +04:00
kfree ( sums ) ;
2008-06-12 00:50:36 +04:00
return btrfs_map_bio ( root , rw , bio , mirror_num , 1 ) ;
2008-04-16 19:14:51 +04:00
}
int btrfs_submit_bio_hook ( struct inode * inode , int rw , struct bio * bio ,
int mirror_num )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
int ret = 0 ;
2008-04-10 00:28:12 +04:00
if ( ! ( rw & ( 1 < < BIO_RW ) ) ) {
ret = btrfs_bio_wq_end_io ( root - > fs_info , bio , 0 ) ;
BUG_ON ( ret ) ;
2008-03-24 22:01:56 +03:00
goto mapit ;
}
2008-02-20 20:07:25 +03:00
if ( btrfs_test_opt ( root , NODATASUM ) | |
2008-03-24 22:01:56 +03:00
btrfs_test_flag ( inode , NODATASUM ) ) {
goto mapit ;
}
2008-02-20 20:07:25 +03:00
2008-04-16 19:14:51 +04:00
return btrfs_wq_submit_bio ( BTRFS_I ( inode ) - > root - > fs_info ,
inode , rw , bio , mirror_num ,
__btrfs_submit_bio_hook ) ;
2008-03-24 22:01:56 +03:00
mapit :
2008-06-12 00:50:36 +04:00
return btrfs_map_bio ( root , rw , bio , mirror_num , 0 ) ;
2008-02-20 20:07:25 +03:00
}
2008-02-21 00:11:05 +03:00
2007-08-30 16:50:51 +04:00
int btrfs_readpage_io_hook ( struct page * page , u64 start , u64 end )
{
int ret = 0 ;
struct inode * inode = page - > mapping - > host ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-08-30 16:50:51 +04:00
struct btrfs_csum_item * item ;
struct btrfs_path * path = NULL ;
2007-10-16 00:22:25 +04:00
u32 csum ;
2008-04-16 20:59:22 +04:00
2008-01-08 23:54:37 +03:00
if ( btrfs_test_opt ( root , NODATASUM ) | |
btrfs_test_flag ( inode , NODATASUM ) )
2007-12-14 23:30:32 +03:00
return 0 ;
2008-04-16 20:59:22 +04:00
2007-08-30 16:50:51 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
path = btrfs_alloc_path ( ) ;
item = btrfs_lookup_csum ( NULL , root , path , inode - > i_ino , start , 0 ) ;
if ( IS_ERR ( item ) ) {
ret = PTR_ERR ( item ) ;
/* a csum that isn't present is a preallocated region. */
if ( ret = = - ENOENT | | ret = = - EFBIG )
ret = 0 ;
2007-10-16 00:22:25 +04:00
csum = 0 ;
2008-01-29 17:10:27 +03:00
printk ( " no csum found for inode %lu start %Lu \n " , inode - > i_ino , start ) ;
2007-08-30 16:50:51 +04:00
goto out ;
}
2007-10-16 00:22:25 +04:00
read_extent_buffer ( path - > nodes [ 0 ] , & csum , ( unsigned long ) item ,
BTRFS_CRC32_SIZE ) ;
2008-01-25 00:13:08 +03:00
set_state_private ( io_tree , start , csum ) ;
2007-08-30 16:50:51 +04:00
out :
if ( path )
btrfs_free_path ( path ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
2008-04-10 00:28:12 +04:00
struct io_failure_record {
struct page * page ;
u64 start ;
u64 len ;
u64 logical ;
int last_mirror ;
} ;
2008-05-12 21:39:03 +04:00
int btrfs_io_failed_hook ( struct bio * failed_bio ,
struct page * page , u64 start , u64 end ,
struct extent_state * state )
2008-04-10 00:28:12 +04:00
{
struct io_failure_record * failrec = NULL ;
u64 private ;
struct extent_map * em ;
struct inode * inode = page - > mapping - > host ;
struct extent_io_tree * failure_tree = & BTRFS_I ( inode ) - > io_failure_tree ;
2008-04-17 19:29:12 +04:00
struct extent_map_tree * em_tree = & BTRFS_I ( inode ) - > extent_tree ;
2008-04-10 00:28:12 +04:00
struct bio * bio ;
int num_copies ;
int ret ;
2008-05-12 21:39:03 +04:00
int rw ;
2008-04-10 00:28:12 +04:00
u64 logical ;
ret = get_state_private ( failure_tree , start , & private ) ;
if ( ret ) {
failrec = kmalloc ( sizeof ( * failrec ) , GFP_NOFS ) ;
if ( ! failrec )
return - ENOMEM ;
failrec - > start = start ;
failrec - > len = end - start + 1 ;
failrec - > last_mirror = 0 ;
2008-04-17 19:29:12 +04:00
spin_lock ( & em_tree - > lock ) ;
em = lookup_extent_mapping ( em_tree , start , failrec - > len ) ;
if ( em - > start > start | | em - > start + em - > len < start ) {
free_extent_map ( em ) ;
em = NULL ;
}
spin_unlock ( & em_tree - > lock ) ;
2008-04-10 00:28:12 +04:00
if ( ! em | | IS_ERR ( em ) ) {
kfree ( failrec ) ;
return - EIO ;
}
logical = start - em - > start ;
logical = em - > block_start + logical ;
failrec - > logical = logical ;
free_extent_map ( em ) ;
set_extent_bits ( failure_tree , start , end , EXTENT_LOCKED |
EXTENT_DIRTY , GFP_NOFS ) ;
2008-04-11 20:16:46 +04:00
set_state_private ( failure_tree , start ,
( u64 ) ( unsigned long ) failrec ) ;
2008-04-10 00:28:12 +04:00
} else {
2008-04-11 20:16:46 +04:00
failrec = ( struct io_failure_record * ) ( unsigned long ) private ;
2008-04-10 00:28:12 +04:00
}
num_copies = btrfs_num_copies (
& BTRFS_I ( inode ) - > root - > fs_info - > mapping_tree ,
failrec - > logical , failrec - > len ) ;
failrec - > last_mirror + + ;
if ( ! state ) {
spin_lock_irq ( & BTRFS_I ( inode ) - > io_tree . lock ) ;
state = find_first_extent_bit_state ( & BTRFS_I ( inode ) - > io_tree ,
failrec - > start ,
EXTENT_LOCKED ) ;
if ( state & & state - > start ! = failrec - > start )
state = NULL ;
spin_unlock_irq ( & BTRFS_I ( inode ) - > io_tree . lock ) ;
}
if ( ! state | | failrec - > last_mirror > num_copies ) {
set_state_private ( failure_tree , failrec - > start , 0 ) ;
clear_extent_bits ( failure_tree , failrec - > start ,
failrec - > start + failrec - > len - 1 ,
EXTENT_LOCKED | EXTENT_DIRTY , GFP_NOFS ) ;
kfree ( failrec ) ;
return - EIO ;
}
bio = bio_alloc ( GFP_NOFS , 1 ) ;
bio - > bi_private = state ;
bio - > bi_end_io = failed_bio - > bi_end_io ;
bio - > bi_sector = failrec - > logical > > 9 ;
bio - > bi_bdev = failed_bio - > bi_bdev ;
2008-04-22 21:26:46 +04:00
bio - > bi_size = 0 ;
2008-04-10 00:28:12 +04:00
bio_add_page ( bio , page , failrec - > len , start - page_offset ( page ) ) ;
2008-05-12 21:39:03 +04:00
if ( failed_bio - > bi_rw & ( 1 < < BIO_RW ) )
rw = WRITE ;
else
rw = READ ;
BTRFS_I ( inode ) - > io_tree . ops - > submit_bio_hook ( inode , rw , bio ,
failrec - > last_mirror ) ;
return 0 ;
}
int btrfs_clean_io_failures ( struct inode * inode , u64 start )
{
u64 private ;
u64 private_failure ;
struct io_failure_record * failure ;
int ret ;
private = 0 ;
if ( count_range_bits ( & BTRFS_I ( inode ) - > io_failure_tree , & private ,
( u64 ) - 1 , 1 , EXTENT_DIRTY ) ) {
ret = get_state_private ( & BTRFS_I ( inode ) - > io_failure_tree ,
start , & private_failure ) ;
if ( ret = = 0 ) {
failure = ( struct io_failure_record * ) ( unsigned long )
private_failure ;
set_state_private ( & BTRFS_I ( inode ) - > io_failure_tree ,
failure - > start , 0 ) ;
clear_extent_bits ( & BTRFS_I ( inode ) - > io_failure_tree ,
failure - > start ,
failure - > start + failure - > len - 1 ,
EXTENT_DIRTY | EXTENT_LOCKED ,
GFP_NOFS ) ;
kfree ( failure ) ;
}
}
2008-04-10 00:28:12 +04:00
return 0 ;
}
2008-01-29 17:59:12 +03:00
int btrfs_readpage_end_io_hook ( struct page * page , u64 start , u64 end ,
struct extent_state * state )
2007-08-30 16:50:51 +04:00
{
2007-10-30 23:56:53 +03:00
size_t offset = start - ( ( u64 ) page - > index < < PAGE_CACHE_SHIFT ) ;
2007-08-30 16:50:51 +04:00
struct inode * inode = page - > mapping - > host ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-08-30 16:50:51 +04:00
char * kaddr ;
2008-01-29 17:10:27 +03:00
u64 private = ~ ( u32 ) 0 ;
2007-08-30 16:50:51 +04:00
int ret ;
2007-10-16 00:22:25 +04:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
u32 csum = ~ ( u32 ) 0 ;
2007-10-19 17:23:07 +04:00
unsigned long flags ;
2008-01-25 00:13:08 +03:00
2008-01-08 23:54:37 +03:00
if ( btrfs_test_opt ( root , NODATASUM ) | |
btrfs_test_flag ( inode , NODATASUM ) )
2007-12-14 23:30:32 +03:00
return 0 ;
2008-02-04 16:57:25 +03:00
if ( state & & state - > start = = start ) {
2008-01-29 17:59:12 +03:00
private = state - > private ;
ret = 0 ;
} else {
ret = get_state_private ( io_tree , start , & private ) ;
}
2007-10-19 17:23:07 +04:00
local_irq_save ( flags ) ;
2007-08-30 16:50:51 +04:00
kaddr = kmap_atomic ( page , KM_IRQ0 ) ;
if ( ret ) {
goto zeroit ;
}
2007-10-16 00:22:25 +04:00
csum = btrfs_csum_data ( root , kaddr + offset , csum , end - start + 1 ) ;
btrfs_csum_final ( csum , ( char * ) & csum ) ;
if ( csum ! = private ) {
2007-08-30 16:50:51 +04:00
goto zeroit ;
}
kunmap_atomic ( kaddr , KM_IRQ0 ) ;
2007-10-19 17:23:07 +04:00
local_irq_restore ( flags ) ;
2008-04-10 00:28:12 +04:00
/* if the io failure tree for this inode is non-empty,
* check to see if we ' ve recovered from a failed IO
*/
2008-05-12 21:39:03 +04:00
btrfs_clean_io_failures ( inode , start ) ;
2007-08-30 16:50:51 +04:00
return 0 ;
zeroit :
2008-01-29 17:10:27 +03:00
printk ( " btrfs csum failed ino %lu off %llu csum %u private %Lu \n " ,
page - > mapping - > host - > i_ino , ( unsigned long long ) start , csum ,
private ) ;
2007-10-16 00:15:53 +04:00
memset ( kaddr + offset , 1 , end - start + 1 ) ;
flush_dcache_page ( page ) ;
2007-08-30 16:50:51 +04:00
kunmap_atomic ( kaddr , KM_IRQ0 ) ;
2007-10-19 17:23:07 +04:00
local_irq_restore ( flags ) ;
2008-04-17 19:29:12 +04:00
if ( private = = 0 )
return 0 ;
2008-04-10 00:28:12 +04:00
return - EIO ;
2007-08-30 16:50:51 +04:00
}
2007-08-28 00:49:44 +04:00
2007-06-12 14:35:45 +04:00
void btrfs_read_locked_inode ( struct inode * inode )
{
struct btrfs_path * path ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-06-12 14:35:45 +04:00
struct btrfs_inode_item * inode_item ;
2008-03-24 22:01:56 +03:00
struct btrfs_timespec * tspec ;
2007-06-12 14:35:45 +04:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_key location ;
u64 alloc_group_block ;
2007-07-11 18:18:17 +04:00
u32 rdev ;
2007-06-12 14:35:45 +04:00
int ret ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
memcpy ( & location , & BTRFS_I ( inode ) - > location , sizeof ( location ) ) ;
2008-01-08 23:46:30 +03:00
2007-06-12 14:35:45 +04:00
ret = btrfs_lookup_inode ( NULL , root , path , & location , 0 ) ;
2007-10-16 00:14:19 +04:00
if ( ret )
2007-06-12 14:35:45 +04:00
goto make_bad ;
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
inode_item = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
struct btrfs_inode_item ) ;
inode - > i_mode = btrfs_inode_mode ( leaf , inode_item ) ;
inode - > i_nlink = btrfs_inode_nlink ( leaf , inode_item ) ;
inode - > i_uid = btrfs_inode_uid ( leaf , inode_item ) ;
inode - > i_gid = btrfs_inode_gid ( leaf , inode_item ) ;
inode - > i_size = btrfs_inode_size ( leaf , inode_item ) ;
tspec = btrfs_inode_atime ( inode_item ) ;
inode - > i_atime . tv_sec = btrfs_timespec_sec ( leaf , tspec ) ;
inode - > i_atime . tv_nsec = btrfs_timespec_nsec ( leaf , tspec ) ;
tspec = btrfs_inode_mtime ( inode_item ) ;
inode - > i_mtime . tv_sec = btrfs_timespec_sec ( leaf , tspec ) ;
inode - > i_mtime . tv_nsec = btrfs_timespec_nsec ( leaf , tspec ) ;
tspec = btrfs_inode_ctime ( inode_item ) ;
inode - > i_ctime . tv_sec = btrfs_timespec_sec ( leaf , tspec ) ;
inode - > i_ctime . tv_nsec = btrfs_timespec_nsec ( leaf , tspec ) ;
inode - > i_blocks = btrfs_inode_nblocks ( leaf , inode_item ) ;
inode - > i_generation = btrfs_inode_generation ( leaf , inode_item ) ;
2007-07-11 18:18:17 +04:00
inode - > i_rdev = 0 ;
2007-10-16 00:14:19 +04:00
rdev = btrfs_inode_rdev ( leaf , inode_item ) ;
alloc_group_block = btrfs_inode_block_group ( leaf , inode_item ) ;
2007-06-12 14:35:45 +04:00
BTRFS_I ( inode ) - > block_group = btrfs_lookup_block_group ( root - > fs_info ,
alloc_group_block ) ;
2008-01-08 23:54:37 +03:00
BTRFS_I ( inode ) - > flags = btrfs_inode_flags ( leaf , inode_item ) ;
2007-12-22 00:36:24 +03:00
if ( ! BTRFS_I ( inode ) - > block_group ) {
BTRFS_I ( inode ) - > block_group = btrfs_find_block_group ( root ,
2008-03-24 22:01:56 +03:00
NULL , 0 ,
BTRFS_BLOCK_GROUP_METADATA , 0 ) ;
2007-12-22 00:36:24 +03:00
}
2007-06-12 14:35:45 +04:00
btrfs_free_path ( path ) ;
inode_item = NULL ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
switch ( inode - > i_mode & S_IFMT ) {
case S_IFREG :
inode - > i_mapping - > a_ops = & btrfs_aops ;
2008-03-26 17:28:07 +03:00
inode - > i_mapping - > backing_dev_info = & root - > fs_info - > bdi ;
2008-01-25 00:13:08 +03:00
BTRFS_I ( inode ) - > io_tree . ops = & btrfs_extent_io_ops ;
2007-06-12 14:35:45 +04:00
inode - > i_fop = & btrfs_file_operations ;
inode - > i_op = & btrfs_file_inode_operations ;
break ;
case S_IFDIR :
inode - > i_fop = & btrfs_dir_file_operations ;
if ( root = = root - > fs_info - > tree_root )
inode - > i_op = & btrfs_dir_ro_inode_operations ;
else
inode - > i_op = & btrfs_dir_inode_operations ;
break ;
case S_IFLNK :
inode - > i_op = & btrfs_symlink_inode_operations ;
inode - > i_mapping - > a_ops = & btrfs_symlink_aops ;
2008-03-26 17:28:07 +03:00
inode - > i_mapping - > backing_dev_info = & root - > fs_info - > bdi ;
2007-06-12 14:35:45 +04:00
break ;
2007-07-11 18:18:17 +04:00
default :
init_special_inode ( inode , inode - > i_mode , rdev ) ;
break ;
2007-06-12 14:35:45 +04:00
}
return ;
make_bad :
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
make_bad_inode ( inode ) ;
}
2007-10-16 00:14:19 +04:00
static void fill_inode_item ( struct extent_buffer * leaf ,
struct btrfs_inode_item * item ,
2007-06-12 14:35:45 +04:00
struct inode * inode )
{
2007-10-16 00:14:19 +04:00
btrfs_set_inode_uid ( leaf , item , inode - > i_uid ) ;
btrfs_set_inode_gid ( leaf , item , inode - > i_gid ) ;
btrfs_set_inode_size ( leaf , item , inode - > i_size ) ;
btrfs_set_inode_mode ( leaf , item , inode - > i_mode ) ;
btrfs_set_inode_nlink ( leaf , item , inode - > i_nlink ) ;
btrfs_set_timespec_sec ( leaf , btrfs_inode_atime ( item ) ,
inode - > i_atime . tv_sec ) ;
btrfs_set_timespec_nsec ( leaf , btrfs_inode_atime ( item ) ,
inode - > i_atime . tv_nsec ) ;
btrfs_set_timespec_sec ( leaf , btrfs_inode_mtime ( item ) ,
inode - > i_mtime . tv_sec ) ;
btrfs_set_timespec_nsec ( leaf , btrfs_inode_mtime ( item ) ,
inode - > i_mtime . tv_nsec ) ;
btrfs_set_timespec_sec ( leaf , btrfs_inode_ctime ( item ) ,
inode - > i_ctime . tv_sec ) ;
btrfs_set_timespec_nsec ( leaf , btrfs_inode_ctime ( item ) ,
inode - > i_ctime . tv_nsec ) ;
btrfs_set_inode_nblocks ( leaf , item , inode - > i_blocks ) ;
btrfs_set_inode_generation ( leaf , item , inode - > i_generation ) ;
btrfs_set_inode_rdev ( leaf , item , inode - > i_rdev ) ;
2008-01-08 23:54:37 +03:00
btrfs_set_inode_flags ( leaf , item , BTRFS_I ( inode ) - > flags ) ;
2007-10-16 00:14:19 +04:00
btrfs_set_inode_block_group ( leaf , item ,
2007-06-12 14:35:45 +04:00
BTRFS_I ( inode ) - > block_group - > key . objectid ) ;
}
2007-08-28 00:49:44 +04:00
int btrfs_update_inode ( struct btrfs_trans_handle * trans ,
2007-06-12 14:35:45 +04:00
struct btrfs_root * root ,
struct inode * inode )
{
struct btrfs_inode_item * inode_item ;
struct btrfs_path * path ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-06-12 14:35:45 +04:00
int ret ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
ret = btrfs_lookup_inode ( trans , root , path ,
& BTRFS_I ( inode ) - > location , 1 ) ;
if ( ret ) {
if ( ret > 0 )
ret = - ENOENT ;
goto failed ;
}
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
inode_item = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
2007-06-12 14:35:45 +04:00
struct btrfs_inode_item ) ;
2007-10-16 00:14:19 +04:00
fill_inode_item ( leaf , inode_item , inode ) ;
btrfs_mark_buffer_dirty ( leaf ) ;
2007-08-11 00:22:09 +04:00
btrfs_set_inode_last_trans ( trans , inode ) ;
2007-06-12 14:35:45 +04:00
ret = 0 ;
failed :
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
return ret ;
}
static int btrfs_unlink_trans ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
struct inode * dir ,
struct dentry * dentry )
{
struct btrfs_path * path ;
const char * name = dentry - > d_name . name ;
int name_len = dentry - > d_name . len ;
int ret = 0 ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-06-12 14:35:45 +04:00
struct btrfs_dir_item * di ;
2007-10-16 00:14:19 +04:00
struct btrfs_key key ;
2007-06-12 14:35:45 +04:00
path = btrfs_alloc_path ( ) ;
2007-06-22 22:16:25 +04:00
if ( ! path ) {
ret = - ENOMEM ;
goto err ;
}
2007-06-12 14:35:45 +04:00
di = btrfs_lookup_dir_item ( trans , root , path , dir - > i_ino ,
name , name_len , - 1 ) ;
if ( IS_ERR ( di ) ) {
ret = PTR_ERR ( di ) ;
goto err ;
}
if ( ! di ) {
ret = - ENOENT ;
goto err ;
}
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
btrfs_dir_item_key_to_cpu ( leaf , di , & key ) ;
2007-06-12 14:35:45 +04:00
ret = btrfs_delete_one_dir_name ( trans , root , path , di ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto err ;
2007-06-12 14:35:45 +04:00
btrfs_release_path ( root , path ) ;
di = btrfs_lookup_dir_index_item ( trans , root , path , dir - > i_ino ,
2007-10-16 00:14:19 +04:00
key . objectid , name , name_len , - 1 ) ;
2007-06-12 14:35:45 +04:00
if ( IS_ERR ( di ) ) {
ret = PTR_ERR ( di ) ;
goto err ;
}
if ( ! di ) {
ret = - ENOENT ;
goto err ;
}
ret = btrfs_delete_one_dir_name ( trans , root , path , di ) ;
dentry - > d_inode - > i_ctime = dir - > i_ctime ;
2007-12-13 17:06:01 +03:00
ret = btrfs_del_inode_ref ( trans , root , name , name_len ,
dentry - > d_inode - > i_ino ,
dentry - > d_parent - > d_inode - > i_ino ) ;
if ( ret ) {
printk ( " failed to delete reference to %.*s, "
" inode %lu parent %lu \n " , name_len , name ,
dentry - > d_inode - > i_ino ,
dentry - > d_parent - > d_inode - > i_ino ) ;
2007-12-12 22:38:19 +03:00
}
2007-06-12 14:35:45 +04:00
err :
btrfs_free_path ( path ) ;
if ( ! ret ) {
dir - > i_size - = name_len * 2 ;
2007-06-25 18:09:33 +04:00
dir - > i_mtime = dir - > i_ctime = CURRENT_TIME ;
2007-06-12 14:35:45 +04:00
btrfs_update_inode ( trans , root , dir ) ;
2007-12-19 00:15:09 +03:00
# if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,18)
dentry - > d_inode - > i_nlink - - ;
# else
2007-06-12 14:35:45 +04:00
drop_nlink ( dentry - > d_inode ) ;
2007-12-19 00:15:09 +03:00
# endif
2007-06-22 22:16:25 +04:00
ret = btrfs_update_inode ( trans , root , dentry - > d_inode ) ;
2007-06-12 14:35:45 +04:00
dir - > i_sb - > s_dirt = 1 ;
}
return ret ;
}
static int btrfs_unlink ( struct inode * dir , struct dentry * dentry )
{
struct btrfs_root * root ;
struct btrfs_trans_handle * trans ;
2008-01-16 19:44:43 +03:00
struct inode * inode = dentry - > d_inode ;
2007-06-12 14:35:45 +04:00
int ret ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
root = BTRFS_I ( dir ) - > root ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
ret = btrfs_check_free_space ( root , 1 , 1 ) ;
if ( ret )
goto fail ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
btrfs_set_trans_block_group ( trans , dir ) ;
ret = btrfs_unlink_trans ( trans , root , dir , dentry ) ;
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-10-16 00:14:19 +04:00
2008-01-16 19:44:43 +03:00
if ( inode - > i_nlink = = 0 ) {
/* if the inode isn't linked anywhere,
* we don ' t need to worry about
* data = ordered
*/
2008-05-27 18:55:43 +04:00
btrfs_del_ordered_inode ( inode ) ;
2008-01-16 19:44:43 +03:00
}
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return ret ;
}
static int btrfs_rmdir ( struct inode * dir , struct dentry * dentry )
{
struct inode * inode = dentry - > d_inode ;
2007-12-22 00:27:21 +03:00
int err = 0 ;
2007-06-12 14:35:45 +04:00
int ret ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
struct btrfs_trans_handle * trans ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
2007-10-25 23:49:25 +04:00
if ( inode - > i_size > BTRFS_EMPTY_DIR_SIZE )
return - ENOTEMPTY ;
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
ret = btrfs_check_free_space ( root , 1 , 1 ) ;
if ( ret )
goto fail ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , dir ) ;
/* now the directory is empty */
err = btrfs_unlink_trans ( trans , root , dir , dentry ) ;
if ( ! err ) {
inode - > i_size = 0 ;
}
2007-12-12 22:38:19 +03:00
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-12 14:35:45 +04:00
ret = btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-10-25 23:49:25 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-12-12 22:38:19 +03:00
2007-06-12 14:35:45 +04:00
if ( ret & & ! err )
err = ret ;
return err ;
}
/*
* this can truncate away extent items , csum items and directory items .
* It starts at a high offset and removes keys until it can ' t find
* any higher than i_size .
*
* csum items that cross the new i_size are truncated to the new size
* as well .
*/
static int btrfs_truncate_in_trans ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
2008-01-29 23:11:36 +03:00
struct inode * inode ,
u32 min_type )
2007-06-12 14:35:45 +04:00
{
int ret ;
struct btrfs_path * path ;
struct btrfs_key key ;
2007-10-16 00:14:19 +04:00
struct btrfs_key found_key ;
2007-06-12 14:35:45 +04:00
u32 found_type ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-06-12 14:35:45 +04:00
struct btrfs_file_extent_item * fi ;
u64 extent_start = 0 ;
2007-10-16 00:15:53 +04:00
u64 extent_num_bytes = 0 ;
2007-06-12 14:35:45 +04:00
u64 item_end = 0 ;
2007-12-11 17:25:06 +03:00
u64 root_gen = 0 ;
2007-12-11 20:42:00 +03:00
u64 root_owner = 0 ;
2007-06-12 14:35:45 +04:00
int found_extent ;
int del_item ;
2008-01-29 23:11:36 +03:00
int pending_del_nr = 0 ;
int pending_del_slot = 0 ;
2007-11-01 18:28:41 +03:00
int extent_type = - 1 ;
2008-04-17 19:29:12 +04:00
u64 mask = root - > sectorsize - 1 ;
2007-06-12 14:35:45 +04:00
2008-04-17 19:29:12 +04:00
btrfs_drop_extent_cache ( inode , inode - > i_size & ( ~ mask ) , ( u64 ) - 1 ) ;
2007-06-12 14:35:45 +04:00
path = btrfs_alloc_path ( ) ;
2007-08-07 23:52:22 +04:00
path - > reada = - 1 ;
2007-06-12 14:35:45 +04:00
BUG_ON ( ! path ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
/* FIXME, add redo link to tree so we don't leak on crash */
key . objectid = inode - > i_ino ;
key . offset = ( u64 ) - 1 ;
2007-10-16 00:14:19 +04:00
key . type = ( u8 ) - 1 ;
2008-01-29 23:11:36 +03:00
btrfs_init_path ( path ) ;
search_again :
ret = btrfs_search_slot ( trans , root , & key , path , - 1 , 1 ) ;
if ( ret < 0 ) {
goto error ;
}
if ( ret > 0 ) {
BUG_ON ( path - > slots [ 0 ] = = 0 ) ;
path - > slots [ 0 ] - - ;
}
2007-06-12 14:35:45 +04:00
while ( 1 ) {
fi = NULL ;
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
btrfs_item_key_to_cpu ( leaf , & found_key , path - > slots [ 0 ] ) ;
found_type = btrfs_key_type ( & found_key ) ;
2007-06-12 14:35:45 +04:00
2007-10-16 00:14:19 +04:00
if ( found_key . objectid ! = inode - > i_ino )
2007-06-12 14:35:45 +04:00
break ;
2007-10-16 00:14:19 +04:00
2008-01-29 23:11:36 +03:00
if ( found_type < min_type )
2007-06-12 14:35:45 +04:00
break ;
2007-10-16 00:14:19 +04:00
item_end = found_key . offset ;
2007-06-12 14:35:45 +04:00
if ( found_type = = BTRFS_EXTENT_DATA_KEY ) {
2007-10-16 00:14:19 +04:00
fi = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
2007-06-12 14:35:45 +04:00
struct btrfs_file_extent_item ) ;
2007-11-01 18:28:41 +03:00
extent_type = btrfs_file_extent_type ( leaf , fi ) ;
if ( extent_type ! = BTRFS_FILE_EXTENT_INLINE ) {
2007-10-16 00:14:19 +04:00
item_end + =
2007-10-16 00:15:53 +04:00
btrfs_file_extent_num_bytes ( leaf , fi ) ;
2007-11-01 18:28:41 +03:00
} else if ( extent_type = = BTRFS_FILE_EXTENT_INLINE ) {
struct btrfs_item * item = btrfs_item_nr ( leaf ,
path - > slots [ 0 ] ) ;
item_end + = btrfs_file_extent_inline_len ( leaf ,
item ) ;
2007-06-12 14:35:45 +04:00
}
2007-11-07 21:31:09 +03:00
item_end - - ;
2007-06-12 14:35:45 +04:00
}
if ( found_type = = BTRFS_CSUM_ITEM_KEY ) {
ret = btrfs_csum_truncate ( trans , root , path ,
inode - > i_size ) ;
BUG_ON ( ret ) ;
}
2007-11-07 21:31:09 +03:00
if ( item_end < inode - > i_size ) {
2007-08-28 00:49:44 +04:00
if ( found_type = = BTRFS_DIR_ITEM_KEY ) {
found_type = BTRFS_INODE_ITEM_KEY ;
} else if ( found_type = = BTRFS_EXTENT_ITEM_KEY ) {
found_type = BTRFS_CSUM_ITEM_KEY ;
2008-01-29 23:11:36 +03:00
} else if ( found_type = = BTRFS_EXTENT_DATA_KEY ) {
found_type = BTRFS_XATTR_ITEM_KEY ;
} else if ( found_type = = BTRFS_XATTR_ITEM_KEY ) {
found_type = BTRFS_INODE_REF_KEY ;
2007-08-28 00:49:44 +04:00
} else if ( found_type ) {
found_type - - ;
} else {
break ;
2007-06-12 14:35:45 +04:00
}
2007-09-17 19:08:38 +04:00
btrfs_set_key_type ( & key , found_type ) ;
2008-01-29 23:11:36 +03:00
goto next ;
2007-06-12 14:35:45 +04:00
}
2007-10-16 00:14:19 +04:00
if ( found_key . offset > = inode - > i_size )
2007-06-12 14:35:45 +04:00
del_item = 1 ;
else
del_item = 0 ;
found_extent = 0 ;
/* FIXME, shrink the extent if the ref count is only 1 */
2007-11-01 18:28:41 +03:00
if ( found_type ! = BTRFS_EXTENT_DATA_KEY )
goto delete ;
if ( extent_type ! = BTRFS_FILE_EXTENT_INLINE ) {
2007-06-12 14:35:45 +04:00
u64 num_dec ;
2007-10-16 00:15:53 +04:00
extent_start = btrfs_file_extent_disk_bytenr ( leaf , fi ) ;
2007-06-12 14:35:45 +04:00
if ( ! del_item ) {
2007-10-16 00:15:53 +04:00
u64 orig_num_bytes =
btrfs_file_extent_num_bytes ( leaf , fi ) ;
extent_num_bytes = inode - > i_size -
2007-10-16 00:14:19 +04:00
found_key . offset + root - > sectorsize - 1 ;
2008-01-30 19:54:04 +03:00
extent_num_bytes = extent_num_bytes &
~ ( ( u64 ) root - > sectorsize - 1 ) ;
2007-10-16 00:15:53 +04:00
btrfs_set_file_extent_num_bytes ( leaf , fi ,
extent_num_bytes ) ;
num_dec = ( orig_num_bytes -
2008-02-08 21:49:28 +03:00
extent_num_bytes ) ;
if ( extent_start ! = 0 )
dec_i_blocks ( inode , num_dec ) ;
2007-10-16 00:14:19 +04:00
btrfs_mark_buffer_dirty ( leaf ) ;
2007-06-12 14:35:45 +04:00
} else {
2007-10-16 00:15:53 +04:00
extent_num_bytes =
btrfs_file_extent_disk_num_bytes ( leaf ,
fi ) ;
2007-06-12 14:35:45 +04:00
/* FIXME blocksize != 4096 */
2008-02-08 21:49:28 +03:00
num_dec = btrfs_file_extent_num_bytes ( leaf , fi ) ;
2007-06-12 14:35:45 +04:00
if ( extent_start ! = 0 ) {
found_extent = 1 ;
2008-02-08 21:49:28 +03:00
dec_i_blocks ( inode , num_dec ) ;
2007-06-12 14:35:45 +04:00
}
2007-12-11 20:42:00 +03:00
root_gen = btrfs_header_generation ( leaf ) ;
root_owner = btrfs_header_owner ( leaf ) ;
2007-06-12 14:35:45 +04:00
}
2008-02-08 21:49:28 +03:00
} else if ( extent_type = = BTRFS_FILE_EXTENT_INLINE ) {
if ( ! del_item ) {
u32 newsize = inode - > i_size - found_key . offset ;
dec_i_blocks ( inode , item_end + 1 -
found_key . offset - newsize ) ;
newsize =
btrfs_file_extent_calc_inline_size ( newsize ) ;
ret = btrfs_truncate_item ( trans , root , path ,
newsize , 1 ) ;
BUG_ON ( ret ) ;
} else {
dec_i_blocks ( inode , item_end + 1 -
found_key . offset ) ;
}
2007-06-12 14:35:45 +04:00
}
2007-11-01 18:28:41 +03:00
delete :
2007-06-12 14:35:45 +04:00
if ( del_item ) {
2008-01-29 23:11:36 +03:00
if ( ! pending_del_nr ) {
/* no pending yet, add ourselves */
pending_del_slot = path - > slots [ 0 ] ;
pending_del_nr = 1 ;
} else if ( pending_del_nr & &
path - > slots [ 0 ] + 1 = = pending_del_slot ) {
/* hop on the pending chunk */
pending_del_nr + + ;
pending_del_slot = path - > slots [ 0 ] ;
} else {
printk ( " bad pending slot %d pending_del_nr %d pending_del_slot %d \n " , path - > slots [ 0 ] , pending_del_nr , pending_del_slot ) ;
}
2007-06-12 14:35:45 +04:00
} else {
break ;
}
if ( found_extent ) {
ret = btrfs_free_extent ( trans , root , extent_start ,
2007-12-11 17:25:06 +03:00
extent_num_bytes ,
2007-12-11 20:42:00 +03:00
root_owner ,
2007-12-11 17:25:06 +03:00
root_gen , inode - > i_ino ,
found_key . offset , 0 ) ;
2007-06-12 14:35:45 +04:00
BUG_ON ( ret ) ;
}
2008-01-29 23:11:36 +03:00
next :
if ( path - > slots [ 0 ] = = 0 ) {
if ( pending_del_nr )
goto del_pending ;
btrfs_release_path ( root , path ) ;
goto search_again ;
}
path - > slots [ 0 ] - - ;
if ( pending_del_nr & &
path - > slots [ 0 ] + 1 ! = pending_del_slot ) {
struct btrfs_key debug ;
del_pending :
btrfs_item_key_to_cpu ( path - > nodes [ 0 ] , & debug ,
pending_del_slot ) ;
ret = btrfs_del_items ( trans , root , path ,
pending_del_slot ,
pending_del_nr ) ;
BUG_ON ( ret ) ;
pending_del_nr = 0 ;
btrfs_release_path ( root , path ) ;
goto search_again ;
}
2007-06-12 14:35:45 +04:00
}
ret = 0 ;
error :
2008-01-29 23:11:36 +03:00
if ( pending_del_nr ) {
ret = btrfs_del_items ( trans , root , path , pending_del_slot ,
pending_del_nr ) ;
}
2007-06-12 14:35:45 +04:00
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
inode - > i_sb - > s_dirt = 1 ;
return ret ;
}
2007-08-28 00:49:44 +04:00
static int btrfs_cow_one_page ( struct inode * inode , struct page * page ,
2007-08-28 00:49:44 +04:00
size_t zero_start )
{
char * kaddr ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-10-30 23:56:53 +03:00
u64 page_start = ( u64 ) page - > index < < PAGE_CACHE_SHIFT ;
2007-08-28 00:49:44 +04:00
u64 page_end = page_start + PAGE_CACHE_SIZE - 1 ;
2007-12-22 00:27:21 +03:00
int ret = 0 ;
2007-08-28 00:49:44 +04:00
2007-12-19 00:25:45 +03:00
WARN_ON ( ! PageLocked ( page ) ) ;
2007-09-17 19:25:58 +04:00
set_page_extent_mapped ( page ) ;
2007-08-28 00:49:44 +04:00
2008-01-25 00:13:08 +03:00
lock_extent ( io_tree , page_start , page_end , GFP_NOFS ) ;
set_extent_delalloc ( & BTRFS_I ( inode ) - > io_tree , page_start ,
2007-08-28 00:49:44 +04:00
page_end , GFP_NOFS ) ;
2007-12-22 00:27:21 +03:00
2007-08-28 00:49:44 +04:00
if ( zero_start ! = PAGE_CACHE_SIZE ) {
2007-08-28 00:49:44 +04:00
kaddr = kmap ( page ) ;
2007-08-28 00:49:44 +04:00
memset ( kaddr + zero_start , 0 , PAGE_CACHE_SIZE - zero_start ) ;
flush_dcache_page ( page ) ;
2007-08-28 00:49:44 +04:00
kunmap ( page ) ;
2007-08-28 00:49:44 +04:00
}
2007-08-28 00:49:44 +04:00
set_page_dirty ( page ) ;
2008-01-25 00:13:08 +03:00
unlock_extent ( io_tree , page_start , page_end , GFP_NOFS ) ;
2007-08-28 00:49:44 +04:00
return ret ;
}
2007-06-12 14:35:45 +04:00
/*
* taken from block_truncate_page , but does cow as it zeros out
* any bytes left in the last page in the file .
*/
static int btrfs_truncate_page ( struct address_space * mapping , loff_t from )
{
struct inode * inode = mapping - > host ;
2007-10-16 00:15:53 +04:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
u32 blocksize = root - > sectorsize ;
2007-06-12 14:35:45 +04:00
pgoff_t index = from > > PAGE_CACHE_SHIFT ;
unsigned offset = from & ( PAGE_CACHE_SIZE - 1 ) ;
struct page * page ;
int ret = 0 ;
2007-08-28 00:49:44 +04:00
u64 page_start ;
2007-06-12 14:35:45 +04:00
if ( ( offset & ( blocksize - 1 ) ) = = 0 )
goto out ;
ret = - ENOMEM ;
2008-05-15 17:13:45 +04:00
again :
2007-06-12 14:35:45 +04:00
page = grab_cache_page ( mapping , index ) ;
if ( ! page )
goto out ;
if ( ! PageUptodate ( page ) ) {
2007-06-15 21:50:00 +04:00
ret = btrfs_readpage ( NULL , page ) ;
2007-06-12 14:35:45 +04:00
lock_page ( page ) ;
2008-05-15 17:13:45 +04:00
if ( page - > mapping ! = mapping ) {
unlock_page ( page ) ;
page_cache_release ( page ) ;
goto again ;
}
2007-06-12 14:35:45 +04:00
if ( ! PageUptodate ( page ) ) {
ret = - EIO ;
goto out ;
}
}
2007-08-28 00:49:44 +04:00
2008-05-15 17:13:45 +04:00
page_start = ( u64 ) page - > index < < PAGE_CACHE_SHIFT ;
wait_on_page_writeback ( page ) ;
2007-08-28 00:49:44 +04:00
ret = btrfs_cow_one_page ( inode , page , offset ) ;
2007-06-12 14:35:45 +04:00
unlock_page ( page ) ;
page_cache_release ( page ) ;
out :
return ret ;
}
static int btrfs_setattr ( struct dentry * dentry , struct iattr * attr )
{
struct inode * inode = dentry - > d_inode ;
int err ;
err = inode_change_ok ( inode , attr ) ;
if ( err )
return err ;
if ( S_ISREG ( inode - > i_mode ) & &
attr - > ia_valid & ATTR_SIZE & & attr - > ia_size > inode - > i_size ) {
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-08-30 19:54:02 +04:00
2007-10-16 00:14:19 +04:00
u64 mask = root - > sectorsize - 1 ;
2008-01-30 22:33:02 +03:00
u64 hole_start = ( inode - > i_size + mask ) & ~ mask ;
2008-01-30 19:54:05 +03:00
u64 block_end = ( attr - > ia_size + mask ) & ~ mask ;
2007-06-12 14:35:45 +04:00
u64 hole_size ;
2007-11-01 18:28:41 +03:00
u64 alloc_hint = 0 ;
2007-06-12 14:35:45 +04:00
2008-01-30 22:33:02 +03:00
if ( attr - > ia_size < = hole_start )
2007-06-12 14:35:45 +04:00
goto out ;
2007-12-22 00:27:21 +03:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
err = btrfs_check_free_space ( root , 1 , 0 ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( err )
goto fail ;
2007-06-12 14:35:45 +04:00
btrfs_truncate_page ( inode - > i_mapping , inode - > i_size ) ;
2008-01-30 22:33:02 +03:00
lock_extent ( io_tree , hole_start , block_end - 1 , GFP_NOFS ) ;
2008-01-23 00:47:59 +03:00
hole_size = block_end - hole_start ;
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , inode ) ;
2007-08-30 19:54:02 +04:00
err = btrfs_drop_extents ( trans , root , inode ,
2008-01-30 22:33:02 +03:00
hole_start , block_end , hole_start ,
2007-10-16 00:18:25 +04:00
& alloc_hint ) ;
2007-08-30 19:54:02 +04:00
2007-11-01 18:28:41 +03:00
if ( alloc_hint ! = EXTENT_MAP_INLINE ) {
err = btrfs_insert_file_extent ( trans , root ,
inode - > i_ino ,
2008-01-23 00:47:59 +03:00
hole_start , 0 , 0 ,
2008-05-02 22:43:14 +04:00
hole_size , 0 ) ;
2008-01-25 00:13:08 +03:00
btrfs_drop_extent_cache ( inode , hole_start ,
2008-04-17 19:29:12 +04:00
( u64 ) - 1 ) ;
2008-01-23 00:47:59 +03:00
btrfs_check_file ( root , inode ) ;
2007-11-01 18:28:41 +03:00
}
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2008-01-30 22:33:02 +03:00
unlock_extent ( io_tree , hole_start , block_end - 1 , GFP_NOFS ) ;
2007-06-22 22:16:25 +04:00
if ( err )
return err ;
2007-06-12 14:35:45 +04:00
}
out :
err = inode_setattr ( inode , attr ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-06-12 14:35:45 +04:00
return err ;
}
2008-01-15 00:24:38 +03:00
2007-06-12 14:35:45 +04:00
void btrfs_delete_inode ( struct inode * inode )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2007-09-17 18:58:06 +04:00
unsigned long nr ;
2007-06-12 14:35:45 +04:00
int ret ;
truncate_inode_pages ( & inode - > i_data , 0 ) ;
if ( is_bad_inode ( inode ) ) {
goto no_delete ;
}
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
inode - > i_size = 0 ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
btrfs_set_trans_block_group ( trans , inode ) ;
2008-01-29 23:11:36 +03:00
ret = btrfs_truncate_in_trans ( trans , root , inode , 0 ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto no_delete_lock ;
2008-01-29 23:11:36 +03:00
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2008-01-29 23:11:36 +03:00
clear_inode ( inode ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return ;
2007-06-22 22:16:25 +04:00
no_delete_lock :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-22 22:16:25 +04:00
btrfs_end_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
no_delete :
clear_inode ( inode ) ;
}
/*
* this returns the key found in the dir entry in the location pointer .
* If no dir entries were found , location - > objectid is 0.
*/
static int btrfs_inode_by_name ( struct inode * dir , struct dentry * dentry ,
struct btrfs_key * location )
{
const char * name = dentry - > d_name . name ;
int namelen = dentry - > d_name . len ;
struct btrfs_dir_item * di ;
struct btrfs_path * path ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
2007-10-25 23:48:28 +04:00
int ret = 0 ;
2007-06-12 14:35:45 +04:00
2007-12-12 22:38:19 +03:00
if ( namelen = = 1 & & strcmp ( name , " . " ) = = 0 ) {
location - > objectid = dir - > i_ino ;
location - > type = BTRFS_INODE_ITEM_KEY ;
location - > offset = 0 ;
return 0 ;
}
2007-06-12 14:35:45 +04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
2007-12-12 22:38:19 +03:00
2007-12-13 17:06:59 +03:00
if ( namelen = = 2 & & strcmp ( name , " .. " ) = = 0 ) {
2007-12-12 22:38:19 +03:00
struct btrfs_key key ;
struct extent_buffer * leaf ;
u32 nritems ;
int slot ;
key . objectid = dir - > i_ino ;
btrfs_set_key_type ( & key , BTRFS_INODE_REF_KEY ) ;
key . offset = 0 ;
ret = btrfs_search_slot ( NULL , root , & key , path , 0 , 0 ) ;
BUG_ON ( ret = = 0 ) ;
ret = 0 ;
leaf = path - > nodes [ 0 ] ;
slot = path - > slots [ 0 ] ;
nritems = btrfs_header_nritems ( leaf ) ;
if ( slot > = nritems )
goto out_err ;
btrfs_item_key_to_cpu ( leaf , & key , slot ) ;
if ( key . objectid ! = dir - > i_ino | |
key . type ! = BTRFS_INODE_REF_KEY ) {
goto out_err ;
}
location - > objectid = key . offset ;
location - > type = BTRFS_INODE_ITEM_KEY ;
location - > offset = 0 ;
goto out ;
}
2007-06-12 14:35:45 +04:00
di = btrfs_lookup_dir_item ( NULL , root , path , dir - > i_ino , name ,
namelen , 0 ) ;
2007-10-25 23:48:28 +04:00
if ( IS_ERR ( di ) )
ret = PTR_ERR ( di ) ;
2007-06-12 14:35:45 +04:00
if ( ! di | | IS_ERR ( di ) ) {
2007-12-12 22:38:19 +03:00
goto out_err ;
2007-06-12 14:35:45 +04:00
}
2007-10-16 00:14:19 +04:00
btrfs_dir_item_key_to_cpu ( path - > nodes [ 0 ] , di , location ) ;
2007-06-12 14:35:45 +04:00
out :
btrfs_free_path ( path ) ;
return ret ;
2007-12-12 22:38:19 +03:00
out_err :
location - > objectid = 0 ;
goto out ;
2007-06-12 14:35:45 +04:00
}
/*
* when we hit a tree root in a directory , the btrfs part of the inode
* needs to be changed to reflect the root directory of the tree root . This
* is kind of like crossing a mount point .
*/
static int fixup_tree_root_location ( struct btrfs_root * root ,
struct btrfs_key * location ,
2007-08-29 23:47:34 +04:00
struct btrfs_root * * sub_root ,
struct dentry * dentry )
2007-06-12 14:35:45 +04:00
{
struct btrfs_path * path ;
struct btrfs_root_item * ri ;
if ( btrfs_key_type ( location ) ! = BTRFS_ROOT_ITEM_KEY )
return 0 ;
if ( location - > objectid = = BTRFS_ROOT_TREE_OBJECTID )
return 0 ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-08-29 23:47:34 +04:00
* sub_root = btrfs_read_fs_root ( root - > fs_info , location ,
dentry - > d_name . name ,
dentry - > d_name . len ) ;
2007-06-12 14:35:45 +04:00
if ( IS_ERR ( * sub_root ) )
return PTR_ERR ( * sub_root ) ;
ri = & ( * sub_root ) - > root_item ;
location - > objectid = btrfs_root_dirid ( ri ) ;
btrfs_set_key_type ( location , BTRFS_INODE_ITEM_KEY ) ;
location - > offset = 0 ;
btrfs_free_path ( path ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return 0 ;
}
static int btrfs_init_locked_inode ( struct inode * inode , void * p )
{
struct btrfs_iget_args * args = p ;
inode - > i_ino = args - > ino ;
BTRFS_I ( inode ) - > root = args - > root ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes = 0 ;
2008-01-25 00:13:08 +03:00
extent_map_tree_init ( & BTRFS_I ( inode ) - > extent_tree , GFP_NOFS ) ;
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_tree ,
2007-08-28 00:49:44 +04:00
inode - > i_mapping , GFP_NOFS ) ;
2008-04-10 00:28:12 +04:00
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_failure_tree ,
inode - > i_mapping , GFP_NOFS ) ;
2008-04-25 16:51:48 +04:00
atomic_set ( & BTRFS_I ( inode ) - > ordered_writeback , 0 ) ;
2007-06-12 14:35:45 +04:00
return 0 ;
}
static int btrfs_find_actor ( struct inode * inode , void * opaque )
{
struct btrfs_iget_args * args = opaque ;
return ( args - > ino = = inode - > i_ino & &
args - > root = = BTRFS_I ( inode ) - > root ) ;
}
2008-01-08 23:46:30 +03:00
struct inode * btrfs_ilookup ( struct super_block * s , u64 objectid ,
u64 root_objectid )
{
struct btrfs_iget_args args ;
args . ino = objectid ;
args . root = btrfs_lookup_fs_root ( btrfs_sb ( s ) - > fs_info , root_objectid ) ;
if ( ! args . root )
return NULL ;
return ilookup5 ( s , objectid , btrfs_find_actor , ( void * ) & args ) ;
}
2007-06-12 14:35:45 +04:00
struct inode * btrfs_iget_locked ( struct super_block * s , u64 objectid ,
struct btrfs_root * root )
{
struct inode * inode ;
struct btrfs_iget_args args ;
args . ino = objectid ;
args . root = root ;
inode = iget5_locked ( s , objectid , btrfs_find_actor ,
btrfs_init_locked_inode ,
( void * ) & args ) ;
return inode ;
}
static struct dentry * btrfs_lookup ( struct inode * dir , struct dentry * dentry ,
struct nameidata * nd )
{
struct inode * inode ;
struct btrfs_inode * bi = BTRFS_I ( dir ) ;
struct btrfs_root * root = bi - > root ;
struct btrfs_root * sub_root = root ;
struct btrfs_key location ;
int ret ;
if ( dentry - > d_name . len > BTRFS_NAME_LEN )
return ERR_PTR ( - ENAMETOOLONG ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
ret = btrfs_inode_by_name ( dir , dentry , & location ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
if ( ret < 0 )
return ERR_PTR ( ret ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
inode = NULL ;
if ( location . objectid ) {
2007-08-29 23:47:34 +04:00
ret = fixup_tree_root_location ( root , & location , & sub_root ,
dentry ) ;
2007-06-12 14:35:45 +04:00
if ( ret < 0 )
return ERR_PTR ( ret ) ;
if ( ret > 0 )
return ERR_PTR ( - ENOENT ) ;
inode = btrfs_iget_locked ( dir - > i_sb , location . objectid ,
sub_root ) ;
if ( ! inode )
return ERR_PTR ( - EACCES ) ;
if ( inode - > i_state & I_NEW ) {
/* the inode and parent dir are two different roots */
if ( sub_root ! = root ) {
igrab ( inode ) ;
sub_root - > inode = inode ;
}
BTRFS_I ( inode ) - > root = sub_root ;
memcpy ( & BTRFS_I ( inode ) - > location , & location ,
sizeof ( location ) ) ;
btrfs_read_locked_inode ( inode ) ;
unlock_new_inode ( inode ) ;
}
}
return d_splice_alias ( inode , dentry ) ;
}
static unsigned char btrfs_filetype_table [ ] = {
DT_UNKNOWN , DT_REG , DT_DIR , DT_CHR , DT_BLK , DT_FIFO , DT_SOCK , DT_LNK
} ;
static int btrfs_readdir ( struct file * filp , void * dirent , filldir_t filldir )
{
2007-12-19 00:15:09 +03:00
struct inode * inode = filp - > f_dentry - > d_inode ;
2007-06-12 14:35:45 +04:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_item * item ;
struct btrfs_dir_item * di ;
struct btrfs_key key ;
2007-10-16 00:14:19 +04:00
struct btrfs_key found_key ;
2007-06-12 14:35:45 +04:00
struct btrfs_path * path ;
int ret ;
u32 nritems ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-06-12 14:35:45 +04:00
int slot ;
int advance ;
unsigned char d_type ;
int over = 0 ;
u32 di_cur ;
u32 di_total ;
u32 di_len ;
int key_type = BTRFS_DIR_INDEX_KEY ;
2007-10-16 00:14:19 +04:00
char tmp_name [ 32 ] ;
char * name_ptr ;
int name_len ;
2007-06-12 14:35:45 +04:00
/* FIXME, use a real flag for deciding about the key type */
if ( root - > fs_info - > tree_root = = root )
key_type = BTRFS_DIR_ITEM_KEY ;
2007-10-16 00:14:19 +04:00
2007-12-12 22:38:19 +03:00
/* special case for "." */
if ( filp - > f_pos = = 0 ) {
over = filldir ( dirent , " . " , 1 ,
1 , inode - > i_ino ,
DT_DIR ) ;
if ( over )
return 0 ;
filp - > f_pos = 1 ;
}
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
key . objectid = inode - > i_ino ;
2007-12-12 22:38:19 +03:00
path = btrfs_alloc_path ( ) ;
path - > reada = 2 ;
/* special case for .., just use the back ref */
if ( filp - > f_pos = = 1 ) {
btrfs_set_key_type ( & key , BTRFS_INODE_REF_KEY ) ;
key . offset = 0 ;
ret = btrfs_search_slot ( NULL , root , & key , path , 0 , 0 ) ;
BUG_ON ( ret = = 0 ) ;
leaf = path - > nodes [ 0 ] ;
slot = path - > slots [ 0 ] ;
nritems = btrfs_header_nritems ( leaf ) ;
if ( slot > = nritems ) {
btrfs_release_path ( root , path ) ;
goto read_dir_items ;
}
btrfs_item_key_to_cpu ( leaf , & found_key , slot ) ;
btrfs_release_path ( root , path ) ;
if ( found_key . objectid ! = key . objectid | |
found_key . type ! = BTRFS_INODE_REF_KEY )
goto read_dir_items ;
over = filldir ( dirent , " .. " , 2 ,
2 , found_key . offset , DT_DIR ) ;
if ( over )
goto nopos ;
filp - > f_pos = 2 ;
}
read_dir_items :
2007-06-12 14:35:45 +04:00
btrfs_set_key_type ( & key , key_type ) ;
key . offset = filp - > f_pos ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
ret = btrfs_search_slot ( NULL , root , & key , path , 0 , 0 ) ;
if ( ret < 0 )
goto err ;
advance = 0 ;
while ( 1 ) {
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
nritems = btrfs_header_nritems ( leaf ) ;
2007-06-12 14:35:45 +04:00
slot = path - > slots [ 0 ] ;
if ( advance | | slot > = nritems ) {
if ( slot > = nritems - 1 ) {
ret = btrfs_next_leaf ( root , path ) ;
if ( ret )
break ;
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
nritems = btrfs_header_nritems ( leaf ) ;
2007-06-12 14:35:45 +04:00
slot = path - > slots [ 0 ] ;
} else {
slot + + ;
path - > slots [ 0 ] + + ;
}
}
advance = 1 ;
2007-10-16 00:14:19 +04:00
item = btrfs_item_nr ( leaf , slot ) ;
btrfs_item_key_to_cpu ( leaf , & found_key , slot ) ;
if ( found_key . objectid ! = key . objectid )
2007-06-12 14:35:45 +04:00
break ;
2007-10-16 00:14:19 +04:00
if ( btrfs_key_type ( & found_key ) ! = key_type )
2007-06-12 14:35:45 +04:00
break ;
2007-10-16 00:14:19 +04:00
if ( found_key . offset < filp - > f_pos )
2007-06-12 14:35:45 +04:00
continue ;
2007-10-16 00:14:19 +04:00
filp - > f_pos = found_key . offset ;
2007-06-12 14:35:45 +04:00
advance = 1 ;
di = btrfs_item_ptr ( leaf , slot , struct btrfs_dir_item ) ;
di_cur = 0 ;
2007-10-16 00:14:19 +04:00
di_total = btrfs_item_size ( leaf , item ) ;
2007-06-12 14:35:45 +04:00
while ( di_cur < di_total ) {
2007-10-16 00:14:19 +04:00
struct btrfs_key location ;
name_len = btrfs_dir_name_len ( leaf , di ) ;
if ( name_len < 32 ) {
name_ptr = tmp_name ;
} else {
name_ptr = kmalloc ( name_len , GFP_NOFS ) ;
BUG_ON ( ! name_ptr ) ;
}
read_extent_buffer ( leaf , name_ptr ,
( unsigned long ) ( di + 1 ) , name_len ) ;
d_type = btrfs_filetype_table [ btrfs_dir_type ( leaf , di ) ] ;
btrfs_dir_item_key_to_cpu ( leaf , di , & location ) ;
over = filldir ( dirent , name_ptr , name_len ,
found_key . offset ,
location . objectid ,
2007-06-12 14:35:45 +04:00
d_type ) ;
2007-10-16 00:14:19 +04:00
if ( name_ptr ! = tmp_name )
kfree ( name_ptr ) ;
2007-06-12 14:35:45 +04:00
if ( over )
goto nopos ;
2007-11-16 19:45:54 +03:00
di_len = btrfs_dir_name_len ( leaf , di ) +
btrfs_dir_data_len ( leaf , di ) + sizeof ( * di ) ;
2007-06-12 14:35:45 +04:00
di_cur + = di_len ;
di = ( struct btrfs_dir_item * ) ( ( char * ) di + di_len ) ;
}
}
2008-02-19 19:41:02 +03:00
if ( key_type = = BTRFS_DIR_INDEX_KEY )
filp - > f_pos = INT_LIMIT ( typeof ( filp - > f_pos ) ) ;
else
filp - > f_pos + + ;
2007-06-12 14:35:45 +04:00
nopos :
ret = 0 ;
err :
btrfs_release_path ( root , path ) ;
btrfs_free_path ( path ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
int btrfs_write_inode ( struct inode * inode , int wait )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
int ret = 0 ;
if ( wait ) {
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , inode ) ;
ret = btrfs_commit_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
}
return ret ;
}
/*
2007-06-22 22:16:25 +04:00
* This is somewhat expensive , updating the tree every time the
2007-06-12 14:35:45 +04:00
* inode changes . But , it is most likely to find the inode in cache .
* FIXME , needs more benchmarking . . . there are no reasons other than performance
* to keep or drop this code .
*/
void btrfs_dirty_inode ( struct inode * inode )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , inode ) ;
btrfs_update_inode ( trans , root , inode ) ;
btrfs_end_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
}
static struct inode * btrfs_new_inode ( struct btrfs_trans_handle * trans ,
struct btrfs_root * root ,
2008-01-29 23:15:18 +03:00
const char * name , int name_len ,
u64 ref_objectid ,
2007-06-12 14:35:45 +04:00
u64 objectid ,
struct btrfs_block_group_cache * group ,
int mode )
{
struct inode * inode ;
2007-10-16 00:14:19 +04:00
struct btrfs_inode_item * inode_item ;
2008-03-24 22:01:59 +03:00
struct btrfs_block_group_cache * new_inode_group ;
2007-06-12 14:35:45 +04:00
struct btrfs_key * location ;
2007-10-16 00:14:19 +04:00
struct btrfs_path * path ;
2008-01-29 23:15:18 +03:00
struct btrfs_inode_ref * ref ;
struct btrfs_key key [ 2 ] ;
u32 sizes [ 2 ] ;
unsigned long ptr ;
2007-06-12 14:35:45 +04:00
int ret ;
int owner ;
2007-10-16 00:14:19 +04:00
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
2007-06-12 14:35:45 +04:00
inode = new_inode ( root - > fs_info - > sb ) ;
if ( ! inode )
return ERR_PTR ( - ENOMEM ) ;
2008-01-25 00:13:08 +03:00
extent_map_tree_init ( & BTRFS_I ( inode ) - > extent_tree , GFP_NOFS ) ;
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_tree ,
2007-08-28 00:49:44 +04:00
inode - > i_mapping , GFP_NOFS ) ;
2008-04-10 00:28:12 +04:00
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_failure_tree ,
inode - > i_mapping , GFP_NOFS ) ;
2008-04-25 16:51:48 +04:00
atomic_set ( & BTRFS_I ( inode ) - > ordered_writeback , 0 ) ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes = 0 ;
2007-06-12 14:35:45 +04:00
BTRFS_I ( inode ) - > root = root ;
2007-08-28 00:49:44 +04:00
2007-06-12 14:35:45 +04:00
if ( mode & S_IFDIR )
owner = 0 ;
else
owner = 1 ;
2008-03-24 22:01:59 +03:00
new_inode_group = btrfs_find_block_group ( root , group , 0 ,
2008-03-24 22:01:56 +03:00
BTRFS_BLOCK_GROUP_METADATA , owner ) ;
2008-03-24 22:01:59 +03:00
if ( ! new_inode_group ) {
printk ( " find_block group failed \n " ) ;
new_inode_group = group ;
}
BTRFS_I ( inode ) - > block_group = new_inode_group ;
2008-01-08 23:54:37 +03:00
BTRFS_I ( inode ) - > flags = 0 ;
2008-01-29 23:15:18 +03:00
key [ 0 ] . objectid = objectid ;
btrfs_set_key_type ( & key [ 0 ] , BTRFS_INODE_ITEM_KEY ) ;
key [ 0 ] . offset = 0 ;
key [ 1 ] . objectid = objectid ;
btrfs_set_key_type ( & key [ 1 ] , BTRFS_INODE_REF_KEY ) ;
key [ 1 ] . offset = ref_objectid ;
sizes [ 0 ] = sizeof ( struct btrfs_inode_item ) ;
sizes [ 1 ] = name_len + sizeof ( * ref ) ;
ret = btrfs_insert_empty_items ( trans , root , path , key , sizes , 2 ) ;
if ( ret ! = 0 )
2007-10-16 00:14:19 +04:00
goto fail ;
2008-01-29 23:15:18 +03:00
if ( objectid > root - > highest_inode )
root - > highest_inode = objectid ;
2007-06-12 14:35:45 +04:00
inode - > i_uid = current - > fsuid ;
inode - > i_gid = current - > fsgid ;
inode - > i_mode = mode ;
inode - > i_ino = objectid ;
inode - > i_blocks = 0 ;
inode - > i_mtime = inode - > i_atime = inode - > i_ctime = CURRENT_TIME ;
2007-10-16 00:14:19 +04:00
inode_item = btrfs_item_ptr ( path - > nodes [ 0 ] , path - > slots [ 0 ] ,
struct btrfs_inode_item ) ;
fill_inode_item ( path - > nodes [ 0 ] , inode_item , inode ) ;
2008-01-29 23:15:18 +03:00
ref = btrfs_item_ptr ( path - > nodes [ 0 ] , path - > slots [ 0 ] + 1 ,
struct btrfs_inode_ref ) ;
btrfs_set_inode_ref_name_len ( path - > nodes [ 0 ] , ref , name_len ) ;
ptr = ( unsigned long ) ( ref + 1 ) ;
write_extent_buffer ( path - > nodes [ 0 ] , name , ptr , name_len ) ;
2007-10-16 00:14:19 +04:00
btrfs_mark_buffer_dirty ( path - > nodes [ 0 ] ) ;
btrfs_free_path ( path ) ;
2007-06-12 14:35:45 +04:00
location = & BTRFS_I ( inode ) - > location ;
location - > objectid = objectid ;
location - > offset = 0 ;
btrfs_set_key_type ( location , BTRFS_INODE_ITEM_KEY ) ;
insert_inode_hash ( inode ) ;
return inode ;
2007-10-16 00:14:19 +04:00
fail :
btrfs_free_path ( path ) ;
return ERR_PTR ( ret ) ;
2007-06-12 14:35:45 +04:00
}
static inline u8 btrfs_inode_type ( struct inode * inode )
{
return btrfs_type_by_mode [ ( inode - > i_mode & S_IFMT ) > > S_SHIFT ] ;
}
static int btrfs_add_link ( struct btrfs_trans_handle * trans ,
2008-01-29 23:15:18 +03:00
struct dentry * dentry , struct inode * inode ,
int add_backref )
2007-06-12 14:35:45 +04:00
{
int ret ;
struct btrfs_key key ;
struct btrfs_root * root = BTRFS_I ( dentry - > d_parent - > d_inode ) - > root ;
2007-06-25 18:09:33 +04:00
struct inode * parent_inode ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
key . objectid = inode - > i_ino ;
btrfs_set_key_type ( & key , BTRFS_INODE_ITEM_KEY ) ;
key . offset = 0 ;
ret = btrfs_insert_dir_item ( trans , root ,
dentry - > d_name . name , dentry - > d_name . len ,
dentry - > d_parent - > d_inode - > i_ino ,
& key , btrfs_inode_type ( inode ) ) ;
if ( ret = = 0 ) {
2008-01-29 23:15:18 +03:00
if ( add_backref ) {
ret = btrfs_insert_inode_ref ( trans , root ,
dentry - > d_name . name ,
dentry - > d_name . len ,
inode - > i_ino ,
dentry - > d_parent - > d_inode - > i_ino ) ;
}
2007-06-25 18:09:33 +04:00
parent_inode = dentry - > d_parent - > d_inode ;
parent_inode - > i_size + = dentry - > d_name . len * 2 ;
parent_inode - > i_mtime = parent_inode - > i_ctime = CURRENT_TIME ;
2007-06-12 14:35:45 +04:00
ret = btrfs_update_inode ( trans , root ,
dentry - > d_parent - > d_inode ) ;
}
return ret ;
}
static int btrfs_add_nondir ( struct btrfs_trans_handle * trans ,
2008-01-29 23:15:18 +03:00
struct dentry * dentry , struct inode * inode ,
int backref )
2007-06-12 14:35:45 +04:00
{
2008-01-29 23:15:18 +03:00
int err = btrfs_add_link ( trans , dentry , inode , backref ) ;
2007-06-12 14:35:45 +04:00
if ( ! err ) {
d_instantiate ( dentry , inode ) ;
return 0 ;
}
if ( err > 0 )
err = - EEXIST ;
return err ;
}
2007-07-11 18:18:17 +04:00
static int btrfs_mknod ( struct inode * dir , struct dentry * dentry ,
int mode , dev_t rdev )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
2007-12-22 00:27:21 +03:00
struct inode * inode = NULL ;
2007-07-11 18:18:17 +04:00
int err ;
int drop_inode = 0 ;
u64 objectid ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-07-11 18:18:17 +04:00
if ( ! new_valid_dev ( rdev ) )
return - EINVAL ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
err = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( err )
goto fail ;
2007-07-11 18:18:17 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , dir ) ;
err = btrfs_find_free_objectid ( trans , root , dir - > i_ino , & objectid ) ;
if ( err ) {
err = - ENOSPC ;
goto out_unlock ;
}
2008-01-29 23:15:18 +03:00
inode = btrfs_new_inode ( trans , root , dentry - > d_name . name ,
dentry - > d_name . len ,
dentry - > d_parent - > d_inode - > i_ino , objectid ,
2007-07-11 18:18:17 +04:00
BTRFS_I ( dir ) - > block_group , mode ) ;
err = PTR_ERR ( inode ) ;
if ( IS_ERR ( inode ) )
goto out_unlock ;
btrfs_set_trans_block_group ( trans , inode ) ;
2008-01-29 23:15:18 +03:00
err = btrfs_add_nondir ( trans , dentry , inode , 0 ) ;
2007-07-11 18:18:17 +04:00
if ( err )
drop_inode = 1 ;
else {
inode - > i_op = & btrfs_special_inode_operations ;
init_special_inode ( inode , inode - > i_mode , rdev ) ;
2007-08-29 17:11:44 +04:00
btrfs_update_inode ( trans , root , inode ) ;
2007-07-11 18:18:17 +04:00
}
dir - > i_sb - > s_dirt = 1 ;
btrfs_update_inode_block_group ( trans , inode ) ;
btrfs_update_inode_block_group ( trans , dir ) ;
out_unlock :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-07-11 18:18:17 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-07-11 18:18:17 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( drop_inode ) {
inode_dec_link_count ( inode ) ;
iput ( inode ) ;
}
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-07-11 18:18:17 +04:00
return err ;
}
2007-06-12 14:35:45 +04:00
static int btrfs_create ( struct inode * dir , struct dentry * dentry ,
int mode , struct nameidata * nd )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
2007-12-22 00:27:21 +03:00
struct inode * inode = NULL ;
2007-06-12 14:35:45 +04:00
int err ;
int drop_inode = 0 ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
u64 objectid ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
err = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( err )
goto fail ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , dir ) ;
err = btrfs_find_free_objectid ( trans , root , dir - > i_ino , & objectid ) ;
if ( err ) {
err = - ENOSPC ;
goto out_unlock ;
}
2008-01-29 23:15:18 +03:00
inode = btrfs_new_inode ( trans , root , dentry - > d_name . name ,
dentry - > d_name . len ,
dentry - > d_parent - > d_inode - > i_ino ,
objectid , BTRFS_I ( dir ) - > block_group , mode ) ;
2007-06-12 14:35:45 +04:00
err = PTR_ERR ( inode ) ;
if ( IS_ERR ( inode ) )
goto out_unlock ;
btrfs_set_trans_block_group ( trans , inode ) ;
2008-01-29 23:15:18 +03:00
err = btrfs_add_nondir ( trans , dentry , inode , 0 ) ;
2007-06-12 14:35:45 +04:00
if ( err )
drop_inode = 1 ;
else {
inode - > i_mapping - > a_ops = & btrfs_aops ;
2008-03-26 17:28:07 +03:00
inode - > i_mapping - > backing_dev_info = & root - > fs_info - > bdi ;
2007-06-12 14:35:45 +04:00
inode - > i_fop = & btrfs_file_operations ;
inode - > i_op = & btrfs_file_inode_operations ;
2008-01-25 00:13:08 +03:00
extent_map_tree_init ( & BTRFS_I ( inode ) - > extent_tree , GFP_NOFS ) ;
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_tree ,
2007-08-28 00:49:44 +04:00
inode - > i_mapping , GFP_NOFS ) ;
2008-04-10 00:28:12 +04:00
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_failure_tree ,
inode - > i_mapping , GFP_NOFS ) ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes = 0 ;
2008-04-25 16:51:48 +04:00
atomic_set ( & BTRFS_I ( inode ) - > ordered_writeback , 0 ) ;
2008-01-25 00:13:08 +03:00
BTRFS_I ( inode ) - > io_tree . ops = & btrfs_extent_io_ops ;
2007-06-12 14:35:45 +04:00
}
dir - > i_sb - > s_dirt = 1 ;
btrfs_update_inode_block_group ( trans , inode ) ;
btrfs_update_inode_block_group ( trans , dir ) ;
out_unlock :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( drop_inode ) {
inode_dec_link_count ( inode ) ;
iput ( inode ) ;
}
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return err ;
}
static int btrfs_link ( struct dentry * old_dentry , struct inode * dir ,
struct dentry * dentry )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
struct inode * inode = old_dentry - > d_inode ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
int err ;
int drop_inode = 0 ;
if ( inode - > i_nlink = = 0 )
return - ENOENT ;
2007-12-19 00:15:09 +03:00
# if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,18)
inode - > i_nlink + + ;
# else
2007-06-12 14:35:45 +04:00
inc_nlink ( inode ) ;
2007-12-19 00:15:09 +03:00
# endif
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
err = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( err )
goto fail ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
btrfs_set_trans_block_group ( trans , dir ) ;
atomic_inc ( & inode - > i_count ) ;
2008-01-29 23:15:18 +03:00
err = btrfs_add_nondir ( trans , dentry , inode , 1 ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
if ( err )
drop_inode = 1 ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
dir - > i_sb - > s_dirt = 1 ;
btrfs_update_inode_block_group ( trans , dir ) ;
2007-06-22 22:16:25 +04:00
err = btrfs_update_inode ( trans , root , inode ) ;
2007-10-16 00:14:19 +04:00
2007-06-22 22:16:25 +04:00
if ( err )
drop_inode = 1 ;
2007-06-12 14:35:45 +04:00
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
fail :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( drop_inode ) {
inode_dec_link_count ( inode ) ;
iput ( inode ) ;
}
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return err ;
}
static int btrfs_mkdir ( struct inode * dir , struct dentry * dentry , int mode )
{
2008-05-03 00:13:49 +04:00
struct inode * inode = NULL ;
2007-06-12 14:35:45 +04:00
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
int err = 0 ;
int drop_on_err = 0 ;
2008-05-03 00:13:49 +04:00
u64 objectid = 0 ;
2007-09-17 18:58:06 +04:00
unsigned long nr = 1 ;
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
err = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( err )
goto out_unlock ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , dir ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
if ( IS_ERR ( trans ) ) {
err = PTR_ERR ( trans ) ;
goto out_unlock ;
}
err = btrfs_find_free_objectid ( trans , root , dir - > i_ino , & objectid ) ;
if ( err ) {
err = - ENOSPC ;
goto out_unlock ;
}
2008-01-29 23:15:18 +03:00
inode = btrfs_new_inode ( trans , root , dentry - > d_name . name ,
dentry - > d_name . len ,
dentry - > d_parent - > d_inode - > i_ino , objectid ,
2007-06-12 14:35:45 +04:00
BTRFS_I ( dir ) - > block_group , S_IFDIR | mode ) ;
if ( IS_ERR ( inode ) ) {
err = PTR_ERR ( inode ) ;
goto out_fail ;
}
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
drop_on_err = 1 ;
inode - > i_op = & btrfs_dir_inode_operations ;
inode - > i_fop = & btrfs_dir_file_operations ;
btrfs_set_trans_block_group ( trans , inode ) ;
2007-12-12 22:38:19 +03:00
inode - > i_size = 0 ;
2007-06-12 14:35:45 +04:00
err = btrfs_update_inode ( trans , root , inode ) ;
if ( err )
goto out_fail ;
2007-10-16 00:14:19 +04:00
2008-01-29 23:15:18 +03:00
err = btrfs_add_link ( trans , dentry , inode , 0 ) ;
2007-06-12 14:35:45 +04:00
if ( err )
goto out_fail ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
d_instantiate ( dentry , inode ) ;
drop_on_err = 0 ;
dir - > i_sb - > s_dirt = 1 ;
btrfs_update_inode_block_group ( trans , inode ) ;
btrfs_update_inode_block_group ( trans , dir ) ;
out_fail :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
out_unlock :
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( drop_on_err )
iput ( inode ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return err ;
}
2008-04-17 19:29:12 +04:00
static int merge_extent_mapping ( struct extent_map_tree * em_tree ,
struct extent_map * existing ,
struct extent_map * em )
{
u64 start_diff ;
u64 new_end ;
int ret = 0 ;
int real_blocks = existing - > block_start < EXTENT_MAP_LAST_BYTE ;
if ( real_blocks & & em - > block_start > = EXTENT_MAP_LAST_BYTE )
goto invalid ;
if ( ! real_blocks & & em - > block_start ! = existing - > block_start )
goto invalid ;
new_end = max ( existing - > start + existing - > len , em - > start + em - > len ) ;
if ( existing - > start > = em - > start ) {
if ( em - > start + em - > len < existing - > start )
goto invalid ;
start_diff = existing - > start - em - > start ;
if ( real_blocks & & em - > block_start + start_diff ! =
existing - > block_start )
goto invalid ;
em - > len = new_end - em - > start ;
remove_extent_mapping ( em_tree , existing ) ;
/* free for the tree */
free_extent_map ( existing ) ;
ret = add_extent_mapping ( em_tree , em ) ;
} else if ( em - > start > existing - > start ) {
if ( existing - > start + existing - > len < em - > start )
goto invalid ;
start_diff = em - > start - existing - > start ;
if ( real_blocks & & existing - > block_start + start_diff ! =
em - > block_start )
goto invalid ;
remove_extent_mapping ( em_tree , existing ) ;
em - > block_start = existing - > block_start ;
em - > start = existing - > start ;
em - > len = new_end - existing - > start ;
free_extent_map ( existing ) ;
ret = add_extent_mapping ( em_tree , em ) ;
} else {
goto invalid ;
}
return ret ;
invalid :
printk ( " invalid extent map merge [%Lu %Lu %Lu] [%Lu %Lu %Lu] \n " ,
existing - > start , existing - > len , existing - > block_start ,
em - > start , em - > len , em - > block_start ) ;
return - EIO ;
}
2007-08-28 00:49:44 +04:00
struct extent_map * btrfs_get_extent ( struct inode * inode , struct page * page ,
2008-01-29 17:59:12 +03:00
size_t pg_offset , u64 start , u64 len ,
2007-08-28 00:49:44 +04:00
int create )
{
int ret ;
int err = 0 ;
2007-10-16 00:15:53 +04:00
u64 bytenr ;
2007-08-28 00:49:44 +04:00
u64 extent_start = 0 ;
u64 extent_end = 0 ;
u64 objectid = inode - > i_ino ;
u32 found_type ;
struct btrfs_path * path ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_file_extent_item * item ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
struct btrfs_key found_key ;
2007-08-28 00:49:44 +04:00
struct extent_map * em = NULL ;
struct extent_map_tree * em_tree = & BTRFS_I ( inode ) - > extent_tree ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-08-28 00:49:44 +04:00
struct btrfs_trans_handle * trans = NULL ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
again :
2008-01-25 00:13:08 +03:00
spin_lock ( & em_tree - > lock ) ;
em = lookup_extent_mapping ( em_tree , start , len ) ;
2008-05-07 19:43:44 +04:00
if ( em )
em - > bdev = root - > fs_info - > fs_devices - > latest_bdev ;
2008-01-25 00:13:08 +03:00
spin_unlock ( & em_tree - > lock ) ;
2007-08-28 00:49:44 +04:00
if ( em ) {
2008-04-22 21:26:46 +04:00
if ( em - > start > start | | em - > start + em - > len < = start )
free_extent_map ( em ) ;
else if ( em - > block_start = = EXTENT_MAP_INLINE & & page )
2008-01-29 17:59:12 +03:00
free_extent_map ( em ) ;
else
goto out ;
2007-08-28 00:49:44 +04:00
}
2008-01-25 00:13:08 +03:00
em = alloc_extent_map ( GFP_NOFS ) ;
2007-08-28 00:49:44 +04:00
if ( ! em ) {
2008-01-25 00:13:08 +03:00
err = - ENOMEM ;
goto out ;
2007-08-28 00:49:44 +04:00
}
2008-01-25 00:13:08 +03:00
em - > start = EXTENT_MAP_HOLE ;
em - > len = ( u64 ) - 1 ;
2008-05-07 19:43:44 +04:00
em - > bdev = root - > fs_info - > fs_devices - > latest_bdev ;
2007-11-01 18:28:41 +03:00
ret = btrfs_lookup_file_extent ( trans , root , path ,
objectid , start , trans ! = NULL ) ;
2007-08-28 00:49:44 +04:00
if ( ret < 0 ) {
err = ret ;
goto out ;
}
if ( ret ! = 0 ) {
if ( path - > slots [ 0 ] = = 0 )
goto not_found ;
path - > slots [ 0 ] - - ;
}
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
item = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
2007-08-28 00:49:44 +04:00
struct btrfs_file_extent_item ) ;
/* are we inside the extent that was found? */
2007-10-16 00:14:19 +04:00
btrfs_item_key_to_cpu ( leaf , & found_key , path - > slots [ 0 ] ) ;
found_type = btrfs_key_type ( & found_key ) ;
if ( found_key . objectid ! = objectid | |
2007-08-28 00:49:44 +04:00
found_type ! = BTRFS_EXTENT_DATA_KEY ) {
goto not_found ;
}
2007-10-16 00:14:19 +04:00
found_type = btrfs_file_extent_type ( leaf , item ) ;
extent_start = found_key . offset ;
2007-08-28 00:49:44 +04:00
if ( found_type = = BTRFS_FILE_EXTENT_REG ) {
extent_end = extent_start +
2007-10-16 00:15:53 +04:00
btrfs_file_extent_num_bytes ( leaf , item ) ;
2007-08-28 00:49:44 +04:00
err = 0 ;
2007-08-28 00:49:44 +04:00
if ( start < extent_start | | start > = extent_end ) {
2007-08-28 00:49:44 +04:00
em - > start = start ;
if ( start < extent_start ) {
2008-01-25 00:13:08 +03:00
if ( start + len < = extent_start )
2007-08-28 00:49:44 +04:00
goto not_found ;
2008-01-25 00:13:08 +03:00
em - > len = extent_end - extent_start ;
2007-08-28 00:49:44 +04:00
} else {
2008-01-25 00:13:08 +03:00
em - > len = len ;
2007-08-28 00:49:44 +04:00
}
goto not_found_em ;
}
2007-10-16 00:15:53 +04:00
bytenr = btrfs_file_extent_disk_bytenr ( leaf , item ) ;
if ( bytenr = = 0 ) {
2007-08-28 00:49:44 +04:00
em - > start = extent_start ;
2008-01-25 00:13:08 +03:00
em - > len = extent_end - extent_start ;
2007-10-16 00:14:19 +04:00
em - > block_start = EXTENT_MAP_HOLE ;
2007-08-28 00:49:44 +04:00
goto insert ;
}
2007-10-16 00:15:53 +04:00
bytenr + = btrfs_file_extent_offset ( leaf , item ) ;
em - > block_start = bytenr ;
2007-08-28 00:49:44 +04:00
em - > start = extent_start ;
2008-01-25 00:13:08 +03:00
em - > len = extent_end - extent_start ;
2007-08-28 00:49:44 +04:00
goto insert ;
} else if ( found_type = = BTRFS_FILE_EXTENT_INLINE ) {
2008-01-29 17:59:12 +03:00
u64 page_start ;
2007-10-16 00:14:19 +04:00
unsigned long ptr ;
2007-08-28 00:49:44 +04:00
char * map ;
2007-10-16 00:18:25 +04:00
size_t size ;
size_t extent_offset ;
size_t copy_size ;
2007-08-28 00:49:44 +04:00
2007-10-16 00:14:19 +04:00
size = btrfs_file_extent_inline_len ( leaf , btrfs_item_nr ( leaf ,
path - > slots [ 0 ] ) ) ;
2008-01-25 00:13:08 +03:00
extent_end = ( extent_start + size + root - > sectorsize - 1 ) &
~ ( ( u64 ) root - > sectorsize - 1 ) ;
2007-08-28 00:49:44 +04:00
if ( start < extent_start | | start > = extent_end ) {
2007-08-28 00:49:44 +04:00
em - > start = start ;
if ( start < extent_start ) {
2008-01-25 00:13:08 +03:00
if ( start + len < = extent_start )
2007-08-28 00:49:44 +04:00
goto not_found ;
2008-01-25 00:13:08 +03:00
em - > len = extent_end - extent_start ;
2007-08-28 00:49:44 +04:00
} else {
2008-01-25 00:13:08 +03:00
em - > len = len ;
2007-08-28 00:49:44 +04:00
}
goto not_found_em ;
}
2007-10-29 18:41:07 +03:00
em - > block_start = EXTENT_MAP_INLINE ;
if ( ! page ) {
em - > start = extent_start ;
2008-01-25 00:13:08 +03:00
em - > len = size ;
2007-10-29 18:41:07 +03:00
goto out ;
}
2007-10-16 00:14:19 +04:00
2008-01-29 17:59:12 +03:00
page_start = page_offset ( page ) + pg_offset ;
extent_offset = page_start - extent_start ;
copy_size = min_t ( u64 , PAGE_CACHE_SIZE - pg_offset ,
2007-10-16 00:18:25 +04:00
size - extent_offset ) ;
em - > start = extent_start + extent_offset ;
2008-01-29 17:59:12 +03:00
em - > len = ( copy_size + root - > sectorsize - 1 ) &
~ ( ( u64 ) root - > sectorsize - 1 ) ;
2007-10-29 18:41:07 +03:00
map = kmap ( page ) ;
ptr = btrfs_file_extent_inline_start ( item ) + extent_offset ;
2007-11-01 18:28:41 +03:00
if ( create = = 0 & & ! PageUptodate ( page ) ) {
2008-01-29 17:59:12 +03:00
read_extent_buffer ( leaf , map + pg_offset , ptr ,
2007-11-01 18:28:41 +03:00
copy_size ) ;
flush_dcache_page ( page ) ;
} else if ( create & & PageUptodate ( page ) ) {
if ( ! trans ) {
kunmap ( page ) ;
free_extent_map ( em ) ;
em = NULL ;
btrfs_release_path ( root , path ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
goto again ;
}
2008-01-29 17:59:12 +03:00
write_extent_buffer ( leaf , map + pg_offset , ptr ,
2007-11-01 18:28:41 +03:00
copy_size ) ;
btrfs_mark_buffer_dirty ( leaf ) ;
2007-08-28 00:49:44 +04:00
}
kunmap ( page ) ;
2008-01-25 00:13:08 +03:00
set_extent_uptodate ( io_tree , em - > start ,
extent_map_end ( em ) - 1 , GFP_NOFS ) ;
2007-08-28 00:49:44 +04:00
goto insert ;
} else {
printk ( " unkknown found_type %d \n " , found_type ) ;
WARN_ON ( 1 ) ;
}
not_found :
em - > start = start ;
2008-01-25 00:13:08 +03:00
em - > len = len ;
2007-08-28 00:49:44 +04:00
not_found_em :
2007-10-16 00:14:19 +04:00
em - > block_start = EXTENT_MAP_HOLE ;
2007-08-28 00:49:44 +04:00
insert :
btrfs_release_path ( root , path ) ;
2008-01-25 00:13:08 +03:00
if ( em - > start > start | | extent_map_end ( em ) < = start ) {
printk ( " bad extent! em: [%Lu %Lu] passed [%Lu %Lu] \n " , em - > start , em - > len , start , len ) ;
2007-08-28 00:49:44 +04:00
err = - EIO ;
goto out ;
}
2008-01-25 00:13:08 +03:00
err = 0 ;
spin_lock ( & em_tree - > lock ) ;
2007-08-28 00:49:44 +04:00
ret = add_extent_mapping ( em_tree , em ) ;
2008-04-17 19:29:12 +04:00
/* it is possible that someone inserted the extent into the tree
* while we had the lock dropped . It is also possible that
* an overlapping map exists in the tree
*/
2007-08-28 00:49:44 +04:00
if ( ret = = - EEXIST ) {
2008-04-17 19:29:12 +04:00
struct extent_map * existing ;
existing = lookup_extent_mapping ( em_tree , start , len ) ;
2008-04-22 21:26:46 +04:00
if ( existing & & ( existing - > start > start | |
existing - > start + existing - > len < = start ) ) {
free_extent_map ( existing ) ;
existing = NULL ;
}
2008-04-17 19:29:12 +04:00
if ( ! existing ) {
existing = lookup_extent_mapping ( em_tree , em - > start ,
em - > len ) ;
if ( existing ) {
err = merge_extent_mapping ( em_tree , existing ,
em ) ;
free_extent_map ( existing ) ;
if ( err ) {
free_extent_map ( em ) ;
em = NULL ;
}
} else {
err = - EIO ;
printk ( " failing to insert %Lu %Lu \n " ,
start , len ) ;
free_extent_map ( em ) ;
em = NULL ;
}
} else {
free_extent_map ( em ) ;
em = existing ;
2007-08-28 00:49:44 +04:00
}
}
2008-01-25 00:13:08 +03:00
spin_unlock ( & em_tree - > lock ) ;
2007-08-28 00:49:44 +04:00
out :
btrfs_free_path ( path ) ;
if ( trans ) {
ret = btrfs_end_transaction ( trans , root ) ;
if ( ! err )
err = ret ;
}
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( err ) {
free_extent_map ( em ) ;
WARN_ON ( 1 ) ;
return ERR_PTR ( err ) ;
}
return em ;
}
2008-04-22 21:26:46 +04:00
#if 0 /* waiting for O_DIRECT reads */
2008-04-10 18:23:21 +04:00
static int btrfs_get_block ( struct inode * inode , sector_t iblock ,
struct buffer_head * bh_result , int create )
{
struct extent_map * em ;
u64 start = ( u64 ) iblock < < inode - > i_blkbits ;
struct btrfs_multi_bio * multi = NULL ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
u64 len ;
u64 logical ;
u64 map_length ;
int ret = 0 ;
em = btrfs_get_extent ( inode , NULL , 0 , start , bh_result - > b_size , 0 ) ;
if ( ! em | | IS_ERR ( em ) )
goto out ;
2008-04-22 21:26:46 +04:00
if ( em - > start > start | | em - > start + em - > len < = start ) {
2008-04-10 18:23:21 +04:00
goto out ;
2008-04-22 21:26:46 +04:00
}
2008-04-10 18:23:21 +04:00
if ( em - > block_start = = EXTENT_MAP_INLINE ) {
ret = - EINVAL ;
goto out ;
}
2008-04-22 21:26:46 +04:00
len = em - > start + em - > len - start ;
len = min_t ( u64 , len , INT_LIMIT ( typeof ( bh_result - > b_size ) ) ) ;
2008-04-10 18:23:21 +04:00
if ( em - > block_start = = EXTENT_MAP_HOLE | |
em - > block_start = = EXTENT_MAP_DELALLOC ) {
2008-04-22 21:26:46 +04:00
bh_result - > b_size = len ;
2008-04-10 18:23:21 +04:00
goto out ;
}
logical = start - em - > start ;
logical = em - > block_start + logical ;
map_length = len ;
ret = btrfs_map_block ( & root - > fs_info - > mapping_tree , READ ,
logical , & map_length , & multi , 0 ) ;
BUG_ON ( ret ) ;
bh_result - > b_blocknr = multi - > stripes [ 0 ] . physical > > inode - > i_blkbits ;
bh_result - > b_size = min ( map_length , len ) ;
2008-04-22 21:26:46 +04:00
2008-04-10 18:23:21 +04:00
bh_result - > b_bdev = multi - > stripes [ 0 ] . dev - > bdev ;
set_buffer_mapped ( bh_result ) ;
kfree ( multi ) ;
out :
free_extent_map ( em ) ;
return ret ;
}
2008-04-22 21:26:46 +04:00
# endif
2008-04-10 18:23:21 +04:00
static ssize_t btrfs_direct_IO ( int rw , struct kiocb * iocb ,
const struct iovec * iov , loff_t offset ,
unsigned long nr_segs )
{
2008-04-22 21:26:46 +04:00
return - EINVAL ;
#if 0
2008-04-10 18:23:21 +04:00
struct file * file = iocb - > ki_filp ;
struct inode * inode = file - > f_mapping - > host ;
if ( rw = = WRITE )
return - EINVAL ;
return blockdev_direct_IO ( rw , iocb , inode , inode - > i_sb - > s_bdev , iov ,
offset , nr_segs , btrfs_get_block , NULL ) ;
2008-04-22 21:26:46 +04:00
# endif
2008-04-10 18:23:21 +04:00
}
2007-09-11 04:02:30 +04:00
static sector_t btrfs_bmap ( struct address_space * mapping , sector_t iblock )
2007-06-12 14:35:45 +04:00
{
2007-09-11 04:02:30 +04:00
return extent_bmap ( mapping , iblock , btrfs_get_extent ) ;
2007-06-12 14:35:45 +04:00
}
2007-08-28 00:49:44 +04:00
int btrfs_readpage ( struct file * file , struct page * page )
2007-06-15 21:50:00 +04:00
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
tree = & BTRFS_I ( page - > mapping - > host ) - > io_tree ;
2007-08-28 00:49:44 +04:00
return extent_read_full_page ( tree , page , btrfs_get_extent ) ;
2007-06-15 21:50:00 +04:00
}
2007-12-22 00:27:21 +03:00
2007-08-28 00:49:44 +04:00
static int btrfs_writepage ( struct page * page , struct writeback_control * wbc )
2007-06-12 14:35:45 +04:00
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
2007-08-28 00:49:44 +04:00
if ( current - > flags & PF_MEMALLOC ) {
redirty_page_for_writepage ( wbc , page ) ;
unlock_page ( page ) ;
return 0 ;
}
2008-01-25 00:13:08 +03:00
tree = & BTRFS_I ( page - > mapping - > host ) - > io_tree ;
2007-08-28 00:49:44 +04:00
return extent_write_full_page ( tree , page , btrfs_get_extent , wbc ) ;
2007-06-15 21:50:00 +04:00
}
2007-11-02 02:45:34 +03:00
static int btrfs_writepages ( struct address_space * mapping ,
struct writeback_control * wbc )
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
tree = & BTRFS_I ( mapping - > host ) - > io_tree ;
2007-11-02 02:45:34 +03:00
return extent_writepages ( tree , mapping , btrfs_get_extent , wbc ) ;
}
2007-11-08 18:59:22 +03:00
static int
btrfs_readpages ( struct file * file , struct address_space * mapping ,
struct list_head * pages , unsigned nr_pages )
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
tree = & BTRFS_I ( mapping - > host ) - > io_tree ;
2007-11-08 18:59:22 +03:00
return extent_readpages ( tree , mapping , pages , nr_pages ,
btrfs_get_extent ) ;
}
2008-01-29 17:59:12 +03:00
static int btrfs_releasepage ( struct page * page , gfp_t gfp_flags )
2007-06-15 21:50:00 +04:00
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
struct extent_map_tree * map ;
2007-08-28 00:49:44 +04:00
int ret ;
2007-06-18 17:57:58 +04:00
2008-01-25 00:13:08 +03:00
tree = & BTRFS_I ( page - > mapping - > host ) - > io_tree ;
map = & BTRFS_I ( page - > mapping - > host ) - > extent_tree ;
2008-01-29 17:59:12 +03:00
ret = try_release_extent_mapping ( map , tree , page , gfp_flags ) ;
2007-08-28 00:49:44 +04:00
if ( ret = = 1 ) {
2008-04-21 16:52:50 +04:00
invalidate_extent_lru ( tree , page_offset ( page ) , PAGE_CACHE_SIZE ) ;
2007-08-28 00:49:44 +04:00
ClearPagePrivate ( page ) ;
set_page_private ( page , 0 ) ;
page_cache_release ( page ) ;
2007-06-12 14:35:45 +04:00
}
2007-08-28 00:49:44 +04:00
return ret ;
2007-06-12 14:35:45 +04:00
}
2007-08-28 00:49:44 +04:00
static void btrfs_invalidatepage ( struct page * page , unsigned long offset )
2007-06-12 14:35:45 +04:00
{
2008-01-25 00:13:08 +03:00
struct extent_io_tree * tree ;
2007-06-12 14:35:45 +04:00
2008-01-25 00:13:08 +03:00
tree = & BTRFS_I ( page - > mapping - > host ) - > io_tree ;
2007-08-28 00:49:44 +04:00
extent_invalidatepage ( tree , page , offset ) ;
btrfs_releasepage ( page , GFP_NOFS ) ;
2008-04-19 00:11:30 +04:00
if ( PagePrivate ( page ) ) {
2008-04-21 16:52:50 +04:00
invalidate_extent_lru ( tree , page_offset ( page ) , PAGE_CACHE_SIZE ) ;
2008-04-19 00:11:30 +04:00
ClearPagePrivate ( page ) ;
set_page_private ( page , 0 ) ;
page_cache_release ( page ) ;
}
2007-06-12 14:35:45 +04:00
}
2007-06-15 21:50:00 +04:00
/*
* btrfs_page_mkwrite ( ) is not allowed to change the file size as it gets
* called from a page fault handler when a page is first dirtied . Hence we must
* be careful to check for EOF conditions here . We set the page up correctly
* for a written page which means we get ENOSPC checking when writing into
* holes and correct delalloc and unwritten extent mapping on filesystems that
* support these features .
*
* We are not allowed to take the i_mutex here so we have to play games to
* protect against truncate races as the page could now be beyond EOF . Because
* vmtruncate ( ) writes the inode size before removing pages , once we have the
* page lock we can determine safely if the page is beyond EOF . If it is not
* beyond EOF , then the page is guaranteed safe against truncation until we
* unlock the page .
*/
int btrfs_page_mkwrite ( struct vm_area_struct * vma , struct page * page )
{
2007-12-19 00:15:09 +03:00
struct inode * inode = fdentry ( vma - > vm_file ) - > d_inode ;
2007-12-22 00:27:21 +03:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2007-06-15 21:50:00 +04:00
unsigned long end ;
loff_t size ;
2007-12-22 00:27:21 +03:00
int ret ;
2007-08-28 00:49:44 +04:00
u64 page_start ;
2007-06-15 21:50:00 +04:00
2007-12-22 00:27:21 +03:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
ret = btrfs_check_free_space ( root , PAGE_CACHE_SIZE , 0 ) ;
2008-01-02 18:01:11 +03:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
if ( ret )
goto out ;
ret = - EINVAL ;
2007-06-15 21:50:00 +04:00
lock_page ( page ) ;
wait_on_page_writeback ( page ) ;
size = i_size_read ( inode ) ;
2007-10-30 23:56:53 +03:00
page_start = ( u64 ) page - > index < < PAGE_CACHE_SHIFT ;
2007-08-28 00:49:44 +04:00
2007-06-15 21:50:00 +04:00
if ( ( page - > mapping ! = inode - > i_mapping ) | |
2007-08-28 00:49:44 +04:00
( page_start > size ) ) {
2007-06-15 21:50:00 +04:00
/* page got truncated out from underneath us */
goto out_unlock ;
}
/* page is wholly or partially inside EOF */
2007-08-28 00:49:44 +04:00
if ( page_start + PAGE_CACHE_SIZE > size )
2007-06-15 21:50:00 +04:00
end = size & ~ PAGE_CACHE_MASK ;
else
end = PAGE_CACHE_SIZE ;
2007-08-28 00:49:44 +04:00
ret = btrfs_cow_one_page ( inode , page , end ) ;
2007-06-15 21:50:00 +04:00
out_unlock :
unlock_page ( page ) ;
2007-12-22 00:27:21 +03:00
out :
2007-06-15 21:50:00 +04:00
return ret ;
}
2007-06-12 14:35:45 +04:00
static void btrfs_truncate ( struct inode * inode )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
int ret ;
struct btrfs_trans_handle * trans ;
2007-09-17 18:58:06 +04:00
unsigned long nr ;
2007-06-12 14:35:45 +04:00
if ( ! S_ISREG ( inode - > i_mode ) )
return ;
if ( IS_APPEND ( inode ) | | IS_IMMUTABLE ( inode ) )
return ;
btrfs_truncate_page ( inode - > i_mapping , inode - > i_size ) ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , inode ) ;
/* FIXME, add redo link to tree so we don't leak on crash */
2008-01-29 23:11:36 +03:00
ret = btrfs_truncate_in_trans ( trans , root , inode ,
BTRFS_EXTENT_DATA_KEY ) ;
2007-06-12 14:35:45 +04:00
btrfs_update_inode ( trans , root , inode ) ;
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
ret = btrfs_end_transaction ( trans , root ) ;
BUG_ON ( ret ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
}
2008-06-10 05:57:42 +04:00
/*
* Invalidate a single dcache entry at the root of the filesystem .
* Needed after creation of snapshot or subvolume .
*/
void btrfs_invalidate_dcache_root ( struct btrfs_root * root , char * name ,
int namelen )
{
struct dentry * alias , * entry ;
struct qstr qstr ;
alias = d_find_alias ( root - > fs_info - > sb - > s_root - > d_inode ) ;
if ( alias ) {
qstr . name = name ;
qstr . len = namelen ;
/* change me if btrfs ever gets a d_hash operation */
qstr . hash = full_name_hash ( qstr . name , qstr . len ) ;
entry = d_lookup ( alias , & qstr ) ;
dput ( alias ) ;
if ( entry ) {
d_invalidate ( entry ) ;
dput ( entry ) ;
}
}
}
2008-01-03 17:08:48 +03:00
static int noinline create_subvol ( struct btrfs_root * root , char * name ,
int namelen )
2007-06-12 14:35:45 +04:00
{
struct btrfs_trans_handle * trans ;
struct btrfs_key key ;
struct btrfs_root_item root_item ;
struct btrfs_inode_item * inode_item ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2008-01-08 23:46:30 +03:00
struct btrfs_root * new_root = root ;
2007-06-12 14:35:45 +04:00
struct inode * inode ;
struct inode * dir ;
int ret ;
2007-06-22 22:16:25 +04:00
int err ;
2007-06-12 14:35:45 +04:00
u64 objectid ;
u64 new_dirid = BTRFS_FIRST_FREE_OBJECTID ;
2007-09-17 18:58:06 +04:00
unsigned long nr = 1 ;
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
ret = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( ret )
goto fail_commit ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
BUG_ON ( ! trans ) ;
2007-12-11 17:25:06 +03:00
ret = btrfs_find_free_objectid ( trans , root - > fs_info - > tree_root ,
0 , & objectid ) ;
if ( ret )
goto fail ;
leaf = __btrfs_alloc_free_block ( trans , root , root - > leafsize ,
objectid , trans - > transid , 0 , 0 ,
0 , 0 ) ;
2007-10-16 00:14:19 +04:00
if ( IS_ERR ( leaf ) )
return PTR_ERR ( leaf ) ;
btrfs_set_header_nritems ( leaf , 0 ) ;
btrfs_set_header_level ( leaf , 0 ) ;
2007-10-16 00:15:53 +04:00
btrfs_set_header_bytenr ( leaf , leaf - > start ) ;
2007-10-16 00:14:19 +04:00
btrfs_set_header_generation ( leaf , trans - > transid ) ;
2007-12-11 17:25:06 +03:00
btrfs_set_header_owner ( leaf , objectid ) ;
2007-10-16 00:14:19 +04:00
write_extent_buffer ( leaf , root - > fs_info - > fsid ,
( unsigned long ) btrfs_header_fsid ( leaf ) ,
BTRFS_FSID_SIZE ) ;
btrfs_mark_buffer_dirty ( leaf ) ;
2007-06-12 14:35:45 +04:00
inode_item = & root_item . inode ;
memset ( inode_item , 0 , sizeof ( * inode_item ) ) ;
2007-10-16 00:14:19 +04:00
inode_item - > generation = cpu_to_le64 ( 1 ) ;
inode_item - > size = cpu_to_le64 ( 3 ) ;
inode_item - > nlink = cpu_to_le32 ( 1 ) ;
inode_item - > nblocks = cpu_to_le64 ( 1 ) ;
inode_item - > mode = cpu_to_le32 ( S_IFDIR | 0755 ) ;
2007-06-12 14:35:45 +04:00
2007-10-16 00:15:53 +04:00
btrfs_set_root_bytenr ( & root_item , leaf - > start ) ;
btrfs_set_root_level ( & root_item , 0 ) ;
2007-06-12 14:35:45 +04:00
btrfs_set_root_refs ( & root_item , 1 ) ;
2007-10-16 00:14:19 +04:00
btrfs_set_root_used ( & root_item , 0 ) ;
2007-06-22 22:16:25 +04:00
memset ( & root_item . drop_progress , 0 , sizeof ( root_item . drop_progress ) ) ;
root_item . drop_level = 0 ;
2007-10-16 00:14:19 +04:00
free_extent_buffer ( leaf ) ;
leaf = NULL ;
2007-06-12 14:35:45 +04:00
btrfs_set_root_dirid ( & root_item , new_dirid ) ;
key . objectid = objectid ;
key . offset = 1 ;
btrfs_set_key_type ( & key , BTRFS_ROOT_ITEM_KEY ) ;
ret = btrfs_insert_root ( trans , root - > fs_info - > tree_root , & key ,
& root_item ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto fail ;
2007-06-12 14:35:45 +04:00
/*
* insert the directory item
*/
key . offset = ( u64 ) - 1 ;
dir = root - > fs_info - > sb - > s_root - > d_inode ;
ret = btrfs_insert_dir_item ( trans , root - > fs_info - > tree_root ,
name , namelen , dir - > i_ino , & key ,
BTRFS_FT_DIR ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto fail ;
2007-06-12 14:35:45 +04:00
2007-12-12 22:38:19 +03:00
ret = btrfs_insert_inode_ref ( trans , root - > fs_info - > tree_root ,
name , namelen , objectid ,
root - > fs_info - > sb - > s_root - > d_inode - > i_ino ) ;
if ( ret )
goto fail ;
2007-06-12 14:35:45 +04:00
ret = btrfs_commit_transaction ( trans , root ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto fail_commit ;
2007-06-12 14:35:45 +04:00
2007-08-29 23:47:34 +04:00
new_root = btrfs_read_fs_root ( root - > fs_info , & key , name , namelen ) ;
2007-06-12 14:35:45 +04:00
BUG_ON ( ! new_root ) ;
trans = btrfs_start_transaction ( new_root , 1 ) ;
BUG_ON ( ! trans ) ;
2008-01-29 23:15:18 +03:00
inode = btrfs_new_inode ( trans , new_root , " .. " , 2 , new_dirid ,
new_dirid ,
2007-06-12 14:35:45 +04:00
BTRFS_I ( dir ) - > block_group , S_IFDIR | 0700 ) ;
2007-06-22 22:16:25 +04:00
if ( IS_ERR ( inode ) )
goto fail ;
2007-06-12 14:35:45 +04:00
inode - > i_op = & btrfs_dir_inode_operations ;
inode - > i_fop = & btrfs_dir_file_operations ;
2007-06-12 19:36:58 +04:00
new_root - > inode = inode ;
2007-06-12 14:35:45 +04:00
2007-12-12 22:38:19 +03:00
ret = btrfs_insert_inode_ref ( trans , new_root , " .. " , 2 , new_dirid ,
new_dirid ) ;
2007-06-12 14:35:45 +04:00
inode - > i_nlink = 1 ;
2007-12-12 22:38:19 +03:00
inode - > i_size = 0 ;
2007-06-12 14:35:45 +04:00
ret = btrfs_update_inode ( trans , new_root , inode ) ;
2007-06-22 22:16:25 +04:00
if ( ret )
goto fail ;
2008-06-10 05:57:42 +04:00
/* Invalidate existing dcache entry for new subvolume. */
btrfs_invalidate_dcache_root ( root , name , namelen ) ;
2007-06-22 22:16:25 +04:00
fail :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2008-01-08 23:46:30 +03:00
err = btrfs_commit_transaction ( trans , new_root ) ;
2007-06-22 22:16:25 +04:00
if ( err & & ! ret )
ret = err ;
fail_commit :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-22 22:16:25 +04:00
return ret ;
2007-06-12 14:35:45 +04:00
}
static int create_snapshot ( struct btrfs_root * root , char * name , int namelen )
{
2008-01-08 23:46:30 +03:00
struct btrfs_pending_snapshot * pending_snapshot ;
2007-06-12 14:35:45 +04:00
struct btrfs_trans_handle * trans ;
int ret ;
2007-06-22 22:16:25 +04:00
int err ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
if ( ! root - > ref_cows )
return - EINVAL ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
ret = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( ret )
goto fail_unlock ;
2008-01-08 23:46:30 +03:00
pending_snapshot = kmalloc ( sizeof ( * pending_snapshot ) , GFP_NOFS ) ;
if ( ! pending_snapshot ) {
ret = - ENOMEM ;
goto fail_unlock ;
}
2008-01-17 19:59:51 +03:00
pending_snapshot - > name = kmalloc ( namelen + 1 , GFP_NOFS ) ;
2008-01-08 23:46:30 +03:00
if ( ! pending_snapshot - > name ) {
ret = - ENOMEM ;
kfree ( pending_snapshot ) ;
goto fail_unlock ;
}
2008-01-17 19:59:51 +03:00
memcpy ( pending_snapshot - > name , name , namelen ) ;
pending_snapshot - > name [ namelen ] = ' \0 ' ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
BUG_ON ( ! trans ) ;
2008-01-08 23:46:30 +03:00
pending_snapshot - > root = root ;
list_add ( & pending_snapshot - > list ,
& trans - > transaction - > pending_snapshots ) ;
2007-06-12 14:35:45 +04:00
ret = btrfs_update_inode ( trans , root , root - > inode ) ;
2007-06-22 22:16:25 +04:00
err = btrfs_commit_transaction ( trans , root ) ;
2007-10-16 00:14:19 +04:00
2007-12-22 00:27:21 +03:00
fail_unlock :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-22 22:16:25 +04:00
return ret ;
2007-06-12 14:35:45 +04:00
}
2007-12-22 00:27:24 +03:00
unsigned long btrfs_force_ra ( struct address_space * mapping ,
2007-09-11 03:58:16 +04:00
struct file_ra_state * ra , struct file * file ,
pgoff_t offset , pgoff_t last_index )
{
2008-04-28 17:02:36 +04:00
pgoff_t req_size = last_index - offset + 1 ;
2007-09-11 03:58:16 +04:00
# if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,23)
offset = page_cache_readahead ( mapping , ra , file , offset , req_size ) ;
return offset ;
# else
page_cache_sync_readahead ( mapping , ra , file , offset , req_size ) ;
return offset + req_size ;
# endif
}
int btrfs_defrag_file ( struct file * file ) {
2007-12-19 00:15:09 +03:00
struct inode * inode = fdentry ( file ) - > d_inode ;
2007-12-22 00:27:21 +03:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
2008-01-25 00:13:08 +03:00
struct extent_io_tree * io_tree = & BTRFS_I ( inode ) - > io_tree ;
2007-09-11 03:58:16 +04:00
struct page * page ;
unsigned long last_index ;
2008-04-28 17:02:36 +04:00
unsigned long ra_pages = root - > fs_info - > bdi . ra_pages ;
unsigned long total_read = 0 ;
2007-09-11 03:58:16 +04:00
u64 page_start ;
u64 page_end ;
unsigned long i ;
2007-12-22 00:27:21 +03:00
int ret ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
ret = btrfs_check_free_space ( root , inode - > i_size , 0 ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( ret )
return - ENOSPC ;
2007-09-11 03:58:16 +04:00
mutex_lock ( & inode - > i_mutex ) ;
last_index = inode - > i_size > > PAGE_CACHE_SHIFT ;
for ( i = 0 ; i < = last_index ; i + + ) {
2008-04-28 17:02:36 +04:00
if ( total_read % ra_pages = = 0 ) {
btrfs_force_ra ( inode - > i_mapping , & file - > f_ra , file , i ,
min ( last_index , i + ra_pages - 1 ) ) ;
2007-09-11 03:58:16 +04:00
}
2008-04-28 17:02:36 +04:00
total_read + + ;
2007-09-11 03:58:16 +04:00
page = grab_cache_page ( inode - > i_mapping , i ) ;
if ( ! page )
goto out_unlock ;
if ( ! PageUptodate ( page ) ) {
btrfs_readpage ( NULL , page ) ;
lock_page ( page ) ;
if ( ! PageUptodate ( page ) ) {
unlock_page ( page ) ;
page_cache_release ( page ) ;
goto out_unlock ;
}
}
2008-04-28 23:29:52 +04:00
# if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,18)
ClearPageDirty ( page ) ;
# else
cancel_dirty_page ( page , PAGE_CACHE_SIZE ) ;
# endif
wait_on_page_writeback ( page ) ;
set_page_extent_mapped ( page ) ;
2007-10-30 23:56:53 +03:00
page_start = ( u64 ) page - > index < < PAGE_CACHE_SHIFT ;
2007-09-11 03:58:16 +04:00
page_end = page_start + PAGE_CACHE_SIZE - 1 ;
2008-01-25 00:13:08 +03:00
lock_extent ( io_tree , page_start , page_end , GFP_NOFS ) ;
set_extent_delalloc ( io_tree , page_start ,
2007-09-11 03:58:16 +04:00
page_end , GFP_NOFS ) ;
2007-12-22 00:27:24 +03:00
2008-01-25 00:13:08 +03:00
unlock_extent ( io_tree , page_start , page_end , GFP_NOFS ) ;
2007-09-11 03:58:16 +04:00
set_page_dirty ( page ) ;
unlock_page ( page ) ;
page_cache_release ( page ) ;
balance_dirty_pages_ratelimited_nr ( inode - > i_mapping , 1 ) ;
}
out_unlock :
mutex_unlock ( & inode - > i_mutex ) ;
return 0 ;
}
2007-12-22 00:27:24 +03:00
static int btrfs_ioctl_resize ( struct btrfs_root * root , void __user * arg )
{
u64 new_size ;
u64 old_size ;
2008-04-26 00:53:30 +04:00
u64 devid = 1 ;
2007-12-22 00:27:24 +03:00
struct btrfs_ioctl_vol_args * vol_args ;
struct btrfs_trans_handle * trans ;
2008-04-26 00:53:30 +04:00
struct btrfs_device * device = NULL ;
2007-12-22 00:27:24 +03:00
char * sizestr ;
2008-04-26 00:53:30 +04:00
char * devstr = NULL ;
2007-12-22 00:27:24 +03:00
int ret = 0 ;
int namelen ;
int mod = 0 ;
vol_args = kmalloc ( sizeof ( * vol_args ) , GFP_NOFS ) ;
if ( ! vol_args )
return - ENOMEM ;
if ( copy_from_user ( vol_args , arg , sizeof ( * vol_args ) ) ) {
ret = - EFAULT ;
goto out ;
}
namelen = strlen ( vol_args - > name ) ;
if ( namelen > BTRFS_VOL_NAME_MAX ) {
ret = - EINVAL ;
goto out ;
}
2008-04-26 00:53:30 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:24 +03:00
sizestr = vol_args - > name ;
2008-04-26 00:53:30 +04:00
devstr = strchr ( sizestr , ' : ' ) ;
if ( devstr ) {
char * end ;
sizestr = devstr + 1 ;
* devstr = ' \0 ' ;
devstr = vol_args - > name ;
devid = simple_strtoull ( devstr , & end , 10 ) ;
printk ( " resizing devid %Lu \n " , devid ) ;
}
device = btrfs_find_device ( root , devid , NULL ) ;
if ( ! device ) {
printk ( " resizer unable to find device %Lu \n " , devid ) ;
ret = - EINVAL ;
goto out_unlock ;
}
2007-12-22 00:27:24 +03:00
if ( ! strcmp ( sizestr , " max " ) )
2008-04-26 00:53:30 +04:00
new_size = device - > bdev - > bd_inode - > i_size ;
2007-12-22 00:27:24 +03:00
else {
if ( sizestr [ 0 ] = = ' - ' ) {
mod = - 1 ;
sizestr + + ;
} else if ( sizestr [ 0 ] = = ' + ' ) {
mod = 1 ;
sizestr + + ;
}
new_size = btrfs_parse_size ( sizestr ) ;
if ( new_size = = 0 ) {
ret = - EINVAL ;
2008-04-26 00:53:30 +04:00
goto out_unlock ;
2007-12-22 00:27:24 +03:00
}
}
2008-04-26 00:53:30 +04:00
old_size = device - > total_bytes ;
2007-12-22 00:27:24 +03:00
if ( mod < 0 ) {
if ( new_size > old_size ) {
ret = - EINVAL ;
goto out_unlock ;
}
new_size = old_size - new_size ;
} else if ( mod > 0 ) {
new_size = old_size + new_size ;
}
if ( new_size < 256 * 1024 * 1024 ) {
ret = - EINVAL ;
goto out_unlock ;
}
2008-04-26 00:53:30 +04:00
if ( new_size > device - > bdev - > bd_inode - > i_size ) {
2007-12-22 00:27:24 +03:00
ret = - EFBIG ;
goto out_unlock ;
}
2008-01-03 17:22:38 +03:00
do_div ( new_size , root - > sectorsize ) ;
new_size * = root - > sectorsize ;
2007-12-22 00:27:24 +03:00
2008-04-26 00:53:30 +04:00
printk ( " new size for %s is %llu \n " , device - > name , ( unsigned long long ) new_size ) ;
2007-12-22 00:27:24 +03:00
if ( new_size > old_size ) {
trans = btrfs_start_transaction ( root , 1 ) ;
2008-04-26 00:53:30 +04:00
ret = btrfs_grow_device ( trans , device , new_size ) ;
2007-12-22 00:27:24 +03:00
btrfs_commit_transaction ( trans , root ) ;
} else {
2008-04-26 00:53:30 +04:00
ret = btrfs_shrink_device ( device , new_size ) ;
2007-12-22 00:27:24 +03:00
}
out_unlock :
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
out :
kfree ( vol_args ) ;
return ret ;
}
2008-01-03 17:08:48 +03:00
static int noinline btrfs_ioctl_snap_create ( struct btrfs_root * root ,
void __user * arg )
2007-06-12 14:35:45 +04:00
{
2007-12-19 00:25:45 +03:00
struct btrfs_ioctl_vol_args * vol_args ;
2007-06-12 14:35:45 +04:00
struct btrfs_dir_item * di ;
struct btrfs_path * path ;
u64 root_dirid ;
2007-12-19 00:25:45 +03:00
int namelen ;
int ret ;
2007-06-12 14:35:45 +04:00
2007-12-19 00:25:45 +03:00
vol_args = kmalloc ( sizeof ( * vol_args ) , GFP_NOFS ) ;
2007-10-16 00:14:19 +04:00
2007-12-19 00:25:45 +03:00
if ( ! vol_args )
return - ENOMEM ;
if ( copy_from_user ( vol_args , arg , sizeof ( * vol_args ) ) ) {
ret = - EFAULT ;
goto out ;
}
namelen = strlen ( vol_args - > name ) ;
if ( namelen > BTRFS_VOL_NAME_MAX ) {
ret = - EINVAL ;
goto out ;
}
if ( strchr ( vol_args - > name , ' / ' ) ) {
ret = - EINVAL ;
goto out ;
}
2007-09-14 18:22:57 +04:00
path = btrfs_alloc_path ( ) ;
2007-12-19 00:25:45 +03:00
if ( ! path ) {
ret = - ENOMEM ;
goto out ;
}
2007-09-14 18:22:57 +04:00
root_dirid = root - > fs_info - > sb - > s_root - > d_inode - > i_ino ,
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
di = btrfs_lookup_dir_item ( NULL , root - > fs_info - > tree_root ,
path , root_dirid ,
2007-12-19 00:25:45 +03:00
vol_args - > name , namelen , 0 ) ;
2007-09-14 18:22:57 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
btrfs_free_path ( path ) ;
2007-12-19 00:25:45 +03:00
if ( di & & ! IS_ERR ( di ) ) {
ret = - EEXIST ;
goto out ;
}
if ( IS_ERR ( di ) ) {
ret = PTR_ERR ( di ) ;
goto out ;
}
2007-09-14 18:22:57 +04:00
if ( root = = root - > fs_info - > tree_root )
2007-12-19 00:25:45 +03:00
ret = create_subvol ( root , vol_args - > name , namelen ) ;
else
ret = create_snapshot ( root , vol_args - > name , namelen ) ;
out :
kfree ( vol_args ) ;
return ret ;
2007-09-14 18:22:57 +04:00
}
static int btrfs_ioctl_defrag ( struct file * file )
{
2007-12-19 00:15:09 +03:00
struct inode * inode = fdentry ( file ) - > d_inode ;
2007-09-14 18:22:57 +04:00
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
switch ( inode - > i_mode & S_IFMT ) {
case S_IFDIR :
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-09-14 18:22:57 +04:00
btrfs_defrag_root ( root , 0 ) ;
btrfs_defrag_root ( root - > fs_info - > extent_root , 0 ) ;
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
break ;
2007-09-14 18:22:57 +04:00
case S_IFREG :
btrfs_defrag_file ( file ) ;
break ;
}
return 0 ;
}
2007-08-08 00:15:09 +04:00
2008-04-28 23:29:42 +04:00
long btrfs_ioctl_add_dev ( struct btrfs_root * root , void __user * arg )
{
struct btrfs_ioctl_vol_args * vol_args ;
int ret ;
vol_args = kmalloc ( sizeof ( * vol_args ) , GFP_NOFS ) ;
if ( ! vol_args )
return - ENOMEM ;
if ( copy_from_user ( vol_args , arg , sizeof ( * vol_args ) ) ) {
ret = - EFAULT ;
goto out ;
}
ret = btrfs_init_new_device ( root , vol_args - > name ) ;
out :
kfree ( vol_args ) ;
return ret ;
}
2008-05-07 19:43:44 +04:00
long btrfs_ioctl_rm_dev ( struct btrfs_root * root , void __user * arg )
{
struct btrfs_ioctl_vol_args * vol_args ;
int ret ;
vol_args = kmalloc ( sizeof ( * vol_args ) , GFP_NOFS ) ;
if ( ! vol_args )
return - ENOMEM ;
if ( copy_from_user ( vol_args , arg , sizeof ( * vol_args ) ) ) {
ret = - EFAULT ;
goto out ;
}
ret = btrfs_rm_device ( root , vol_args - > name ) ;
out :
kfree ( vol_args ) ;
return ret ;
}
2008-05-05 14:26:21 +04:00
int dup_item_to_inode ( struct btrfs_trans_handle * trans ,
2008-05-02 22:43:14 +04:00
struct btrfs_root * root ,
struct btrfs_path * path ,
struct extent_buffer * leaf ,
int slot ,
struct btrfs_key * key ,
u64 destino )
{
2008-05-05 14:26:21 +04:00
char * dup ;
2008-05-02 22:43:14 +04:00
int len = btrfs_item_size_nr ( leaf , slot ) ;
struct btrfs_key ckey = * key ;
2008-05-05 14:26:21 +04:00
int ret = 0 ;
dup = kmalloc ( len , GFP_NOFS ) ;
if ( ! dup )
return - ENOMEM ;
read_extent_buffer ( leaf , dup , btrfs_item_ptr_offset ( leaf , slot ) , len ) ;
btrfs_release_path ( root , path ) ;
2008-05-02 22:43:14 +04:00
ckey . objectid = destino ;
2008-05-05 14:26:21 +04:00
ret = btrfs_insert_item ( trans , root , & ckey , dup , len ) ;
kfree ( dup ) ;
return ret ;
2008-05-02 22:43:14 +04:00
}
long btrfs_ioctl_clone ( struct file * file , unsigned long src_fd )
{
struct inode * inode = fdentry ( file ) - > d_inode ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct file * src_file ;
struct inode * src ;
struct btrfs_trans_handle * trans ;
int ret ;
u64 pos ;
struct btrfs_path * path ;
struct btrfs_key key ;
struct extent_buffer * leaf ;
u32 nritems ;
int slot ;
src_file = fget ( src_fd ) ;
if ( ! src_file )
return - EBADF ;
src = src_file - > f_dentry - > d_inode ;
ret = - EXDEV ;
if ( src - > i_sb ! = inode - > i_sb )
goto out_fput ;
if ( inode < src ) {
mutex_lock ( & inode - > i_mutex ) ;
mutex_lock ( & src - > i_mutex ) ;
} else {
mutex_lock ( & src - > i_mutex ) ;
mutex_lock ( & inode - > i_mutex ) ;
}
ret = - ENOTEMPTY ;
if ( inode - > i_size )
goto out_unlock ;
/* do any pending delalloc/csum calc on src, one way or
another , and lock file content */
while ( 1 ) {
filemap_write_and_wait ( src - > i_mapping ) ;
lock_extent ( & BTRFS_I ( src ) - > io_tree , 0 , ( u64 ) - 1 , GFP_NOFS ) ;
if ( BTRFS_I ( src ) - > delalloc_bytes = = 0 )
break ;
unlock_extent ( & BTRFS_I ( src ) - > io_tree , 0 , ( u64 ) - 1 , GFP_NOFS ) ;
}
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = btrfs_start_transaction ( root , 0 ) ;
path = btrfs_alloc_path ( ) ;
2008-05-05 14:26:21 +04:00
if ( ! path ) {
ret = - ENOMEM ;
goto out ;
}
key . offset = 0 ;
key . type = BTRFS_EXTENT_DATA_KEY ;
key . objectid = src - > i_ino ;
2008-05-02 22:43:14 +04:00
pos = 0 ;
2008-05-05 14:26:21 +04:00
path - > reada = 2 ;
2008-05-02 22:43:14 +04:00
while ( 1 ) {
2008-05-05 14:26:21 +04:00
/*
* note the key will change type as we walk through the
* tree .
*/
ret = btrfs_search_slot ( trans , root , & key , path , 0 , 0 ) ;
2008-05-02 22:43:14 +04:00
if ( ret < 0 )
goto out ;
2008-05-05 14:26:21 +04:00
if ( path - > slots [ 0 ] > = btrfs_header_nritems ( path - > nodes [ 0 ] ) ) {
ret = btrfs_next_leaf ( root , path ) ;
if ( ret < 0 )
2008-05-02 22:43:14 +04:00
goto out ;
2008-05-05 14:26:21 +04:00
if ( ret > 0 )
break ;
2008-05-02 22:43:14 +04:00
}
leaf = path - > nodes [ 0 ] ;
slot = path - > slots [ 0 ] ;
btrfs_item_key_to_cpu ( leaf , & key , slot ) ;
nritems = btrfs_header_nritems ( leaf ) ;
if ( btrfs_key_type ( & key ) > BTRFS_CSUM_ITEM_KEY | |
key . objectid ! = src - > i_ino )
2008-05-05 14:26:21 +04:00
break ;
2008-05-02 22:43:14 +04:00
if ( btrfs_key_type ( & key ) = = BTRFS_EXTENT_DATA_KEY ) {
struct btrfs_file_extent_item * extent ;
int found_type ;
pos = key . offset ;
extent = btrfs_item_ptr ( leaf , slot ,
struct btrfs_file_extent_item ) ;
found_type = btrfs_file_extent_type ( leaf , extent ) ;
if ( found_type = = BTRFS_FILE_EXTENT_REG ) {
u64 len = btrfs_file_extent_num_bytes ( leaf ,
extent ) ;
u64 ds = btrfs_file_extent_disk_bytenr ( leaf ,
extent ) ;
u64 dl = btrfs_file_extent_disk_num_bytes ( leaf ,
extent ) ;
u64 off = btrfs_file_extent_offset ( leaf ,
extent ) ;
btrfs_insert_file_extent ( trans , root ,
inode - > i_ino , pos ,
ds , dl , len , off ) ;
/* ds == 0 means there's a hole */
if ( ds ! = 0 ) {
btrfs_inc_extent_ref ( trans , root ,
ds , dl ,
root - > root_key . objectid ,
trans - > transid ,
inode - > i_ino , pos ) ;
}
pos = key . offset + len ;
} else if ( found_type = = BTRFS_FILE_EXTENT_INLINE ) {
2008-05-05 14:26:21 +04:00
ret = dup_item_to_inode ( trans , root , path ,
leaf , slot , & key ,
inode - > i_ino ) ;
if ( ret )
goto out ;
2008-05-02 22:43:14 +04:00
pos = key . offset + btrfs_item_size_nr ( leaf ,
slot ) ;
}
2008-05-05 14:26:21 +04:00
} else if ( btrfs_key_type ( & key ) = = BTRFS_CSUM_ITEM_KEY ) {
ret = dup_item_to_inode ( trans , root , path , leaf ,
slot , & key , inode - > i_ino ) ;
2008-05-02 22:43:14 +04:00
2008-05-05 14:26:21 +04:00
if ( ret )
2008-05-02 22:43:14 +04:00
goto out ;
}
2008-05-05 14:26:21 +04:00
key . offset + + ;
btrfs_release_path ( root , path ) ;
2008-05-02 22:43:14 +04:00
}
2008-05-05 14:26:21 +04:00
ret = 0 ;
2008-05-02 22:43:14 +04:00
out :
btrfs_free_path ( path ) ;
inode - > i_blocks = src - > i_blocks ;
i_size_write ( inode , src - > i_size ) ;
btrfs_update_inode ( trans , root , inode ) ;
unlock_extent ( & BTRFS_I ( src ) - > io_tree , 0 , ( u64 ) - 1 , GFP_NOFS ) ;
btrfs_end_transaction ( trans , root ) ;
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
out_unlock :
mutex_unlock ( & src - > i_mutex ) ;
mutex_unlock ( & inode - > i_mutex ) ;
out_fput :
fput ( src_file ) ;
return ret ;
}
2008-06-10 18:07:39 +04:00
/*
* there are many ways the trans_start and trans_end ioctls can lead
* to deadlocks . They should only be used by applications that
* basically own the machine , and have a very in depth understanding
* of all the possible deadlocks and enospc problems .
*/
long btrfs_ioctl_trans_start ( struct file * file )
{
struct inode * inode = fdentry ( file ) - > d_inode ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
int ret = 0 ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
if ( file - > private_data ) {
ret = - EINPROGRESS ;
goto out ;
}
trans = btrfs_start_transaction ( root , 0 ) ;
if ( trans )
file - > private_data = trans ;
else
ret = - ENOMEM ;
/*printk(KERN_INFO "btrfs_ioctl_trans_start on %p\n", file);*/
out :
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
/*
* there are many ways the trans_start and trans_end ioctls can lead
* to deadlocks . They should only be used by applications that
* basically own the machine , and have a very in depth understanding
* of all the possible deadlocks and enospc problems .
*/
long btrfs_ioctl_trans_end ( struct file * file )
{
struct inode * inode = fdentry ( file ) - > d_inode ;
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
struct btrfs_trans_handle * trans ;
int ret = 0 ;
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
trans = file - > private_data ;
if ( ! trans ) {
ret = - EINVAL ;
goto out ;
}
btrfs_end_transaction ( trans , root ) ;
file - > private_data = 0 ;
out :
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
2007-09-14 18:22:57 +04:00
long btrfs_ioctl ( struct file * file , unsigned int
cmd , unsigned long arg )
{
2007-12-19 00:15:09 +03:00
struct btrfs_root * root = BTRFS_I ( fdentry ( file ) - > d_inode ) - > root ;
2007-09-14 18:22:57 +04:00
switch ( cmd ) {
case BTRFS_IOC_SNAP_CREATE :
return btrfs_ioctl_snap_create ( root , ( void __user * ) arg ) ;
2007-08-08 00:15:09 +04:00
case BTRFS_IOC_DEFRAG :
2007-09-14 18:22:57 +04:00
return btrfs_ioctl_defrag ( file ) ;
2007-12-22 00:27:24 +03:00
case BTRFS_IOC_RESIZE :
return btrfs_ioctl_resize ( root , ( void __user * ) arg ) ;
2008-04-28 23:29:42 +04:00
case BTRFS_IOC_ADD_DEV :
return btrfs_ioctl_add_dev ( root , ( void __user * ) arg ) ;
2008-05-07 19:43:44 +04:00
case BTRFS_IOC_RM_DEV :
return btrfs_ioctl_rm_dev ( root , ( void __user * ) arg ) ;
2008-04-28 23:29:52 +04:00
case BTRFS_IOC_BALANCE :
return btrfs_balance ( root - > fs_info - > dev_root ) ;
2008-05-02 22:43:14 +04:00
case BTRFS_IOC_CLONE :
return btrfs_ioctl_clone ( file , arg ) ;
2008-06-10 18:07:39 +04:00
case BTRFS_IOC_TRANS_START :
return btrfs_ioctl_trans_start ( file ) ;
case BTRFS_IOC_TRANS_END :
return btrfs_ioctl_trans_end ( file ) ;
case BTRFS_IOC_SYNC :
btrfs_sync_fs ( file - > f_dentry - > d_sb , 1 ) ;
return 0 ;
2007-06-12 14:35:45 +04:00
}
2007-09-14 18:22:57 +04:00
return - ENOTTY ;
2007-06-12 14:35:45 +04:00
}
/*
* Called inside transaction , so use GFP_NOFS
*/
struct inode * btrfs_alloc_inode ( struct super_block * sb )
{
struct btrfs_inode * ei ;
ei = kmem_cache_alloc ( btrfs_inode_cachep , GFP_NOFS ) ;
if ( ! ei )
return NULL ;
2007-08-11 00:22:09 +04:00
ei - > last_trans = 0 ;
2008-01-08 23:46:30 +03:00
ei - > ordered_trans = 0 ;
2007-06-12 14:35:45 +04:00
return & ei - > vfs_inode ;
}
void btrfs_destroy_inode ( struct inode * inode )
{
WARN_ON ( ! list_empty ( & inode - > i_dentry ) ) ;
WARN_ON ( inode - > i_data . nrpages ) ;
2008-01-14 23:10:26 +03:00
btrfs_drop_extent_cache ( inode , 0 , ( u64 ) - 1 ) ;
2007-06-12 14:35:45 +04:00
kmem_cache_free ( btrfs_inode_cachep , BTRFS_I ( inode ) ) ;
}
2007-10-29 17:55:05 +03:00
# if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,23)
static void init_once ( struct kmem_cache * cachep , void * foo )
# else
2007-06-12 14:35:45 +04:00
static void init_once ( void * foo , struct kmem_cache * cachep ,
unsigned long flags )
2007-10-29 17:55:05 +03:00
# endif
2007-06-12 14:35:45 +04:00
{
struct btrfs_inode * ei = ( struct btrfs_inode * ) foo ;
inode_init_once ( & ei - > vfs_inode ) ;
}
void btrfs_destroy_cachep ( void )
{
if ( btrfs_inode_cachep )
kmem_cache_destroy ( btrfs_inode_cachep ) ;
if ( btrfs_trans_handle_cachep )
kmem_cache_destroy ( btrfs_trans_handle_cachep ) ;
if ( btrfs_transaction_cachep )
kmem_cache_destroy ( btrfs_transaction_cachep ) ;
if ( btrfs_bit_radix_cachep )
kmem_cache_destroy ( btrfs_bit_radix_cachep ) ;
if ( btrfs_path_cachep )
kmem_cache_destroy ( btrfs_path_cachep ) ;
}
2007-09-11 03:58:16 +04:00
struct kmem_cache * btrfs_cache_create ( const char * name , size_t size ,
2007-07-25 20:31:35 +04:00
unsigned long extra_flags ,
2007-10-29 17:55:05 +03:00
# if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,23)
void ( * ctor ) ( struct kmem_cache * , void * )
# else
2007-07-25 20:31:35 +04:00
void ( * ctor ) ( void * , struct kmem_cache * ,
2007-10-29 17:55:05 +03:00
unsigned long )
# endif
)
2007-07-25 20:31:35 +04:00
{
return kmem_cache_create ( name , size , 0 , ( SLAB_RECLAIM_ACCOUNT |
SLAB_MEM_SPREAD | extra_flags ) , ctor
# if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,23)
, NULL
# endif
) ;
}
2007-06-12 14:35:45 +04:00
int btrfs_init_cachep ( void )
{
2007-09-11 03:58:16 +04:00
btrfs_inode_cachep = btrfs_cache_create ( " btrfs_inode_cache " ,
2007-07-25 20:31:35 +04:00
sizeof ( struct btrfs_inode ) ,
0 , init_once ) ;
2007-06-12 14:35:45 +04:00
if ( ! btrfs_inode_cachep )
goto fail ;
2007-09-11 03:58:16 +04:00
btrfs_trans_handle_cachep =
btrfs_cache_create ( " btrfs_trans_handle_cache " ,
sizeof ( struct btrfs_trans_handle ) ,
0 , NULL ) ;
2007-06-12 14:35:45 +04:00
if ( ! btrfs_trans_handle_cachep )
goto fail ;
2007-09-11 03:58:16 +04:00
btrfs_transaction_cachep = btrfs_cache_create ( " btrfs_transaction_cache " ,
2007-06-12 14:35:45 +04:00
sizeof ( struct btrfs_transaction ) ,
2007-07-25 20:31:35 +04:00
0 , NULL ) ;
2007-06-12 14:35:45 +04:00
if ( ! btrfs_transaction_cachep )
goto fail ;
2007-09-11 03:58:16 +04:00
btrfs_path_cachep = btrfs_cache_create ( " btrfs_path_cache " ,
2007-09-17 19:08:52 +04:00
sizeof ( struct btrfs_path ) ,
2007-07-25 20:31:35 +04:00
0 , NULL ) ;
2007-06-12 14:35:45 +04:00
if ( ! btrfs_path_cachep )
goto fail ;
2007-09-11 03:58:16 +04:00
btrfs_bit_radix_cachep = btrfs_cache_create ( " btrfs_radix " , 256 ,
2007-07-25 20:31:35 +04:00
SLAB_DESTROY_BY_RCU , NULL ) ;
2007-06-12 14:35:45 +04:00
if ( ! btrfs_bit_radix_cachep )
goto fail ;
return 0 ;
fail :
btrfs_destroy_cachep ( ) ;
return - ENOMEM ;
}
static int btrfs_getattr ( struct vfsmount * mnt ,
struct dentry * dentry , struct kstat * stat )
{
struct inode * inode = dentry - > d_inode ;
generic_fillattr ( inode , stat ) ;
2008-01-03 22:51:00 +03:00
stat - > blksize = PAGE_CACHE_SIZE ;
2008-02-08 21:49:28 +03:00
stat - > blocks = inode - > i_blocks + ( BTRFS_I ( inode ) - > delalloc_bytes > > 9 ) ;
2007-06-12 14:35:45 +04:00
return 0 ;
}
static int btrfs_rename ( struct inode * old_dir , struct dentry * old_dentry ,
struct inode * new_dir , struct dentry * new_dentry )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( old_dir ) - > root ;
struct inode * new_inode = new_dentry - > d_inode ;
struct inode * old_inode = old_dentry - > d_inode ;
struct timespec ctime = CURRENT_TIME ;
struct btrfs_path * path ;
int ret ;
if ( S_ISDIR ( old_inode - > i_mode ) & & new_inode & &
new_inode - > i_size > BTRFS_EMPTY_DIR_SIZE ) {
return - ENOTEMPTY ;
}
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
ret = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( ret )
goto out_unlock ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
btrfs_set_trans_block_group ( trans , new_dir ) ;
path = btrfs_alloc_path ( ) ;
if ( ! path ) {
ret = - ENOMEM ;
goto out_fail ;
}
old_dentry - > d_inode - > i_nlink + + ;
old_dir - > i_ctime = old_dir - > i_mtime = ctime ;
new_dir - > i_ctime = new_dir - > i_mtime = ctime ;
old_inode - > i_ctime = ctime ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
ret = btrfs_unlink_trans ( trans , root , old_dir , old_dentry ) ;
if ( ret )
goto out_fail ;
if ( new_inode ) {
new_inode - > i_ctime = CURRENT_TIME ;
ret = btrfs_unlink_trans ( trans , root , new_dir , new_dentry ) ;
if ( ret )
goto out_fail ;
}
2008-01-29 23:15:18 +03:00
ret = btrfs_add_link ( trans , new_dentry , old_inode , 1 ) ;
2007-06-12 14:35:45 +04:00
if ( ret )
goto out_fail ;
out_fail :
btrfs_free_path ( path ) ;
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
out_unlock :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
return ret ;
}
static int btrfs_symlink ( struct inode * dir , struct dentry * dentry ,
const char * symname )
{
struct btrfs_trans_handle * trans ;
struct btrfs_root * root = BTRFS_I ( dir ) - > root ;
struct btrfs_path * path ;
struct btrfs_key key ;
2007-12-22 00:27:21 +03:00
struct inode * inode = NULL ;
2007-06-12 14:35:45 +04:00
int err ;
int drop_inode = 0 ;
u64 objectid ;
int name_len ;
int datasize ;
2007-10-16 00:14:19 +04:00
unsigned long ptr ;
2007-06-12 14:35:45 +04:00
struct btrfs_file_extent_item * ei ;
2007-10-16 00:14:19 +04:00
struct extent_buffer * leaf ;
2007-12-22 00:27:21 +03:00
unsigned long nr = 0 ;
2007-06-12 14:35:45 +04:00
name_len = strlen ( symname ) + 1 ;
if ( name_len > BTRFS_MAX_INLINE_DATA_SIZE ( root ) )
return - ENAMETOOLONG ;
2007-12-22 00:27:21 +03:00
2007-06-12 14:35:45 +04:00
mutex_lock ( & root - > fs_info - > fs_mutex ) ;
2007-12-22 00:27:21 +03:00
err = btrfs_check_free_space ( root , 1 , 0 ) ;
if ( err )
goto out_fail ;
2007-06-12 14:35:45 +04:00
trans = btrfs_start_transaction ( root , 1 ) ;
btrfs_set_trans_block_group ( trans , dir ) ;
err = btrfs_find_free_objectid ( trans , root , dir - > i_ino , & objectid ) ;
if ( err ) {
err = - ENOSPC ;
goto out_unlock ;
}
2008-01-29 23:15:18 +03:00
inode = btrfs_new_inode ( trans , root , dentry - > d_name . name ,
dentry - > d_name . len ,
dentry - > d_parent - > d_inode - > i_ino , objectid ,
2007-06-12 14:35:45 +04:00
BTRFS_I ( dir ) - > block_group , S_IFLNK | S_IRWXUGO ) ;
err = PTR_ERR ( inode ) ;
if ( IS_ERR ( inode ) )
goto out_unlock ;
btrfs_set_trans_block_group ( trans , inode ) ;
2008-01-29 23:15:18 +03:00
err = btrfs_add_nondir ( trans , dentry , inode , 0 ) ;
2007-06-12 14:35:45 +04:00
if ( err )
drop_inode = 1 ;
else {
inode - > i_mapping - > a_ops = & btrfs_aops ;
2008-03-26 17:28:07 +03:00
inode - > i_mapping - > backing_dev_info = & root - > fs_info - > bdi ;
2007-06-12 14:35:45 +04:00
inode - > i_fop = & btrfs_file_operations ;
inode - > i_op = & btrfs_file_inode_operations ;
2008-01-25 00:13:08 +03:00
extent_map_tree_init ( & BTRFS_I ( inode ) - > extent_tree , GFP_NOFS ) ;
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_tree ,
2007-08-28 00:49:44 +04:00
inode - > i_mapping , GFP_NOFS ) ;
2008-04-10 00:28:12 +04:00
extent_io_tree_init ( & BTRFS_I ( inode ) - > io_failure_tree ,
inode - > i_mapping , GFP_NOFS ) ;
2008-02-08 21:49:28 +03:00
BTRFS_I ( inode ) - > delalloc_bytes = 0 ;
2008-04-25 16:51:48 +04:00
atomic_set ( & BTRFS_I ( inode ) - > ordered_writeback , 0 ) ;
2008-01-25 00:13:08 +03:00
BTRFS_I ( inode ) - > io_tree . ops = & btrfs_extent_io_ops ;
2007-06-12 14:35:45 +04:00
}
dir - > i_sb - > s_dirt = 1 ;
btrfs_update_inode_block_group ( trans , inode ) ;
btrfs_update_inode_block_group ( trans , dir ) ;
if ( drop_inode )
goto out_unlock ;
path = btrfs_alloc_path ( ) ;
BUG_ON ( ! path ) ;
key . objectid = inode - > i_ino ;
key . offset = 0 ;
btrfs_set_key_type ( & key , BTRFS_EXTENT_DATA_KEY ) ;
datasize = btrfs_file_extent_calc_inline_size ( name_len ) ;
err = btrfs_insert_empty_item ( trans , root , path , & key ,
datasize ) ;
2007-06-22 22:16:25 +04:00
if ( err ) {
drop_inode = 1 ;
goto out_unlock ;
}
2007-10-16 00:14:19 +04:00
leaf = path - > nodes [ 0 ] ;
ei = btrfs_item_ptr ( leaf , path - > slots [ 0 ] ,
struct btrfs_file_extent_item ) ;
btrfs_set_file_extent_generation ( leaf , ei , trans - > transid ) ;
btrfs_set_file_extent_type ( leaf , ei ,
2007-06-12 14:35:45 +04:00
BTRFS_FILE_EXTENT_INLINE ) ;
ptr = btrfs_file_extent_inline_start ( ei ) ;
2007-10-16 00:14:19 +04:00
write_extent_buffer ( leaf , symname , ptr , name_len ) ;
btrfs_mark_buffer_dirty ( leaf ) ;
2007-06-12 14:35:45 +04:00
btrfs_free_path ( path ) ;
2007-10-16 00:14:19 +04:00
2007-06-12 14:35:45 +04:00
inode - > i_op = & btrfs_symlink_inode_operations ;
inode - > i_mapping - > a_ops = & btrfs_symlink_aops ;
2008-03-26 17:28:07 +03:00
inode - > i_mapping - > backing_dev_info = & root - > fs_info - > bdi ;
2007-06-12 14:35:45 +04:00
inode - > i_size = name_len - 1 ;
2007-06-22 22:16:25 +04:00
err = btrfs_update_inode ( trans , root , inode ) ;
if ( err )
drop_inode = 1 ;
2007-06-12 14:35:45 +04:00
out_unlock :
2007-09-17 18:58:06 +04:00
nr = trans - > blocks_used ;
2007-06-12 14:35:45 +04:00
btrfs_end_transaction ( trans , root ) ;
2007-12-22 00:27:21 +03:00
out_fail :
2007-06-12 14:35:45 +04:00
mutex_unlock ( & root - > fs_info - > fs_mutex ) ;
if ( drop_inode ) {
inode_dec_link_count ( inode ) ;
iput ( inode ) ;
}
2007-09-17 18:58:06 +04:00
btrfs_btree_balance_dirty ( root , nr ) ;
2008-01-08 23:46:30 +03:00
btrfs_throttle ( root ) ;
2007-06-12 14:35:45 +04:00
return err ;
}
2008-04-10 18:23:21 +04:00
2008-01-14 21:26:08 +03:00
static int btrfs_permission ( struct inode * inode , int mask ,
struct nameidata * nd )
{
if ( btrfs_test_flag ( inode , READONLY ) & & ( mask & MAY_WRITE ) )
return - EACCES ;
return generic_permission ( inode , mask , NULL ) ;
}
2007-06-12 14:35:45 +04:00
static struct inode_operations btrfs_dir_inode_operations = {
. lookup = btrfs_lookup ,
. create = btrfs_create ,
. unlink = btrfs_unlink ,
. link = btrfs_link ,
. mkdir = btrfs_mkdir ,
. rmdir = btrfs_rmdir ,
. rename = btrfs_rename ,
. symlink = btrfs_symlink ,
. setattr = btrfs_setattr ,
2007-07-11 18:18:17 +04:00
. mknod = btrfs_mknod ,
2007-11-16 19:45:54 +03:00
. setxattr = generic_setxattr ,
. getxattr = generic_getxattr ,
. listxattr = btrfs_listxattr ,
. removexattr = generic_removexattr ,
2008-01-14 21:26:08 +03:00
. permission = btrfs_permission ,
2007-06-12 14:35:45 +04:00
} ;
static struct inode_operations btrfs_dir_ro_inode_operations = {
. lookup = btrfs_lookup ,
2008-01-14 21:26:08 +03:00
. permission = btrfs_permission ,
2007-06-12 14:35:45 +04:00
} ;
static struct file_operations btrfs_dir_file_operations = {
. llseek = generic_file_llseek ,
. read = generic_read_dir ,
. readdir = btrfs_readdir ,
2007-09-14 18:22:47 +04:00
. unlocked_ioctl = btrfs_ioctl ,
2007-06-12 14:35:45 +04:00
# ifdef CONFIG_COMPAT
2007-09-14 18:22:47 +04:00
. compat_ioctl = btrfs_ioctl ,
2007-06-12 14:35:45 +04:00
# endif
2008-06-10 18:07:39 +04:00
. release = btrfs_release_file ,
2007-06-12 14:35:45 +04:00
} ;
2008-01-25 00:13:08 +03:00
static struct extent_io_ops btrfs_extent_io_ops = {
2007-08-30 16:50:51 +04:00
. fill_delalloc = run_delalloc_range ,
2008-02-20 20:07:25 +03:00
. submit_bio_hook = btrfs_submit_bio_hook ,
2008-03-24 22:02:07 +03:00
. merge_bio_hook = btrfs_merge_bio_hook ,
2007-08-30 16:50:51 +04:00
. readpage_io_hook = btrfs_readpage_io_hook ,
. readpage_end_io_hook = btrfs_readpage_end_io_hook ,
2008-05-12 21:39:03 +04:00
. readpage_io_failed_hook = btrfs_io_failed_hook ,
2008-01-31 19:05:37 +03:00
. set_bit_hook = btrfs_set_bit_hook ,
. clear_bit_hook = btrfs_clear_bit_hook ,
2007-08-30 16:50:51 +04:00
} ;
2007-06-12 14:35:45 +04:00
static struct address_space_operations btrfs_aops = {
. readpage = btrfs_readpage ,
. writepage = btrfs_writepage ,
2007-11-02 02:45:34 +03:00
. writepages = btrfs_writepages ,
2007-11-08 18:59:22 +03:00
. readpages = btrfs_readpages ,
2007-06-12 14:35:45 +04:00
. sync_page = block_sync_page ,
. bmap = btrfs_bmap ,
2008-04-10 18:23:21 +04:00
. direct_IO = btrfs_direct_IO ,
2007-08-28 00:49:44 +04:00
. invalidatepage = btrfs_invalidatepage ,
. releasepage = btrfs_releasepage ,
. set_page_dirty = __set_page_dirty_nobuffers ,
2007-06-12 14:35:45 +04:00
} ;
static struct address_space_operations btrfs_symlink_aops = {
. readpage = btrfs_readpage ,
. writepage = btrfs_writepage ,
2007-08-30 19:54:02 +04:00
. invalidatepage = btrfs_invalidatepage ,
. releasepage = btrfs_releasepage ,
2007-06-12 14:35:45 +04:00
} ;
static struct inode_operations btrfs_file_inode_operations = {
. truncate = btrfs_truncate ,
. getattr = btrfs_getattr ,
. setattr = btrfs_setattr ,
2007-11-16 19:45:54 +03:00
. setxattr = generic_setxattr ,
. getxattr = generic_getxattr ,
. listxattr = btrfs_listxattr ,
. removexattr = generic_removexattr ,
2008-01-14 21:26:08 +03:00
. permission = btrfs_permission ,
2007-06-12 14:35:45 +04:00
} ;
2007-07-11 18:18:17 +04:00
static struct inode_operations btrfs_special_inode_operations = {
. getattr = btrfs_getattr ,
. setattr = btrfs_setattr ,
2008-01-14 21:26:08 +03:00
. permission = btrfs_permission ,
2007-07-11 18:18:17 +04:00
} ;
2007-06-12 14:35:45 +04:00
static struct inode_operations btrfs_symlink_inode_operations = {
. readlink = generic_readlink ,
. follow_link = page_follow_link_light ,
. put_link = page_put_link ,
2008-01-14 21:26:08 +03:00
. permission = btrfs_permission ,
2007-06-12 14:35:45 +04:00
} ;