2008-01-08 15:46:30 -05: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 .
*/
# include <linux/gfp.h>
# include <linux/slab.h>
# include "ctree.h"
# include "transaction.h"
# include "btrfs_inode.h"
struct tree_entry {
u64 root_objectid ;
u64 objectid ;
2008-01-16 16:09:22 -05:00
struct inode * inode ;
2008-01-08 15:46:30 -05:00
struct rb_node rb_node ;
} ;
/*
* returns > 0 if entry passed ( root , objectid ) is > entry ,
* < 0 if ( root , objectid ) < entry and zero if they are equal
*/
static int comp_entry ( struct tree_entry * entry , u64 root_objectid ,
u64 objectid )
{
if ( root_objectid < entry - > root_objectid )
return - 1 ;
if ( root_objectid > entry - > root_objectid )
return 1 ;
if ( objectid < entry - > objectid )
return - 1 ;
if ( objectid > entry - > objectid )
return 1 ;
return 0 ;
}
static struct rb_node * tree_insert ( struct rb_root * root , u64 root_objectid ,
u64 objectid , struct rb_node * node )
{
struct rb_node * * p = & root - > rb_node ;
struct rb_node * parent = NULL ;
struct tree_entry * entry ;
int comp ;
while ( * p ) {
parent = * p ;
entry = rb_entry ( parent , struct tree_entry , rb_node ) ;
comp = comp_entry ( entry , root_objectid , objectid ) ;
if ( comp < 0 )
p = & ( * p ) - > rb_left ;
else if ( comp > 0 )
p = & ( * p ) - > rb_right ;
else
return parent ;
}
rb_link_node ( node , parent , p ) ;
rb_insert_color ( node , root ) ;
return NULL ;
}
static struct rb_node * __tree_search ( struct rb_root * root , u64 root_objectid ,
u64 objectid , struct rb_node * * prev_ret )
{
struct rb_node * n = root - > rb_node ;
struct rb_node * prev = NULL ;
struct tree_entry * entry ;
struct tree_entry * prev_entry = NULL ;
int comp ;
while ( n ) {
entry = rb_entry ( n , struct tree_entry , rb_node ) ;
prev = n ;
prev_entry = entry ;
comp = comp_entry ( entry , root_objectid , objectid ) ;
if ( comp < 0 )
n = n - > rb_left ;
else if ( comp > 0 )
n = n - > rb_right ;
else
return n ;
}
if ( ! prev_ret )
return NULL ;
while ( prev & & comp_entry ( prev_entry , root_objectid , objectid ) > = 0 ) {
prev = rb_next ( prev ) ;
prev_entry = rb_entry ( prev , struct tree_entry , rb_node ) ;
}
* prev_ret = prev ;
return NULL ;
}
static inline struct rb_node * tree_search ( struct rb_root * root ,
u64 root_objectid , u64 objectid )
{
struct rb_node * prev ;
struct rb_node * ret ;
ret = __tree_search ( root , root_objectid , objectid , & prev ) ;
if ( ! ret )
return prev ;
return ret ;
}
int btrfs_add_ordered_inode ( struct inode * inode )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
u64 root_objectid = root - > root_key . objectid ;
u64 transid = root - > fs_info - > running_transaction - > transid ;
struct tree_entry * entry ;
struct rb_node * node ;
struct btrfs_ordered_inode_tree * tree ;
if ( transid < = BTRFS_I ( inode ) - > ordered_trans )
return 0 ;
tree = & root - > fs_info - > running_transaction - > ordered_inode_tree ;
read_lock ( & tree - > lock ) ;
node = __tree_search ( & tree - > tree , root_objectid , inode - > i_ino , NULL ) ;
read_unlock ( & tree - > lock ) ;
if ( node ) {
return 0 ;
}
entry = kmalloc ( sizeof ( * entry ) , GFP_NOFS ) ;
if ( ! entry )
return - ENOMEM ;
write_lock ( & tree - > lock ) ;
entry - > objectid = inode - > i_ino ;
entry - > root_objectid = root_objectid ;
2008-01-16 16:09:22 -05:00
entry - > inode = inode ;
2008-01-08 15:46:30 -05:00
node = tree_insert ( & tree - > tree , root_objectid ,
inode - > i_ino , & entry - > rb_node ) ;
BTRFS_I ( inode ) - > ordered_trans = transid ;
write_unlock ( & tree - > lock ) ;
if ( node )
kfree ( entry ) ;
2008-01-16 11:44:43 -05:00
else
igrab ( inode ) ;
2008-01-08 15:46:30 -05:00
return 0 ;
}
int btrfs_find_first_ordered_inode ( struct btrfs_ordered_inode_tree * tree ,
2008-01-16 16:09:22 -05:00
u64 * root_objectid , u64 * objectid ,
struct inode * * inode )
2008-01-08 15:46:30 -05:00
{
struct tree_entry * entry ;
struct rb_node * node ;
write_lock ( & tree - > lock ) ;
node = tree_search ( & tree - > tree , * root_objectid , * objectid ) ;
if ( ! node ) {
write_unlock ( & tree - > lock ) ;
return 0 ;
}
entry = rb_entry ( node , struct tree_entry , rb_node ) ;
while ( comp_entry ( entry , * root_objectid , * objectid ) > = 0 ) {
node = rb_next ( node ) ;
if ( ! node )
break ;
entry = rb_entry ( node , struct tree_entry , rb_node ) ;
}
if ( ! node ) {
write_unlock ( & tree - > lock ) ;
return 0 ;
}
* root_objectid = entry - > root_objectid ;
2008-01-16 16:09:22 -05:00
* inode = entry - > inode ;
atomic_inc ( & entry - > inode - > i_count ) ;
2008-01-08 15:46:30 -05:00
* objectid = entry - > objectid ;
write_unlock ( & tree - > lock ) ;
return 1 ;
}
int btrfs_find_del_first_ordered_inode ( struct btrfs_ordered_inode_tree * tree ,
2008-01-16 16:09:22 -05:00
u64 * root_objectid , u64 * objectid ,
struct inode * * inode )
2008-01-08 15:46:30 -05:00
{
struct tree_entry * entry ;
struct rb_node * node ;
write_lock ( & tree - > lock ) ;
node = tree_search ( & tree - > tree , * root_objectid , * objectid ) ;
if ( ! node ) {
write_unlock ( & tree - > lock ) ;
return 0 ;
}
entry = rb_entry ( node , struct tree_entry , rb_node ) ;
while ( comp_entry ( entry , * root_objectid , * objectid ) > = 0 ) {
node = rb_next ( node ) ;
if ( ! node )
break ;
entry = rb_entry ( node , struct tree_entry , rb_node ) ;
}
if ( ! node ) {
write_unlock ( & tree - > lock ) ;
return 0 ;
}
* root_objectid = entry - > root_objectid ;
* objectid = entry - > objectid ;
2008-01-16 16:09:22 -05:00
* inode = entry - > inode ;
atomic_inc ( & entry - > inode - > i_count ) ;
2008-01-08 15:46:30 -05:00
rb_erase ( node , & tree - > tree ) ;
write_unlock ( & tree - > lock ) ;
kfree ( entry ) ;
return 1 ;
}
2008-01-15 08:40:48 -05:00
static int __btrfs_del_ordered_inode ( struct btrfs_ordered_inode_tree * tree ,
2008-01-16 11:44:43 -05:00
struct inode * inode ,
2008-01-15 08:40:48 -05:00
u64 root_objectid , u64 objectid )
{
struct tree_entry * entry ;
struct rb_node * node ;
struct rb_node * prev ;
write_lock ( & tree - > lock ) ;
node = __tree_search ( & tree - > tree , root_objectid , objectid , & prev ) ;
if ( ! node ) {
write_unlock ( & tree - > lock ) ;
return 0 ;
}
rb_erase ( node , & tree - > tree ) ;
2008-01-16 11:44:43 -05:00
BTRFS_I ( inode ) - > ordered_trans = 0 ;
2008-01-15 08:40:48 -05:00
write_unlock ( & tree - > lock ) ;
entry = rb_entry ( node , struct tree_entry , rb_node ) ;
kfree ( entry ) ;
return 1 ;
}
int btrfs_del_ordered_inode ( struct inode * inode )
{
struct btrfs_root * root = BTRFS_I ( inode ) - > root ;
u64 root_objectid = root - > root_key . objectid ;
2008-01-16 11:44:43 -05:00
int ret = 0 ;
2008-01-15 08:40:48 -05:00
spin_lock ( & root - > fs_info - > new_trans_lock ) ;
if ( root - > fs_info - > running_transaction ) {
struct btrfs_ordered_inode_tree * tree ;
tree = & root - > fs_info - > running_transaction - > ordered_inode_tree ;
2008-01-16 11:44:43 -05:00
ret = __btrfs_del_ordered_inode ( tree , inode , root_objectid ,
inode - > i_ino ) ;
2008-01-15 08:40:48 -05:00
}
spin_unlock ( & root - > fs_info - > new_trans_lock ) ;
2008-01-16 11:44:43 -05:00
return ret ;
2008-01-15 08:40:48 -05:00
}
2008-04-25 08:51:48 -04:00
int btrfs_ordered_throttle ( struct btrfs_root * root , struct inode * inode )
{
struct btrfs_transaction * cur = root - > fs_info - > running_transaction ;
while ( cur = = root - > fs_info - > running_transaction & &
atomic_read ( & BTRFS_I ( inode ) - > ordered_writeback ) ) {
# if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,18)
congestion_wait ( WRITE , HZ / 20 ) ;
# else
blk_congestion_wait ( WRITE , HZ / 20 ) ;
# endif
}
return 0 ;
}