diff options
| author | Huang Shijie <shijie8@gmail.com> | 2009-06-16 18:33:42 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-16 22:47:49 -0400 |
| commit | b72b71c6cb6ecc564d4d5f9c512a7df269837846 (patch) | |
| tree | 9923a6e1d59f92825df87cfd2bbd613d0f47a5f7 /lib | |
| parent | 30639b6af85a92491b22dd14c17b14ca11da60e6 (diff) | |
lib: do code optimization for radix_tree_lookup() and radix_tree_lookup_slot()
radix_tree_lookup() and radix_tree_lookup_slot() have much the
same code except for the return value.
Introduce radix_tree_lookup_element() to do the real work.
/*
* is_slot == 1 : search for the slot.
* is_slot == 0 : search for the node.
*/
static void * radix_tree_lookup_element(struct radix_tree_root *root,
unsigned long index, int is_slot);
Signed-off-by: Huang Shijie <shijie8@gmail.com>
Cc: Nick Piggin <nickpiggin@yahoo.com.au>
Cc: Christoph Lameter <cl@linux-foundation.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/radix-tree.c | 73 |
1 files changed, 26 insertions, 47 deletions
diff --git a/lib/radix-tree.c b/lib/radix-tree.c index 5301a52cdb4d..23abbd93cae1 100644 --- a/lib/radix-tree.c +++ b/lib/radix-tree.c | |||
| @@ -351,20 +351,12 @@ int radix_tree_insert(struct radix_tree_root *root, | |||
| 351 | } | 351 | } |
| 352 | EXPORT_SYMBOL(radix_tree_insert); | 352 | EXPORT_SYMBOL(radix_tree_insert); |
| 353 | 353 | ||
| 354 | /** | 354 | /* |
| 355 | * radix_tree_lookup_slot - lookup a slot in a radix tree | 355 | * is_slot == 1 : search for the slot. |
| 356 | * @root: radix tree root | 356 | * is_slot == 0 : search for the node. |
| 357 | * @index: index key | ||
| 358 | * | ||
| 359 | * Returns: the slot corresponding to the position @index in the | ||
| 360 | * radix tree @root. This is useful for update-if-exists operations. | ||
| 361 | * | ||
| 362 | * This function can be called under rcu_read_lock iff the slot is not | ||
| 363 | * modified by radix_tree_replace_slot, otherwise it must be called | ||
| 364 | * exclusive from other writers. Any dereference of the slot must be done | ||
| 365 | * using radix_tree_deref_slot. | ||
| 366 | */ | 357 | */ |
| 367 | void **radix_tree_lookup_slot(struct radix_tree_root *root, unsigned long index) | 358 | static void *radix_tree_lookup_element(struct radix_tree_root *root, |
| 359 | unsigned long index, int is_slot) | ||
| 368 | { | 360 | { |
| 369 | unsigned int height, shift; | 361 | unsigned int height, shift; |
| 370 | struct radix_tree_node *node, **slot; | 362 | struct radix_tree_node *node, **slot; |
| @@ -376,7 +368,7 @@ void **radix_tree_lookup_slot(struct radix_tree_root *root, unsigned long index) | |||
| 376 | if (!radix_tree_is_indirect_ptr(node)) { | 368 | if (!radix_tree_is_indirect_ptr(node)) { |
| 377 | if (index > 0) | 369 | if (index > 0) |
| 378 | return NULL; | 370 | return NULL; |
| 379 | return (void **)&root->rnode; | 371 | return is_slot ? (void *)&root->rnode : node; |
| 380 | } | 372 | } |
| 381 | node = radix_tree_indirect_to_ptr(node); | 373 | node = radix_tree_indirect_to_ptr(node); |
| 382 | 374 | ||
| @@ -397,7 +389,25 @@ void **radix_tree_lookup_slot(struct radix_tree_root *root, unsigned long index) | |||
| 397 | height--; | 389 | height--; |
| 398 | } while (height > 0); | 390 | } while (height > 0); |
| 399 | 391 | ||
| 400 | return (void **)slot; | 392 | return is_slot ? (void *)slot:node; |
| 393 | } | ||
| 394 | |||
| 395 | /** | ||
| 396 | * radix_tree_lookup_slot - lookup a slot in a radix tree | ||
| 397 | * @root: radix tree root | ||
| 398 | * @index: index key | ||
| 399 | * | ||
| 400 | * Returns: the slot corresponding to the position @index in the | ||
| 401 | * radix tree @root. This is useful for update-if-exists operations. | ||
| 402 | * | ||
| 403 | * This function can be called under rcu_read_lock iff the slot is not | ||
| 404 | * modified by radix_tree_replace_slot, otherwise it must be called | ||
| 405 | * exclusive from other writers. Any dereference of the slot must be done | ||
| 406 | * using radix_tree_deref_slot. | ||
| 407 | */ | ||
| 408 | void **radix_tree_lookup_slot(struct radix_tree_root *root, unsigned long index) | ||
| 409 | { | ||
| 410 | return (void **)radix_tree_lookup_element(root, index, 1); | ||
| 401 | } | 411 | } |
| 402 | EXPORT_SYMBOL(radix_tree_lookup_slot); | 412 | EXPORT_SYMBOL(radix_tree_lookup_slot); |
| 403 | 413 | ||
| @@ -415,38 +425,7 @@ EXPORT_SYMBOL(radix_tree_lookup_slot); | |||
| 415 | */ | 425 | */ |
| 416 | void *radix_tree_lookup(struct radix_tree_root *root, unsigned long index) | 426 | void *radix_tree_lookup(struct radix_tree_root *root, unsigned long index) |
| 417 | { | 427 | { |
| 418 | unsigned int height, shift; | 428 | return radix_tree_lookup_element(root, index, 0); |
| 419 | struct radix_tree_node *node, **slot; | ||
| 420 | |||
| 421 | node = rcu_dereference(root->rnode); | ||
| 422 | if (node == NULL) | ||
| 423 | return NULL; | ||
| 424 | |||
| 425 | if (!radix_tree_is_indirect_ptr(node)) { | ||
| 426 | if (index > 0) | ||
| 427 | return NULL; | ||
| 428 | return node; | ||
| 429 | } | ||
| 430 | node = radix_tree_indirect_to_ptr(node); | ||
| 431 | |||
| 432 | height = node->height; | ||
| 433 | if (index > radix_tree_maxindex(height)) | ||
| 434 | return NULL; | ||
| 435 | |||
| 436 | shift = (height-1) * RADIX_TREE_MAP_SHIFT; | ||
| 437 | |||
| 438 | do { | ||
| 439 | slot = (struct radix_tree_node **) | ||
| 440 | (node->slots + ((index>>shift) & RADIX_TREE_MAP_MASK)); | ||
| 441 | node = rcu_dereference(*slot); | ||
| 442 | if (node == NULL) | ||
| 443 | return NULL; | ||
| 444 | |||
| 445 | shift -= RADIX_TREE_MAP_SHIFT; | ||
| 446 | height--; | ||
| 447 | } while (height > 0); | ||
| 448 | |||
| 449 | return node; | ||
| 450 | } | 429 | } |
| 451 | EXPORT_SYMBOL(radix_tree_lookup); | 430 | EXPORT_SYMBOL(radix_tree_lookup); |
| 452 | 431 | ||
