2005-04-17 02:20:36 +04:00
/*
* Copyright ( C ) 2001 Momchil Velikov
* Portions Copyright ( C ) 2001 Christoph Hellwig
2005-09-07 02:16:46 +04:00
* Copyright ( C ) 2005 SGI , Christoph Lameter < clameter @ sgi . com >
2006-12-07 07:33:44 +03:00
* Copyright ( C ) 2006 Nick Piggin
2005-04-17 02:20:36 +04:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation ; either version 2 , or ( at
* your option ) any later version .
*
* 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 . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/radix-tree.h>
# include <linux/percpu.h>
# include <linux/slab.h>
# include <linux/notifier.h>
# include <linux/cpu.h>
# include <linux/gfp.h>
# include <linux/string.h>
# include <linux/bitops.h>
2006-12-07 07:33:44 +03:00
# include <linux/rcupdate.h>
2005-04-17 02:20:36 +04:00
# ifdef __KERNEL__
2006-06-23 13:03:22 +04:00
# define RADIX_TREE_MAP_SHIFT (CONFIG_BASE_SMALL ? 4 : 6)
2005-04-17 02:20:36 +04:00
# else
# define RADIX_TREE_MAP_SHIFT 3 /* For more stressful testing */
# endif
# define RADIX_TREE_MAP_SIZE (1UL << RADIX_TREE_MAP_SHIFT)
# define RADIX_TREE_MAP_MASK (RADIX_TREE_MAP_SIZE-1)
# define RADIX_TREE_TAG_LONGS \
( ( RADIX_TREE_MAP_SIZE + BITS_PER_LONG - 1 ) / BITS_PER_LONG )
struct radix_tree_node {
2006-12-07 07:33:44 +03:00
unsigned int height ; /* Height from the bottom */
2005-04-17 02:20:36 +04:00
unsigned int count ;
2006-12-07 07:33:44 +03:00
struct rcu_head rcu_head ;
2005-04-17 02:20:36 +04:00
void * slots [ RADIX_TREE_MAP_SIZE ] ;
2006-03-25 14:08:05 +03:00
unsigned long tags [ RADIX_TREE_MAX_TAGS ] [ RADIX_TREE_TAG_LONGS ] ;
2005-04-17 02:20:36 +04:00
} ;
struct radix_tree_path {
2005-09-07 02:16:46 +04:00
struct radix_tree_node * node ;
2005-04-17 02:20:36 +04:00
int offset ;
} ;
# define RADIX_TREE_INDEX_BITS (8 /* CHAR_BIT */ * sizeof(unsigned long))
# define RADIX_TREE_MAX_PATH (RADIX_TREE_INDEX_BITS / RADIX_TREE_MAP_SHIFT + 2)
2005-07-08 04:56:59 +04:00
static unsigned long height_to_maxindex [ RADIX_TREE_MAX_PATH ] __read_mostly ;
2005-04-17 02:20:36 +04:00
/*
* Radix tree node cache .
*/
2006-12-07 07:33:20 +03:00
static struct kmem_cache * radix_tree_node_cachep ;
2005-04-17 02:20:36 +04:00
/*
* Per - cpu pool of preloaded nodes
*/
struct radix_tree_preload {
int nr ;
struct radix_tree_node * nodes [ RADIX_TREE_MAX_PATH ] ;
} ;
DEFINE_PER_CPU ( struct radix_tree_preload , radix_tree_preloads ) = { 0 , } ;
2006-06-23 13:03:22 +04:00
static inline gfp_t root_gfp_mask ( struct radix_tree_root * root )
{
return root - > gfp_mask & __GFP_BITS_MASK ;
}
2005-04-17 02:20:36 +04:00
/*
* This assumes that the caller has performed appropriate preallocation , and
* that the caller has pinned this thread of control to the current CPU .
*/
static struct radix_tree_node *
radix_tree_node_alloc ( struct radix_tree_root * root )
{
struct radix_tree_node * ret ;
2006-06-23 13:03:22 +04:00
gfp_t gfp_mask = root_gfp_mask ( root ) ;
2005-04-17 02:20:36 +04:00
2006-06-23 13:03:22 +04:00
ret = kmem_cache_alloc ( radix_tree_node_cachep , gfp_mask ) ;
if ( ret = = NULL & & ! ( gfp_mask & __GFP_WAIT ) ) {
2005-04-17 02:20:36 +04:00
struct radix_tree_preload * rtp ;
rtp = & __get_cpu_var ( radix_tree_preloads ) ;
if ( rtp - > nr ) {
ret = rtp - > nodes [ rtp - > nr - 1 ] ;
rtp - > nodes [ rtp - > nr - 1 ] = NULL ;
rtp - > nr - - ;
}
}
2006-12-07 07:33:44 +03:00
BUG_ON ( radix_tree_is_direct_ptr ( ret ) ) ;
2005-04-17 02:20:36 +04:00
return ret ;
}
2006-12-07 07:33:44 +03:00
static void radix_tree_node_rcu_free ( struct rcu_head * head )
{
struct radix_tree_node * node =
container_of ( head , struct radix_tree_node , rcu_head ) ;
kmem_cache_free ( radix_tree_node_cachep , node ) ;
}
2005-04-17 02:20:36 +04:00
static inline void
radix_tree_node_free ( struct radix_tree_node * node )
{
2006-12-07 07:33:44 +03:00
call_rcu ( & node - > rcu_head , radix_tree_node_rcu_free ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Load up this CPU ' s radix_tree_node buffer with sufficient objects to
* ensure that the addition of a single element in the tree cannot fail . On
* success , return zero , with preemption disabled . On error , return - ENOMEM
* with preemption not disabled .
*/
2005-10-07 10:46:04 +04:00
int radix_tree_preload ( gfp_t gfp_mask )
2005-04-17 02:20:36 +04:00
{
struct radix_tree_preload * rtp ;
struct radix_tree_node * node ;
int ret = - ENOMEM ;
preempt_disable ( ) ;
rtp = & __get_cpu_var ( radix_tree_preloads ) ;
while ( rtp - > nr < ARRAY_SIZE ( rtp - > nodes ) ) {
preempt_enable ( ) ;
node = kmem_cache_alloc ( radix_tree_node_cachep , gfp_mask ) ;
if ( node = = NULL )
goto out ;
preempt_disable ( ) ;
rtp = & __get_cpu_var ( radix_tree_preloads ) ;
if ( rtp - > nr < ARRAY_SIZE ( rtp - > nodes ) )
rtp - > nodes [ rtp - > nr + + ] = node ;
else
kmem_cache_free ( radix_tree_node_cachep , node ) ;
}
ret = 0 ;
out :
return ret ;
}
2007-07-14 10:05:04 +04:00
EXPORT_SYMBOL ( radix_tree_preload ) ;
2005-04-17 02:20:36 +04:00
2006-03-25 14:08:05 +03:00
static inline void tag_set ( struct radix_tree_node * node , unsigned int tag ,
int offset )
2005-04-17 02:20:36 +04:00
{
2006-01-08 12:01:41 +03:00
__set_bit ( offset , node - > tags [ tag ] ) ;
2005-04-17 02:20:36 +04:00
}
2006-03-25 14:08:05 +03:00
static inline void tag_clear ( struct radix_tree_node * node , unsigned int tag ,
int offset )
2005-04-17 02:20:36 +04:00
{
2006-01-08 12:01:41 +03:00
__clear_bit ( offset , node - > tags [ tag ] ) ;
2005-04-17 02:20:36 +04:00
}
2006-03-25 14:08:05 +03:00
static inline int tag_get ( struct radix_tree_node * node , unsigned int tag ,
int offset )
2005-04-17 02:20:36 +04:00
{
2006-01-08 12:01:41 +03:00
return test_bit ( offset , node - > tags [ tag ] ) ;
2005-04-17 02:20:36 +04:00
}
2006-06-23 13:03:22 +04:00
static inline void root_tag_set ( struct radix_tree_root * root , unsigned int tag )
{
2006-10-11 01:47:57 +04:00
root - > gfp_mask | = ( __force gfp_t ) ( 1 < < ( tag + __GFP_BITS_SHIFT ) ) ;
2006-06-23 13:03:22 +04:00
}
static inline void root_tag_clear ( struct radix_tree_root * root , unsigned int tag )
{
2006-10-11 01:47:57 +04:00
root - > gfp_mask & = ( __force gfp_t ) ~ ( 1 < < ( tag + __GFP_BITS_SHIFT ) ) ;
2006-06-23 13:03:22 +04:00
}
static inline void root_tag_clear_all ( struct radix_tree_root * root )
{
root - > gfp_mask & = __GFP_BITS_MASK ;
}
static inline int root_tag_get ( struct radix_tree_root * root , unsigned int tag )
{
2006-10-11 01:47:57 +04:00
return ( __force unsigned ) root - > gfp_mask & ( 1 < < ( tag + __GFP_BITS_SHIFT ) ) ;
2006-06-23 13:03:22 +04:00
}
2006-01-08 12:01:40 +03:00
/*
* Returns 1 if any slot in the node has this tag set .
* Otherwise returns 0.
*/
2006-03-25 14:08:05 +03:00
static inline int any_tag_set ( struct radix_tree_node * node , unsigned int tag )
2006-01-08 12:01:40 +03:00
{
int idx ;
for ( idx = 0 ; idx < RADIX_TREE_TAG_LONGS ; idx + + ) {
if ( node - > tags [ tag ] [ idx ] )
return 1 ;
}
return 0 ;
}
2005-04-17 02:20:36 +04:00
/*
* Return the maximum key which can be store into a
* radix tree with height HEIGHT .
*/
static inline unsigned long radix_tree_maxindex ( unsigned int height )
{
return height_to_maxindex [ height ] ;
}
/*
* Extend a radix tree so it can store key @ index .
*/
static int radix_tree_extend ( struct radix_tree_root * root , unsigned long index )
{
struct radix_tree_node * node ;
unsigned int height ;
int tag ;
/* Figure out what the height should be. */
height = root - > height + 1 ;
while ( index > radix_tree_maxindex ( height ) )
height + + ;
if ( root - > rnode = = NULL ) {
root - > height = height ;
goto out ;
}
do {
2006-12-07 07:33:44 +03:00
unsigned int newheight ;
2005-04-17 02:20:36 +04:00
if ( ! ( node = radix_tree_node_alloc ( root ) ) )
return - ENOMEM ;
/* Increase the height. */
2006-12-07 07:33:44 +03:00
node - > slots [ 0 ] = radix_tree_direct_to_ptr ( root - > rnode ) ;
2005-04-17 02:20:36 +04:00
/* Propagate the aggregated tag info into the new root */
2006-03-25 14:08:05 +03:00
for ( tag = 0 ; tag < RADIX_TREE_MAX_TAGS ; tag + + ) {
2006-06-23 13:03:22 +04:00
if ( root_tag_get ( root , tag ) )
2005-04-17 02:20:36 +04:00
tag_set ( node , tag , 0 ) ;
}
2006-12-07 07:33:44 +03:00
newheight = root - > height + 1 ;
node - > height = newheight ;
2005-04-17 02:20:36 +04:00
node - > count = 1 ;
2006-12-07 07:33:44 +03:00
rcu_assign_pointer ( root - > rnode , node ) ;
root - > height = newheight ;
2005-04-17 02:20:36 +04:00
} while ( height > root - > height ) ;
out :
return 0 ;
}
/**
* radix_tree_insert - insert into a radix tree
* @ root : radix tree root
* @ index : index key
* @ item : item to insert
*
* Insert an item into the radix tree at position @ index .
*/
int radix_tree_insert ( struct radix_tree_root * root ,
unsigned long index , void * item )
{
2005-09-07 02:16:46 +04:00
struct radix_tree_node * node = NULL , * slot ;
2005-04-17 02:20:36 +04:00
unsigned int height , shift ;
int offset ;
int error ;
2006-12-07 07:33:44 +03:00
BUG_ON ( radix_tree_is_direct_ptr ( item ) ) ;
2005-04-17 02:20:36 +04:00
/* Make sure the tree is high enough. */
2006-06-23 13:03:22 +04:00
if ( index > radix_tree_maxindex ( root - > height ) ) {
2005-04-17 02:20:36 +04:00
error = radix_tree_extend ( root , index ) ;
if ( error )
return error ;
}
2005-09-07 02:16:46 +04:00
slot = root - > rnode ;
2005-04-17 02:20:36 +04:00
height = root - > height ;
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
offset = 0 ; /* uninitialised var warning */
2006-06-23 13:03:22 +04:00
while ( height > 0 ) {
2005-09-07 02:16:46 +04:00
if ( slot = = NULL ) {
2005-04-17 02:20:36 +04:00
/* Have to add a child node. */
2005-09-07 02:16:46 +04:00
if ( ! ( slot = radix_tree_node_alloc ( root ) ) )
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
2006-12-07 07:33:44 +03:00
slot - > height = height ;
2005-09-07 02:16:46 +04:00
if ( node ) {
2006-12-07 07:33:44 +03:00
rcu_assign_pointer ( node - > slots [ offset ] , slot ) ;
2005-04-17 02:20:36 +04:00
node - > count + + ;
2005-09-07 02:16:46 +04:00
} else
2006-12-07 07:33:44 +03:00
rcu_assign_pointer ( root - > rnode , slot ) ;
2005-04-17 02:20:36 +04:00
}
/* Go a level down */
offset = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
2005-09-07 02:16:46 +04:00
node = slot ;
slot = node - > slots [ offset ] ;
2005-04-17 02:20:36 +04:00
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
2006-06-23 13:03:22 +04:00
}
2005-04-17 02:20:36 +04:00
2005-09-07 02:16:46 +04:00
if ( slot ! = NULL )
2005-04-17 02:20:36 +04:00
return - EEXIST ;
2005-09-07 02:16:46 +04:00
2006-06-23 13:03:22 +04:00
if ( node ) {
node - > count + + ;
2006-12-07 07:33:44 +03:00
rcu_assign_pointer ( node - > slots [ offset ] , item ) ;
2006-06-23 13:03:22 +04:00
BUG_ON ( tag_get ( node , 0 , offset ) ) ;
BUG_ON ( tag_get ( node , 1 , offset ) ) ;
} else {
2006-12-07 07:33:44 +03:00
rcu_assign_pointer ( root - > rnode , radix_tree_ptr_to_direct ( item ) ) ;
2006-06-23 13:03:22 +04:00
BUG_ON ( root_tag_get ( root , 0 ) ) ;
BUG_ON ( root_tag_get ( root , 1 ) ) ;
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
EXPORT_SYMBOL ( radix_tree_insert ) ;
2006-12-07 07:33:44 +03:00
/**
* radix_tree_lookup_slot - lookup a slot in a radix tree
* @ root : radix tree root
* @ index : index key
*
* Returns : the slot corresponding to the position @ index in the
* radix tree @ root . This is useful for update - if - exists operations .
*
* This function cannot be called under rcu_read_lock , it must be
* excluded from writers , as must the returned slot for subsequent
* use by radix_tree_deref_slot ( ) and radix_tree_replace slot .
* Caller must hold tree write locked across slot lookup and
* replace .
*/
void * * radix_tree_lookup_slot ( struct radix_tree_root * root , unsigned long index )
2005-04-17 02:20:36 +04:00
{
unsigned int height , shift ;
2006-12-07 07:33:44 +03:00
struct radix_tree_node * node , * * slot ;
2006-06-23 13:03:22 +04:00
2006-12-07 07:33:44 +03:00
node = root - > rnode ;
if ( node = = NULL )
2005-04-17 02:20:36 +04:00
return NULL ;
2006-12-07 07:33:44 +03:00
if ( radix_tree_is_direct_ptr ( node ) ) {
if ( index > 0 )
return NULL ;
2006-06-23 13:03:22 +04:00
return ( void * * ) & root - > rnode ;
2006-12-07 07:33:44 +03:00
}
height = node - > height ;
if ( index > radix_tree_maxindex ( height ) )
return NULL ;
2006-06-23 13:03:22 +04:00
2005-04-17 02:20:36 +04:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
2006-12-07 07:33:44 +03:00
do {
slot = ( struct radix_tree_node * * )
( node - > slots + ( ( index > > shift ) & RADIX_TREE_MAP_MASK ) ) ;
node = * slot ;
if ( node = = NULL )
2005-04-17 02:20:36 +04:00
return NULL ;
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
2006-12-07 07:33:44 +03:00
} while ( height > 0 ) ;
2005-04-17 02:20:36 +04:00
2005-11-07 11:59:29 +03:00
return ( void * * ) slot ;
}
EXPORT_SYMBOL ( radix_tree_lookup_slot ) ;
/**
* radix_tree_lookup - perform lookup operation on a radix tree
* @ root : radix tree root
* @ index : index key
*
* Lookup the item at the position @ index in the radix tree @ root .
2006-12-07 07:33:44 +03:00
*
* This function can be called under rcu_read_lock , however the caller
* must manage lifetimes of leaf nodes ( eg . RCU may also be used to free
* them safely ) . No RCU barriers are required to access or modify the
* returned item , however .
2005-11-07 11:59:29 +03:00
*/
void * radix_tree_lookup ( struct radix_tree_root * root , unsigned long index )
{
2006-12-07 07:33:44 +03:00
unsigned int height , shift ;
struct radix_tree_node * node , * * slot ;
node = rcu_dereference ( root - > rnode ) ;
if ( node = = NULL )
return NULL ;
if ( radix_tree_is_direct_ptr ( node ) ) {
if ( index > 0 )
return NULL ;
return radix_tree_direct_to_ptr ( node ) ;
}
height = node - > height ;
if ( index > radix_tree_maxindex ( height ) )
return NULL ;
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
2005-11-07 11:59:29 +03:00
2006-12-07 07:33:44 +03:00
do {
slot = ( struct radix_tree_node * * )
( node - > slots + ( ( index > > shift ) & RADIX_TREE_MAP_MASK ) ) ;
node = rcu_dereference ( * slot ) ;
if ( node = = NULL )
return NULL ;
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
} while ( height > 0 ) ;
return node ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( radix_tree_lookup ) ;
/**
* radix_tree_tag_set - set a tag on a radix tree node
* @ root : radix tree root
* @ index : index key
* @ tag : tag index
*
2006-03-25 14:08:05 +03:00
* Set the search tag ( which must be < RADIX_TREE_MAX_TAGS )
* corresponding to @ index in the radix tree . From
2005-04-17 02:20:36 +04:00
* the root all the way down to the leaf node .
*
* Returns the address of the tagged item . Setting a tag on a not - present
* item is a bug .
*/
void * radix_tree_tag_set ( struct radix_tree_root * root ,
2006-03-25 14:08:05 +03:00
unsigned long index , unsigned int tag )
2005-04-17 02:20:36 +04:00
{
unsigned int height , shift ;
2005-09-07 02:16:46 +04:00
struct radix_tree_node * slot ;
2005-04-17 02:20:36 +04:00
height = root - > height ;
2006-06-23 13:03:25 +04:00
BUG_ON ( index > radix_tree_maxindex ( height ) ) ;
2005-04-17 02:20:36 +04:00
2005-09-07 02:16:46 +04:00
slot = root - > rnode ;
2006-06-23 13:03:22 +04:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
2005-04-17 02:20:36 +04:00
while ( height > 0 ) {
int offset ;
offset = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
2006-01-08 12:01:41 +03:00
if ( ! tag_get ( slot , tag , offset ) )
tag_set ( slot , tag , offset ) ;
2005-09-07 02:16:46 +04:00
slot = slot - > slots [ offset ] ;
BUG_ON ( slot = = NULL ) ;
2005-04-17 02:20:36 +04:00
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
}
2006-06-23 13:03:22 +04:00
/* set the root's tag bit */
if ( slot & & ! root_tag_get ( root , tag ) )
root_tag_set ( root , tag ) ;
2005-09-07 02:16:46 +04:00
return slot ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( radix_tree_tag_set ) ;
/**
* radix_tree_tag_clear - clear a tag on a radix tree node
* @ root : radix tree root
* @ index : index key
* @ tag : tag index
*
2006-03-25 14:08:05 +03:00
* Clear the search tag ( which must be < RADIX_TREE_MAX_TAGS )
* corresponding to @ index in the radix tree . If
2005-04-17 02:20:36 +04:00
* this causes the leaf node to have no tags set then clear the tag in the
* next - to - leaf node , etc .
*
* Returns the address of the tagged item on success , else NULL . ie :
* has the same return value and semantics as radix_tree_lookup ( ) .
*/
void * radix_tree_tag_clear ( struct radix_tree_root * root ,
2006-03-25 14:08:05 +03:00
unsigned long index , unsigned int tag )
2005-04-17 02:20:36 +04:00
{
struct radix_tree_path path [ RADIX_TREE_MAX_PATH ] , * pathp = path ;
2006-06-23 13:03:22 +04:00
struct radix_tree_node * slot = NULL ;
2005-04-17 02:20:36 +04:00
unsigned int height , shift ;
height = root - > height ;
if ( index > radix_tree_maxindex ( height ) )
goto out ;
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
pathp - > node = NULL ;
2005-09-07 02:16:46 +04:00
slot = root - > rnode ;
2005-04-17 02:20:36 +04:00
while ( height > 0 ) {
int offset ;
2005-09-07 02:16:46 +04:00
if ( slot = = NULL )
2005-04-17 02:20:36 +04:00
goto out ;
offset = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
pathp [ 1 ] . offset = offset ;
2005-09-07 02:16:46 +04:00
pathp [ 1 ] . node = slot ;
slot = slot - > slots [ offset ] ;
2005-04-17 02:20:36 +04:00
pathp + + ;
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
}
2006-06-23 13:03:22 +04:00
if ( slot = = NULL )
2005-04-17 02:20:36 +04:00
goto out ;
2006-06-23 13:03:22 +04:00
while ( pathp - > node ) {
2006-01-08 12:01:41 +03:00
if ( ! tag_get ( pathp - > node , tag , pathp - > offset ) )
goto out ;
2005-09-07 02:16:46 +04:00
tag_clear ( pathp - > node , tag , pathp - > offset ) ;
2006-01-08 12:01:40 +03:00
if ( any_tag_set ( pathp - > node , tag ) )
goto out ;
2005-04-17 02:20:36 +04:00
pathp - - ;
2006-06-23 13:03:22 +04:00
}
/* clear the root's tag bit */
if ( root_tag_get ( root , tag ) )
root_tag_clear ( root , tag ) ;
2005-04-17 02:20:36 +04:00
out :
2006-06-23 13:03:22 +04:00
return slot ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( radix_tree_tag_clear ) ;
# ifndef __KERNEL__ /* Only the test harness uses this at present */
/**
2005-09-07 02:16:48 +04:00
* radix_tree_tag_get - get a tag on a radix tree node
* @ root : radix tree root
* @ index : index key
2006-03-25 14:08:05 +03:00
* @ tag : tag index ( < RADIX_TREE_MAX_TAGS )
2005-04-17 02:20:36 +04:00
*
2005-09-07 02:16:48 +04:00
* Return values :
2005-04-17 02:20:36 +04:00
*
2006-06-23 13:03:22 +04:00
* 0 : tag not present or not set
* 1 : tag set
2005-04-17 02:20:36 +04:00
*/
int radix_tree_tag_get ( struct radix_tree_root * root ,
2006-03-25 14:08:05 +03:00
unsigned long index , unsigned int tag )
2005-04-17 02:20:36 +04:00
{
unsigned int height , shift ;
2006-12-07 07:33:44 +03:00
struct radix_tree_node * node ;
2005-04-17 02:20:36 +04:00
int saw_unset_tag = 0 ;
2006-06-23 13:03:22 +04:00
/* check the root's tag bit */
if ( ! root_tag_get ( root , tag ) )
return 0 ;
2006-12-07 07:33:44 +03:00
node = rcu_dereference ( root - > rnode ) ;
if ( node = = NULL )
return 0 ;
if ( radix_tree_is_direct_ptr ( node ) )
return ( index = = 0 ) ;
height = node - > height ;
if ( index > radix_tree_maxindex ( height ) )
return 0 ;
2006-06-23 13:03:22 +04:00
2005-04-17 02:20:36 +04:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
for ( ; ; ) {
int offset ;
2006-12-07 07:33:44 +03:00
if ( node = = NULL )
2005-04-17 02:20:36 +04:00
return 0 ;
offset = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
/*
* This is just a debug check . Later , we can bale as soon as
* we see an unset tag .
*/
2006-12-07 07:33:44 +03:00
if ( ! tag_get ( node , tag , offset ) )
2005-04-17 02:20:36 +04:00
saw_unset_tag = 1 ;
if ( height = = 1 ) {
2006-12-07 07:33:44 +03:00
int ret = tag_get ( node , tag , offset ) ;
2005-04-17 02:20:36 +04:00
BUG_ON ( ret & & saw_unset_tag ) ;
2006-06-25 16:48:14 +04:00
return ! ! ret ;
2005-04-17 02:20:36 +04:00
}
2006-12-07 07:33:44 +03:00
node = rcu_dereference ( node - > slots [ offset ] ) ;
2005-04-17 02:20:36 +04:00
shift - = RADIX_TREE_MAP_SHIFT ;
height - - ;
}
}
EXPORT_SYMBOL ( radix_tree_tag_get ) ;
# endif
static unsigned int
2006-12-07 07:33:44 +03:00
__lookup ( struct radix_tree_node * slot , void * * results , unsigned long index ,
2005-04-17 02:20:36 +04:00
unsigned int max_items , unsigned long * next_index )
{
unsigned int nr_found = 0 ;
2005-09-07 02:16:46 +04:00
unsigned int shift , height ;
unsigned long i ;
2006-12-07 07:33:44 +03:00
height = slot - > height ;
if ( height = = 0 )
2005-09-07 02:16:46 +04:00
goto out ;
2005-04-17 02:20:36 +04:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
2005-09-07 02:16:46 +04:00
for ( ; height > 1 ; height - - ) {
2006-12-07 07:33:44 +03:00
i = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
for ( ; ; ) {
2005-04-17 02:20:36 +04:00
if ( slot - > slots [ i ] ! = NULL )
break ;
index & = ~ ( ( 1UL < < shift ) - 1 ) ;
index + = 1UL < < shift ;
if ( index = = 0 )
goto out ; /* 32-bit wraparound */
2006-12-07 07:33:44 +03:00
i + + ;
if ( i = = RADIX_TREE_MAP_SIZE )
goto out ;
2005-04-17 02:20:36 +04:00
}
shift - = RADIX_TREE_MAP_SHIFT ;
2006-12-07 07:33:44 +03:00
slot = rcu_dereference ( slot - > slots [ i ] ) ;
if ( slot = = NULL )
goto out ;
2005-04-17 02:20:36 +04:00
}
2005-09-07 02:16:46 +04:00
/* Bottom level: grab some items */
for ( i = index & RADIX_TREE_MAP_MASK ; i < RADIX_TREE_MAP_SIZE ; i + + ) {
2006-12-07 07:33:44 +03:00
struct radix_tree_node * node ;
2005-09-07 02:16:46 +04:00
index + + ;
2006-12-07 07:33:44 +03:00
node = slot - > slots [ i ] ;
if ( node ) {
results [ nr_found + + ] = rcu_dereference ( node ) ;
2005-09-07 02:16:46 +04:00
if ( nr_found = = max_items )
goto out ;
}
}
2005-04-17 02:20:36 +04:00
out :
* next_index = index ;
return nr_found ;
}
/**
* radix_tree_gang_lookup - perform multiple lookup on a radix tree
* @ root : radix tree root
* @ results : where the results of the lookup are placed
* @ first_index : start the lookup from this key
* @ max_items : place up to this many items at * results
*
* Performs an index - ascending scan of the tree for present items . Places
* them at * @ results and returns the number of items which were placed at
* * @ results .
*
* The implementation is naive .
2006-12-07 07:33:44 +03:00
*
* Like radix_tree_lookup , radix_tree_gang_lookup may be called under
* rcu_read_lock . In this case , rather than the returned results being
* an atomic snapshot of the tree at a single point in time , the semantics
* of an RCU protected gang lookup are as though multiple radix_tree_lookups
* have been issued in individual locks , and results stored in ' results ' .
2005-04-17 02:20:36 +04:00
*/
unsigned int
radix_tree_gang_lookup ( struct radix_tree_root * root , void * * results ,
unsigned long first_index , unsigned int max_items )
{
2006-12-07 07:33:44 +03:00
unsigned long max_index ;
struct radix_tree_node * node ;
2005-04-17 02:20:36 +04:00
unsigned long cur_index = first_index ;
2006-12-07 07:33:44 +03:00
unsigned int ret ;
node = rcu_dereference ( root - > rnode ) ;
if ( ! node )
return 0 ;
2005-04-17 02:20:36 +04:00
2006-12-07 07:33:44 +03:00
if ( radix_tree_is_direct_ptr ( node ) ) {
if ( first_index > 0 )
return 0 ;
node = radix_tree_direct_to_ptr ( node ) ;
results [ 0 ] = rcu_dereference ( node ) ;
return 1 ;
}
max_index = radix_tree_maxindex ( node - > height ) ;
ret = 0 ;
2005-04-17 02:20:36 +04:00
while ( ret < max_items ) {
unsigned int nr_found ;
unsigned long next_index ; /* Index of next search */
if ( cur_index > max_index )
break ;
2006-12-07 07:33:44 +03:00
nr_found = __lookup ( node , results + ret , cur_index ,
2005-04-17 02:20:36 +04:00
max_items - ret , & next_index ) ;
ret + = nr_found ;
if ( next_index = = 0 )
break ;
cur_index = next_index ;
}
2006-12-07 07:33:44 +03:00
2005-04-17 02:20:36 +04:00
return ret ;
}
EXPORT_SYMBOL ( radix_tree_gang_lookup ) ;
/*
* FIXME : the two tag_get ( ) s here should use find_next_bit ( ) instead of
* open - coding the search .
*/
static unsigned int
2006-12-07 07:33:44 +03:00
__lookup_tag ( struct radix_tree_node * slot , void * * results , unsigned long index ,
2006-03-25 14:08:05 +03:00
unsigned int max_items , unsigned long * next_index , unsigned int tag )
2005-04-17 02:20:36 +04:00
{
unsigned int nr_found = 0 ;
2006-12-07 07:33:44 +03:00
unsigned int shift , height ;
2005-04-17 02:20:36 +04:00
2006-12-07 07:33:44 +03:00
height = slot - > height ;
if ( height = = 0 )
2006-06-23 13:03:22 +04:00
goto out ;
2006-12-07 07:33:44 +03:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
2005-04-17 02:20:36 +04:00
2006-12-07 07:33:44 +03:00
while ( height > 0 ) {
unsigned long i = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
2005-04-17 02:20:36 +04:00
2006-12-07 07:33:44 +03:00
for ( ; ; ) {
if ( tag_get ( slot , tag , i ) )
2005-04-17 02:20:36 +04:00
break ;
index & = ~ ( ( 1UL < < shift ) - 1 ) ;
index + = 1UL < < shift ;
if ( index = = 0 )
goto out ; /* 32-bit wraparound */
2006-12-07 07:33:44 +03:00
i + + ;
if ( i = = RADIX_TREE_MAP_SIZE )
goto out ;
2005-04-17 02:20:36 +04:00
}
height - - ;
if ( height = = 0 ) { /* Bottom level: grab some items */
unsigned long j = index & RADIX_TREE_MAP_MASK ;
for ( ; j < RADIX_TREE_MAP_SIZE ; j + + ) {
2006-12-07 07:33:44 +03:00
struct radix_tree_node * node ;
2005-04-17 02:20:36 +04:00
index + + ;
2006-12-07 07:33:44 +03:00
if ( ! tag_get ( slot , tag , j ) )
continue ;
node = slot - > slots [ j ] ;
/*
* Even though the tag was found set , we need to
* recheck that we have a non - NULL node , because
* if this lookup is lockless , it may have been
* subsequently deleted .
*
* Similar care must be taken in any place that
* lookup - > slots [ x ] without a lock ( ie . can ' t
* rely on its value remaining the same ) .
*/
if ( node ) {
node = rcu_dereference ( node ) ;
results [ nr_found + + ] = node ;
2005-04-17 02:20:36 +04:00
if ( nr_found = = max_items )
goto out ;
}
}
}
shift - = RADIX_TREE_MAP_SHIFT ;
2006-12-07 07:33:44 +03:00
slot = rcu_dereference ( slot - > slots [ i ] ) ;
if ( slot = = NULL )
break ;
}
2005-04-17 02:20:36 +04:00
out :
* next_index = index ;
return nr_found ;
}
/**
* radix_tree_gang_lookup_tag - perform multiple lookup on a radix tree
* based on a tag
* @ root : radix tree root
* @ results : where the results of the lookup are placed
* @ first_index : start the lookup from this key
* @ max_items : place up to this many items at * results
2006-03-25 14:08:05 +03:00
* @ tag : the tag index ( < RADIX_TREE_MAX_TAGS )
2005-04-17 02:20:36 +04:00
*
* Performs an index - ascending scan of the tree for present items which
* have the tag indexed by @ tag set . Places the items at * @ results and
* returns the number of items which were placed at * @ results .
*/
unsigned int
radix_tree_gang_lookup_tag ( struct radix_tree_root * root , void * * results ,
2006-03-25 14:08:05 +03:00
unsigned long first_index , unsigned int max_items ,
unsigned int tag )
2005-04-17 02:20:36 +04:00
{
2006-12-07 07:33:44 +03:00
struct radix_tree_node * node ;
unsigned long max_index ;
2005-04-17 02:20:36 +04:00
unsigned long cur_index = first_index ;
2006-12-07 07:33:44 +03:00
unsigned int ret ;
2005-04-17 02:20:36 +04:00
2006-06-23 13:03:22 +04:00
/* check the root's tag bit */
if ( ! root_tag_get ( root , tag ) )
return 0 ;
2006-12-07 07:33:44 +03:00
node = rcu_dereference ( root - > rnode ) ;
if ( ! node )
return 0 ;
if ( radix_tree_is_direct_ptr ( node ) ) {
if ( first_index > 0 )
return 0 ;
node = radix_tree_direct_to_ptr ( node ) ;
results [ 0 ] = rcu_dereference ( node ) ;
return 1 ;
}
max_index = radix_tree_maxindex ( node - > height ) ;
ret = 0 ;
2005-04-17 02:20:36 +04:00
while ( ret < max_items ) {
unsigned int nr_found ;
unsigned long next_index ; /* Index of next search */
if ( cur_index > max_index )
break ;
2006-12-07 07:33:44 +03:00
nr_found = __lookup_tag ( node , results + ret , cur_index ,
2005-04-17 02:20:36 +04:00
max_items - ret , & next_index , tag ) ;
ret + = nr_found ;
if ( next_index = = 0 )
break ;
cur_index = next_index ;
}
2006-12-07 07:33:44 +03:00
2005-04-17 02:20:36 +04:00
return ret ;
}
EXPORT_SYMBOL ( radix_tree_gang_lookup_tag ) ;
2006-01-08 12:01:41 +03:00
/**
* radix_tree_shrink - shrink height of a radix tree to minimal
* @ root radix tree root
*/
static inline void radix_tree_shrink ( struct radix_tree_root * root )
{
/* try to shrink tree height */
2006-06-23 13:03:22 +04:00
while ( root - > height > 0 & &
2006-01-08 12:01:41 +03:00
root - > rnode - > count = = 1 & &
root - > rnode - > slots [ 0 ] ) {
struct radix_tree_node * to_free = root - > rnode ;
2006-12-07 07:33:44 +03:00
void * newptr ;
2006-01-08 12:01:41 +03:00
2006-12-07 07:33:44 +03:00
/*
* We don ' t need rcu_assign_pointer ( ) , since we are simply
* moving the node from one part of the tree to another . If
* it was safe to dereference the old pointer to it
* ( to_free - > slots [ 0 ] ) , it will be safe to dereference the new
* one ( root - > rnode ) .
*/
newptr = to_free - > slots [ 0 ] ;
if ( root - > height = = 1 )
newptr = radix_tree_ptr_to_direct ( newptr ) ;
root - > rnode = newptr ;
2006-01-08 12:01:41 +03:00
root - > height - - ;
/* must only free zeroed nodes into the slab */
tag_clear ( to_free , 0 , 0 ) ;
tag_clear ( to_free , 1 , 0 ) ;
to_free - > slots [ 0 ] = NULL ;
to_free - > count = 0 ;
radix_tree_node_free ( to_free ) ;
}
}
2005-04-17 02:20:36 +04:00
/**
* radix_tree_delete - delete an item from a radix tree
* @ root : radix tree root
* @ index : index key
*
* Remove the item at @ index from the radix tree rooted at @ root .
*
* Returns the address of the deleted item , or NULL if it was not present .
*/
void * radix_tree_delete ( struct radix_tree_root * root , unsigned long index )
{
struct radix_tree_path path [ RADIX_TREE_MAX_PATH ] , * pathp = path ;
2006-06-23 13:03:22 +04:00
struct radix_tree_node * slot = NULL ;
2006-12-07 07:33:44 +03:00
struct radix_tree_node * to_free ;
2005-04-17 02:20:36 +04:00
unsigned int height , shift ;
2006-01-08 12:01:41 +03:00
int tag ;
int offset ;
2005-04-17 02:20:36 +04:00
height = root - > height ;
if ( index > radix_tree_maxindex ( height ) )
goto out ;
2006-06-23 13:03:22 +04:00
slot = root - > rnode ;
if ( height = = 0 & & root - > rnode ) {
2006-12-07 07:33:44 +03:00
slot = radix_tree_direct_to_ptr ( slot ) ;
2006-06-23 13:03:22 +04:00
root_tag_clear_all ( root ) ;
root - > rnode = NULL ;
goto out ;
}
2005-04-17 02:20:36 +04:00
shift = ( height - 1 ) * RADIX_TREE_MAP_SHIFT ;
pathp - > node = NULL ;
2006-06-23 13:03:22 +04:00
do {
2005-09-07 02:16:46 +04:00
if ( slot = = NULL )
2005-04-17 02:20:36 +04:00
goto out ;
2006-01-08 12:01:41 +03:00
pathp + + ;
2005-04-17 02:20:36 +04:00
offset = ( index > > shift ) & RADIX_TREE_MAP_MASK ;
2006-01-08 12:01:41 +03:00
pathp - > offset = offset ;
pathp - > node = slot ;
2005-09-07 02:16:46 +04:00
slot = slot - > slots [ offset ] ;
2005-04-17 02:20:36 +04:00
shift - = RADIX_TREE_MAP_SHIFT ;
2006-06-23 13:03:22 +04:00
height - - ;
} while ( height > 0 ) ;
2005-04-17 02:20:36 +04:00
2006-06-23 13:03:22 +04:00
if ( slot = = NULL )
2005-04-17 02:20:36 +04:00
goto out ;
/*
* Clear all tags associated with the just - deleted item
*/
2006-03-25 14:08:05 +03:00
for ( tag = 0 ; tag < RADIX_TREE_MAX_TAGS ; tag + + ) {
2006-06-23 13:03:22 +04:00
if ( tag_get ( pathp - > node , tag , pathp - > offset ) )
radix_tree_tag_clear ( root , index , tag ) ;
2006-01-08 12:01:41 +03:00
}
2005-04-17 02:20:36 +04:00
2006-12-07 07:33:44 +03:00
to_free = NULL ;
2005-09-07 02:16:46 +04:00
/* Now free the nodes we do not need anymore */
2006-06-23 13:03:22 +04:00
while ( pathp - > node ) {
2005-09-07 02:16:46 +04:00
pathp - > node - > slots [ pathp - > offset ] = NULL ;
2006-01-08 12:01:41 +03:00
pathp - > node - > count - - ;
2006-12-07 07:33:44 +03:00
/*
* Queue the node for deferred freeing after the
* last reference to it disappears ( set NULL , above ) .
*/
if ( to_free )
radix_tree_node_free ( to_free ) ;
2006-01-08 12:01:41 +03:00
if ( pathp - > node - > count ) {
if ( pathp - > node = = root - > rnode )
radix_tree_shrink ( root ) ;
2005-09-07 02:16:46 +04:00
goto out ;
2006-01-08 12:01:41 +03:00
}
2005-09-07 02:16:46 +04:00
/* Node with zero slots in use so free it */
2006-12-07 07:33:44 +03:00
to_free = pathp - > node ;
2006-06-23 13:03:22 +04:00
pathp - - ;
2006-12-07 07:33:44 +03:00
2005-04-17 02:20:36 +04:00
}
2006-06-23 13:03:22 +04:00
root_tag_clear_all ( root ) ;
2005-09-07 02:16:46 +04:00
root - > height = 0 ;
2006-06-23 13:03:22 +04:00
root - > rnode = NULL ;
2006-12-07 07:33:44 +03:00
if ( to_free )
radix_tree_node_free ( to_free ) ;
2006-06-23 13:03:22 +04:00
2005-04-17 02:20:36 +04:00
out :
2006-06-23 13:03:22 +04:00
return slot ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( radix_tree_delete ) ;
/**
* radix_tree_tagged - test whether any items in the tree are tagged
* @ root : radix tree root
* @ tag : tag to test
*/
2006-03-25 14:08:05 +03:00
int radix_tree_tagged ( struct radix_tree_root * root , unsigned int tag )
2005-04-17 02:20:36 +04:00
{
2006-06-23 13:03:22 +04:00
return root_tag_get ( root , tag ) ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( radix_tree_tagged ) ;
static void
2006-12-07 07:33:20 +03:00
radix_tree_node_ctor ( void * node , struct kmem_cache * cachep , unsigned long flags )
2005-04-17 02:20:36 +04:00
{
memset ( node , 0 , sizeof ( struct radix_tree_node ) ) ;
}
static __init unsigned long __maxindex ( unsigned int height )
{
unsigned int tmp = height * RADIX_TREE_MAP_SHIFT ;
unsigned long index = ( ~ 0UL > > ( RADIX_TREE_INDEX_BITS - tmp - 1 ) ) > > 1 ;
if ( tmp > = RADIX_TREE_INDEX_BITS )
index = ~ 0UL ;
return index ;
}
static __init void radix_tree_init_maxindex ( void )
{
unsigned int i ;
for ( i = 0 ; i < ARRAY_SIZE ( height_to_maxindex ) ; i + + )
height_to_maxindex [ i ] = __maxindex ( i ) ;
}
static int radix_tree_callback ( struct notifier_block * nfb ,
unsigned long action ,
void * hcpu )
{
int cpu = ( long ) hcpu ;
struct radix_tree_preload * rtp ;
/* Free per-cpu pool of perloaded nodes */
2007-05-09 13:35:10 +04:00
if ( action = = CPU_DEAD | | action = = CPU_DEAD_FROZEN ) {
2005-04-17 02:20:36 +04:00
rtp = & per_cpu ( radix_tree_preloads , cpu ) ;
while ( rtp - > nr ) {
kmem_cache_free ( radix_tree_node_cachep ,
rtp - > nodes [ rtp - > nr - 1 ] ) ;
rtp - > nodes [ rtp - > nr - 1 ] = NULL ;
rtp - > nr - - ;
}
}
return NOTIFY_OK ;
}
void __init radix_tree_init ( void )
{
radix_tree_node_cachep = kmem_cache_create ( " radix_tree_node " ,
sizeof ( struct radix_tree_node ) , 0 ,
2007-07-20 05:11:58 +04:00
SLAB_PANIC , radix_tree_node_ctor ) ;
2005-04-17 02:20:36 +04:00
radix_tree_init_maxindex ( ) ;
hotcpu_notifier ( radix_tree_callback , 0 ) ;
}