Inline genradix_ptr() for trees that only contain 1 data page. While this increases code size somewhat it should improve performance. Signed-off-by: David Laight <david.laight@xxxxxxxxxx> --- include/linux/generic-radix-tree.h | 38 ++++++++++++++++++++++++++---- lib/generic-radix-tree.c | 19 +++++---------- 2 files changed, 39 insertions(+), 18 deletions(-) diff --git a/include/linux/generic-radix-tree.h b/include/linux/generic-radix-tree.h index c486fb410855..7575862f02d5 100644 --- a/include/linux/generic-radix-tree.h +++ b/include/linux/generic-radix-tree.h @@ -42,6 +42,16 @@ #include <linux/log2.h> struct genradix_root; +struct genradix_node; + +/* The depth of the tree is encoded in low bits of __genradix.root. */ +#define GENRADIX_SHIFT_MASK 0xff +#define GENRADIX_ROOT_SHIFT ilog2(sizeof(struct genradix_node *)) + +static inline unsigned int __genradix_root_to_shift(struct genradix_root *root) +{ + return (unsigned long)root & GENRADIX_SHIFT_MASK; +} struct __genradix { struct genradix_root *root; @@ -98,11 +108,12 @@ void __genradix_free(struct __genradix *); */ #define genradix_free(_radix) __genradix_free(&(_radix)->tree) +#define __genradix_offset_adjust(idx, obj_size) \ + ((PAGE_SIZE % obj_size) * (idx / (PAGE_SIZE / obj_size))) + static inline size_t __idx_to_offset(size_t idx, size_t obj_size) { - size_t objs_per_page = PAGE_SIZE / obj_size; - - return idx * obj_size + (idx / objs_per_page) * (PAGE_SIZE % obj_size); + return idx * obj_size + __genradix_offset_adjust(idx, obj_size); } #define __genradix_cast(_radix) (typeof((_radix)->type[0]) *) @@ -112,6 +123,23 @@ static inline size_t __idx_to_offset(size_t idx, size_t obj_size) void *__genradix_ptr(struct genradix_root *, size_t); +static inline void *__genradix_ptr1(struct genradix_root *root, size_t idx, + size_t sz) +{ + size_t offset = idx * sz; + + /* Optimise accesses into small trees */ + if (likely(offset <= PAGE_SIZE - sz)) { + if (likely(__genradix_root_to_shift(root) == GENRADIX_ROOT_SHIFT)) + return (u8 *)root - GENRADIX_ROOT_SHIFT + offset; + } else { + offset += __genradix_offset_adjust(idx, sz); + } + + return __genradix_ptr(root, offset); +} + + /** * genradix_ptr - get a pointer to a genradix entry * @_radix: genradix to access @@ -121,8 +149,8 @@ void *__genradix_ptr(struct genradix_root *, size_t); */ #define genradix_ptr(_radix, _idx) \ (__genradix_cast(_radix) \ - __genradix_ptr(READ_ONCE(_radix)->tree.root), \ - __genradix_idx_to_offset(_radix, _idx))) + __genradix_ptr1(READ_ONCE((_radix)->tree.root), _idx, \ + __genradix_obj_size(_radix))) void *__genradix_ptr_alloc(struct __genradix *, size_t, gfp_t); diff --git a/lib/generic-radix-tree.c b/lib/generic-radix-tree.c index 037a6456a17b..7029a14eed97 100644 --- a/lib/generic-radix-tree.c +++ b/lib/generic-radix-tree.c @@ -6,7 +6,6 @@ #define GENRADIX_ARY (PAGE_SIZE / sizeof(struct genradix_node *)) #define GENRADIX_ARY_SHIFT ilog2(GENRADIX_ARY) -#define GENRADIX_ROOT_SHIFT ilog2(sizeof(struct genradix_node *)) struct genradix_node { union { @@ -30,12 +29,6 @@ struct genradix_node { * With 4k pages on a 64bit system the values are 3, 12, 21 etc. * A 0 shift only ever happens when the pointer is NULL. */ -#define GENRADIX_SHIFT_MASK 0xff - -static inline unsigned genradix_root_to_shift(struct genradix_root *r) -{ - return (unsigned long)r & GENRADIX_SHIFT_MASK; -} static inline struct genradix_node *genradix_root_to_node(struct genradix_root *r) { @@ -49,7 +42,7 @@ static inline struct genradix_node *genradix_root_to_node(struct genradix_root * void *__genradix_ptr(struct genradix_root *r, size_t offset) { struct genradix_node *n = genradix_root_to_node(r); - unsigned int shift = genradix_root_to_shift(r); + unsigned int shift = __genradix_root_to_shift(r); unsigned int idx; if (likely(shift == GENRADIX_ROOT_SHIFT)) @@ -138,7 +131,7 @@ void *__genradix_ptr_alloc(struct __genradix *radix, size_t offset, gfp_t gfp) unsigned int idx; r = READ_ONCE(radix->root); - shift = genradix_root_to_shift(r); + shift = __genradix_root_to_shift(r); n = genradix_root_to_node(r); /* Optimise non-tree structures */ @@ -150,7 +143,7 @@ void *__genradix_ptr_alloc(struct __genradix *radix, size_t offset, gfp_t gfp) r = alloc_root(&radix->root, NULL, NULL, GENRADIX_ROOT_SHIFT, gfp); if (!r) return NULL; - shift = genradix_root_to_shift(r); + shift = __genradix_root_to_shift(r); n = genradix_root_to_node(r); if (shift == GENRADIX_ROOT_SHIFT) return n->data + offset; @@ -173,7 +166,7 @@ void *__genradix_ptr_alloc(struct __genradix *radix, size_t offset, gfp_t gfp) return NULL; /* Many levels could have been added by someone else. */ - shift = genradix_root_to_shift(r); + shift = __genradix_root_to_shift(r); n = genradix_root_to_node(r); } } @@ -210,7 +203,7 @@ void *__genradix_iter_peek(struct genradix_iter *iter, return NULL; n = genradix_root_to_node(r); - shift = genradix_root_to_shift(r); + shift = __genradix_root_to_shift(r); if (iter->offset >> shift) return NULL; @@ -268,6 +261,6 @@ void __genradix_free(struct __genradix *radix) struct genradix_root *r = xchg(&radix->root, NULL); genradix_free_recurse(genradix_root_to_node(r), - genradix_root_to_shift(r)); + __genradix_root_to_shift(r)); } EXPORT_SYMBOL(__genradix_free); -- 2.25.1 - Registered Address Lakeside, Bramley Road, Mount Farm, Milton Keynes, MK1 1PT, UK Registration No: 1397386 (Wales)