diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-16 22:50:13 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-16 22:50:13 -0400 |
| commit | 517d08699b250021303f9a7cf0d758b6dc0748ed (patch) | |
| tree | 5e5b0134c3fffb78fe9d8b1641a64ff28fdd7bbc /lib | |
| parent | 8eeee4e2f04fc551f50c9d9847da2d73d7d33728 (diff) | |
| parent | a34601c5d84134055782ee031d58d82f5440e918 (diff) | |
Merge branch 'akpm'
* akpm: (182 commits)
fbdev: bf54x-lq043fb: use kzalloc over kmalloc/memset
fbdev: *bfin*: fix __dev{init,exit} markings
fbdev: *bfin*: drop unnecessary calls to memset
fbdev: bfin-t350mcqb-fb: drop unused local variables
fbdev: blackfin has __raw I/O accessors, so use them in fb.h
fbdev: s1d13xxxfb: add accelerated bitblt functions
tcx: use standard fields for framebuffer physical address and length
fbdev: add support for handoff from firmware to hw framebuffers
intelfb: fix a bug when changing video timing
fbdev: use framebuffer_release() for freeing fb_info structures
radeon: P2G2CLK_ALWAYS_ONb tested twice, should 2nd be P2G2CLK_DAC_ALWAYS_ONb?
s3c-fb: CPUFREQ frequency scaling support
s3c-fb: fix resource releasing on error during probing
carminefb: fix possible access beyond end of carmine_modedb[]
acornfb: remove fb_mmap function
mb862xxfb: use CONFIG_OF instead of CONFIG_PPC_OF
mb862xxfb: restrict compliation of platform driver to PPC
Samsung SoC Framebuffer driver: add Alpha Channel support
atmel-lcdc: fix pixclock upper bound detection
offb: use framebuffer_alloc() to allocate fb_info struct
...
Manually fix up conflicts due to kmemcheck in mm/slab.c
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/dec_and_lock.c | 3 | ||||
| -rw-r--r-- | lib/genalloc.c | 1 | ||||
| -rw-r--r-- | lib/hexdump.c | 15 | ||||
| -rw-r--r-- | lib/radix-tree.c | 110 | ||||
| -rw-r--r-- | lib/rbtree.c | 34 |
5 files changed, 90 insertions, 73 deletions
diff --git a/lib/dec_and_lock.c b/lib/dec_and_lock.c index a65c31455541..e73822aa6e9a 100644 --- a/lib/dec_and_lock.c +++ b/lib/dec_and_lock.c | |||
| @@ -19,11 +19,10 @@ | |||
| 19 | */ | 19 | */ |
| 20 | int _atomic_dec_and_lock(atomic_t *atomic, spinlock_t *lock) | 20 | int _atomic_dec_and_lock(atomic_t *atomic, spinlock_t *lock) |
| 21 | { | 21 | { |
| 22 | #ifdef CONFIG_SMP | ||
| 23 | /* Subtract 1 from counter unless that drops it to 0 (ie. it was 1) */ | 22 | /* Subtract 1 from counter unless that drops it to 0 (ie. it was 1) */ |
| 24 | if (atomic_add_unless(atomic, -1, 1)) | 23 | if (atomic_add_unless(atomic, -1, 1)) |
| 25 | return 0; | 24 | return 0; |
| 26 | #endif | 25 | |
| 27 | /* Otherwise do it the slow way */ | 26 | /* Otherwise do it the slow way */ |
| 28 | spin_lock(lock); | 27 | spin_lock(lock); |
| 29 | if (atomic_dec_and_test(atomic)) | 28 | if (atomic_dec_and_test(atomic)) |
diff --git a/lib/genalloc.c b/lib/genalloc.c index f6d276db2d58..eed2bdb865e7 100644 --- a/lib/genalloc.c +++ b/lib/genalloc.c | |||
| @@ -85,7 +85,6 @@ void gen_pool_destroy(struct gen_pool *pool) | |||
| 85 | int bit, end_bit; | 85 | int bit, end_bit; |
| 86 | 86 | ||
| 87 | 87 | ||
| 88 | write_lock(&pool->lock); | ||
| 89 | list_for_each_safe(_chunk, _next_chunk, &pool->chunks) { | 88 | list_for_each_safe(_chunk, _next_chunk, &pool->chunks) { |
| 90 | chunk = list_entry(_chunk, struct gen_pool_chunk, next_chunk); | 89 | chunk = list_entry(_chunk, struct gen_pool_chunk, next_chunk); |
| 91 | list_del(&chunk->next_chunk); | 90 | list_del(&chunk->next_chunk); |
diff --git a/lib/hexdump.c b/lib/hexdump.c index f07c0db81d26..39af2560f765 100644 --- a/lib/hexdump.c +++ b/lib/hexdump.c | |||
| @@ -65,7 +65,8 @@ void hex_dump_to_buffer(const void *buf, size_t len, int rowsize, | |||
| 65 | 65 | ||
| 66 | for (j = 0; j < ngroups; j++) | 66 | for (j = 0; j < ngroups; j++) |
| 67 | lx += scnprintf(linebuf + lx, linebuflen - lx, | 67 | lx += scnprintf(linebuf + lx, linebuflen - lx, |
| 68 | "%16.16llx ", (unsigned long long)*(ptr8 + j)); | 68 | "%s%16.16llx", j ? " " : "", |
| 69 | (unsigned long long)*(ptr8 + j)); | ||
| 69 | ascii_column = 17 * ngroups + 2; | 70 | ascii_column = 17 * ngroups + 2; |
| 70 | break; | 71 | break; |
| 71 | } | 72 | } |
| @@ -76,7 +77,7 @@ void hex_dump_to_buffer(const void *buf, size_t len, int rowsize, | |||
| 76 | 77 | ||
| 77 | for (j = 0; j < ngroups; j++) | 78 | for (j = 0; j < ngroups; j++) |
| 78 | lx += scnprintf(linebuf + lx, linebuflen - lx, | 79 | lx += scnprintf(linebuf + lx, linebuflen - lx, |
| 79 | "%8.8x ", *(ptr4 + j)); | 80 | "%s%8.8x", j ? " " : "", *(ptr4 + j)); |
| 80 | ascii_column = 9 * ngroups + 2; | 81 | ascii_column = 9 * ngroups + 2; |
| 81 | break; | 82 | break; |
| 82 | } | 83 | } |
| @@ -87,19 +88,21 @@ void hex_dump_to_buffer(const void *buf, size_t len, int rowsize, | |||
| 87 | 88 | ||
| 88 | for (j = 0; j < ngroups; j++) | 89 | for (j = 0; j < ngroups; j++) |
| 89 | lx += scnprintf(linebuf + lx, linebuflen - lx, | 90 | lx += scnprintf(linebuf + lx, linebuflen - lx, |
| 90 | "%4.4x ", *(ptr2 + j)); | 91 | "%s%4.4x", j ? " " : "", *(ptr2 + j)); |
| 91 | ascii_column = 5 * ngroups + 2; | 92 | ascii_column = 5 * ngroups + 2; |
| 92 | break; | 93 | break; |
| 93 | } | 94 | } |
| 94 | 95 | ||
| 95 | default: | 96 | default: |
| 96 | for (j = 0; (j < rowsize) && (j < len) && (lx + 4) < linebuflen; | 97 | for (j = 0; (j < len) && (lx + 3) <= linebuflen; j++) { |
| 97 | j++) { | ||
| 98 | ch = ptr[j]; | 98 | ch = ptr[j]; |
| 99 | linebuf[lx++] = hex_asc_hi(ch); | 99 | linebuf[lx++] = hex_asc_hi(ch); |
| 100 | linebuf[lx++] = hex_asc_lo(ch); | 100 | linebuf[lx++] = hex_asc_lo(ch); |
| 101 | linebuf[lx++] = ' '; | 101 | linebuf[lx++] = ' '; |
| 102 | } | 102 | } |
| 103 | if (j) | ||
| 104 | lx--; | ||
| 105 | |||
| 103 | ascii_column = 3 * rowsize + 2; | 106 | ascii_column = 3 * rowsize + 2; |
| 104 | break; | 107 | break; |
| 105 | } | 108 | } |
| @@ -108,7 +111,7 @@ void hex_dump_to_buffer(const void *buf, size_t len, int rowsize, | |||
| 108 | 111 | ||
| 109 | while (lx < (linebuflen - 1) && lx < (ascii_column - 1)) | 112 | while (lx < (linebuflen - 1) && lx < (ascii_column - 1)) |
| 110 | linebuf[lx++] = ' '; | 113 | linebuf[lx++] = ' '; |
| 111 | for (j = 0; (j < rowsize) && (j < len) && (lx + 2) < linebuflen; j++) | 114 | for (j = 0; (j < len) && (lx + 2) < linebuflen; j++) |
| 112 | linebuf[lx++] = (isascii(ptr[j]) && isprint(ptr[j])) ? ptr[j] | 115 | linebuf[lx++] = (isascii(ptr[j]) && isprint(ptr[j])) ? ptr[j] |
| 113 | : '.'; | 116 | : '.'; |
| 114 | nil: | 117 | nil: |
diff --git a/lib/radix-tree.c b/lib/radix-tree.c index 4bb42a0344ec..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 | ||
| @@ -666,6 +645,43 @@ unsigned long radix_tree_next_hole(struct radix_tree_root *root, | |||
| 666 | } | 645 | } |
| 667 | EXPORT_SYMBOL(radix_tree_next_hole); | 646 | EXPORT_SYMBOL(radix_tree_next_hole); |
| 668 | 647 | ||
| 648 | /** | ||
| 649 | * radix_tree_prev_hole - find the prev hole (not-present entry) | ||
| 650 | * @root: tree root | ||
| 651 | * @index: index key | ||
| 652 | * @max_scan: maximum range to search | ||
| 653 | * | ||
| 654 | * Search backwards in the range [max(index-max_scan+1, 0), index] | ||
| 655 | * for the first hole. | ||
| 656 | * | ||
| 657 | * Returns: the index of the hole if found, otherwise returns an index | ||
| 658 | * outside of the set specified (in which case 'index - return >= max_scan' | ||
| 659 | * will be true). In rare cases of wrap-around, LONG_MAX will be returned. | ||
| 660 | * | ||
| 661 | * radix_tree_next_hole may be called under rcu_read_lock. However, like | ||
| 662 | * radix_tree_gang_lookup, this will not atomically search a snapshot of | ||
| 663 | * the tree at a single point in time. For example, if a hole is created | ||
| 664 | * at index 10, then subsequently a hole is created at index 5, | ||
| 665 | * radix_tree_prev_hole covering both indexes may return 5 if called under | ||
| 666 | * rcu_read_lock. | ||
| 667 | */ | ||
| 668 | unsigned long radix_tree_prev_hole(struct radix_tree_root *root, | ||
| 669 | unsigned long index, unsigned long max_scan) | ||
| 670 | { | ||
| 671 | unsigned long i; | ||
| 672 | |||
| 673 | for (i = 0; i < max_scan; i++) { | ||
| 674 | if (!radix_tree_lookup(root, index)) | ||
| 675 | break; | ||
| 676 | index--; | ||
| 677 | if (index == LONG_MAX) | ||
| 678 | break; | ||
| 679 | } | ||
| 680 | |||
| 681 | return index; | ||
| 682 | } | ||
| 683 | EXPORT_SYMBOL(radix_tree_prev_hole); | ||
| 684 | |||
| 669 | static unsigned int | 685 | static unsigned int |
| 670 | __lookup(struct radix_tree_node *slot, void ***results, unsigned long index, | 686 | __lookup(struct radix_tree_node *slot, void ***results, unsigned long index, |
| 671 | unsigned int max_items, unsigned long *next_index) | 687 | unsigned int max_items, unsigned long *next_index) |
diff --git a/lib/rbtree.c b/lib/rbtree.c index f653659e0bc1..e2aa3be29858 100644 --- a/lib/rbtree.c +++ b/lib/rbtree.c | |||
| @@ -231,34 +231,34 @@ void rb_erase(struct rb_node *node, struct rb_root *root) | |||
| 231 | node = node->rb_right; | 231 | node = node->rb_right; |
| 232 | while ((left = node->rb_left) != NULL) | 232 | while ((left = node->rb_left) != NULL) |
| 233 | node = left; | 233 | node = left; |
| 234 | |||
| 235 | if (rb_parent(old)) { | ||
| 236 | if (rb_parent(old)->rb_left == old) | ||
| 237 | rb_parent(old)->rb_left = node; | ||
| 238 | else | ||
| 239 | rb_parent(old)->rb_right = node; | ||
| 240 | } else | ||
| 241 | root->rb_node = node; | ||
| 242 | |||
| 234 | child = node->rb_right; | 243 | child = node->rb_right; |
| 235 | parent = rb_parent(node); | 244 | parent = rb_parent(node); |
| 236 | color = rb_color(node); | 245 | color = rb_color(node); |
| 237 | 246 | ||
| 238 | if (child) | ||
| 239 | rb_set_parent(child, parent); | ||
| 240 | if (parent == old) { | 247 | if (parent == old) { |
| 241 | parent->rb_right = child; | ||
| 242 | parent = node; | 248 | parent = node; |
| 243 | } else | 249 | } else { |
| 250 | if (child) | ||
| 251 | rb_set_parent(child, parent); | ||
| 244 | parent->rb_left = child; | 252 | parent->rb_left = child; |
| 245 | 253 | ||
| 254 | node->rb_right = old->rb_right; | ||
| 255 | rb_set_parent(old->rb_right, node); | ||
| 256 | } | ||
| 257 | |||
| 246 | node->rb_parent_color = old->rb_parent_color; | 258 | node->rb_parent_color = old->rb_parent_color; |
| 247 | node->rb_right = old->rb_right; | ||
| 248 | node->rb_left = old->rb_left; | 259 | node->rb_left = old->rb_left; |
| 249 | |||
| 250 | if (rb_parent(old)) | ||
| 251 | { | ||
| 252 | if (rb_parent(old)->rb_left == old) | ||
| 253 | rb_parent(old)->rb_left = node; | ||
| 254 | else | ||
| 255 | rb_parent(old)->rb_right = node; | ||
| 256 | } else | ||
| 257 | root->rb_node = node; | ||
| 258 | |||
| 259 | rb_set_parent(old->rb_left, node); | 260 | rb_set_parent(old->rb_left, node); |
| 260 | if (old->rb_right) | 261 | |
| 261 | rb_set_parent(old->rb_right, node); | ||
| 262 | goto color; | 262 | goto color; |
| 263 | } | 263 | } |
| 264 | 264 | ||
