rbtree: add RB_DECLARE_CALLBACKS() macro
authorMichel Lespinasse <walken@google.com>
Mon, 8 Oct 2012 23:31:21 +0000 (16:31 -0700)
committerLinus Torvalds <torvalds@linux-foundation.org>
Tue, 9 Oct 2012 07:22:38 +0000 (16:22 +0900)
As proposed by Peter Zijlstra, this makes it easier to define the augmented
rbtree callbacks.

Signed-off-by: Michel Lespinasse <walken@google.com>
Cc: Rik van Riel <riel@redhat.com>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: David Woodhouse <dwmw2@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
arch/x86/mm/pat_rbtree.c
include/linux/rbtree.h
lib/rbtree_test.c

index 7e1515bd47700375b437d66209cac901480afb3f..4d116959075dccedc87c46e17e5ecf9256999bd6 100644 (file)
@@ -69,41 +69,8 @@ static u64 compute_subtree_max_end(struct memtype *data)
        return max_end;
 }
 
-/* Update 'subtree_max_end' for node and its parents */
-static void memtype_rb_propagate_cb(struct rb_node *node, struct rb_node *stop)
-{
-       while (node != stop) {
-               struct memtype *data = container_of(node, struct memtype, rb);
-               u64 subtree_max_end = compute_subtree_max_end(data);
-               if (data->subtree_max_end == subtree_max_end)
-                       break;
-               data->subtree_max_end = subtree_max_end;
-               node = rb_parent(&data->rb);
-       }
-}
-
-static void memtype_rb_copy_cb(struct rb_node *old, struct rb_node *new)
-{
-       struct memtype *old_data = container_of(old, struct memtype, rb);
-       struct memtype *new_data = container_of(new, struct memtype, rb);
-
-       new_data->subtree_max_end = old_data->subtree_max_end;
-}
-
-/* Update 'subtree_max_end' after tree rotation. old and new are the
- * former and current subtree roots */
-static void memtype_rb_rotate_cb(struct rb_node *old, struct rb_node *new)
-{
-       struct memtype *old_data = container_of(old, struct memtype, rb);
-       struct memtype *new_data = container_of(new, struct memtype, rb);
-
-       new_data->subtree_max_end = old_data->subtree_max_end;
-       old_data->subtree_max_end = compute_subtree_max_end(old_data);
-}
-
-static const struct rb_augment_callbacks memtype_rb_augment_cb = {
-       memtype_rb_propagate_cb, memtype_rb_copy_cb, memtype_rb_rotate_cb
-};
+RB_DECLARE_CALLBACKS(static, memtype_rb_augment_cb, struct memtype, rb,
+                    u64, subtree_max_end, compute_subtree_max_end)
 
 /* Find the first (lowest start addr) overlapping range from rb tree */
 static struct memtype *memtype_rb_lowest_match(struct rb_root *root,
index 4ace31b33380e5672f59d16c07592392bf9511bd..8d1e83b1c87b6a6c45f3f1aeae2ceb807a8e7b61 100644 (file)
@@ -79,6 +79,36 @@ rb_insert_augmented(struct rb_node *node, struct rb_root *root,
        __rb_insert_augmented(node, root, augment->rotate);
 }
 
+#define RB_DECLARE_CALLBACKS(rbstatic, rbname, rbstruct, rbfield,            \
+                            rbtype, rbaugmented, rbcompute)                  \
+static void rbname ## _propagate(struct rb_node *rb, struct rb_node *stop)    \
+{                                                                            \
+       while (rb != stop) {                                                  \
+               rbstruct *node = rb_entry(rb, rbstruct, rbfield);             \
+               rbtype augmented = rbcompute(node);                           \
+               if (node->rbaugmented == augmented)                           \
+                       break;                                                \
+               node->rbaugmented = augmented;                                \
+               rb = rb_parent(&node->rbfield);                               \
+       }                                                                     \
+}                                                                            \
+static void rbname ## _copy(struct rb_node *rb_old, struct rb_node *rb_new)   \
+{                                                                            \
+       rbstruct *old = rb_entry(rb_old, rbstruct, rbfield);                  \
+       rbstruct *new = rb_entry(rb_new, rbstruct, rbfield);                  \
+       new->rbaugmented = old->rbaugmented;                                  \
+}                                                                            \
+static void rbname ## _rotate(struct rb_node *rb_old, struct rb_node *rb_new) \
+{                                                                            \
+       rbstruct *old = rb_entry(rb_old, rbstruct, rbfield);                  \
+       rbstruct *new = rb_entry(rb_new, rbstruct, rbfield);                  \
+       new->rbaugmented = old->rbaugmented;                                  \
+       old->rbaugmented = rbcompute(old);                                    \
+}                                                                            \
+rbstatic const struct rb_augment_callbacks rbname = {                        \
+       rbname ## _propagate, rbname ## _copy, rbname ## _rotate              \
+};
+
 
 /* Find logical next and previous nodes in a tree */
 extern struct rb_node *rb_next(const struct rb_node *);
index e28345df09bff66a87fa79d8de4648b06af6f213..b20e99969b0ff59488a3c4cc57bcf5a4e16a57cd 100644 (file)
@@ -61,38 +61,8 @@ static inline u32 augment_recompute(struct test_node *node)
        return max;
 }
 
-static void augment_propagate(struct rb_node *rb, struct rb_node *stop)
-{
-       while (rb != stop) {
-               struct test_node *node = rb_entry(rb, struct test_node, rb);
-               u32 augmented = augment_recompute(node);
-               if (node->augmented == augmented)
-                       break;
-               node->augmented = augmented;
-               rb = rb_parent(&node->rb);
-       }
-}
-
-static void augment_copy(struct rb_node *rb_old, struct rb_node *rb_new)
-{
-       struct test_node *old = rb_entry(rb_old, struct test_node, rb);
-       struct test_node *new = rb_entry(rb_new, struct test_node, rb);
-       new->augmented = old->augmented;
-}
-
-static void augment_rotate(struct rb_node *rb_old, struct rb_node *rb_new)
-{
-       struct test_node *old = rb_entry(rb_old, struct test_node, rb);
-       struct test_node *new = rb_entry(rb_new, struct test_node, rb);
-
-       /* Rotation doesn't change subtree's augmented value */
-       new->augmented = old->augmented;
-       old->augmented = augment_recompute(old);
-}
-
-static const struct rb_augment_callbacks augment_callbacks = {
-       augment_propagate, augment_copy, augment_rotate
-};
+RB_DECLARE_CALLBACKS(static, augment_callbacks, struct test_node, rb,
+                    u32, augmented, augment_recompute)
 
 static void insert_augmented(struct test_node *node, struct rb_root *root)
 {