aboutsummaryrefslogtreecommitdiffstats
path: root/mm/interval_tree.c
diff options
context:
space:
mode:
authorMichel Lespinasse <walken@google.com>2012-10-08 19:31:35 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2012-10-09 03:22:40 -0400
commit9826a516ff77c5820e591211e4f3e58ff36f46be (patch)
treebdec1e2fe5ff95569795069bac73977faba17d57 /mm/interval_tree.c
parent9c079add0d0f45220f4bb37febf0621137ec2d38 (diff)
mm: interval tree updates
Update the generic interval tree code that was introduced in "mm: replace vma prio_tree with an interval tree". Changes: - fixed 'endpoing' typo noticed by Andrew Morton - replaced include/linux/interval_tree_tmpl.h, which was used as a template (including it automatically defined the interval tree functions) with include/linux/interval_tree_generic.h, which only defines a preprocessor macro INTERVAL_TREE_DEFINE(), which itself defines the interval tree functions when invoked. Now that is a very long macro which is unfortunate, but it does make the usage sites (lib/interval_tree.c and mm/interval_tree.c) a bit nicer than previously. - make use of RB_DECLARE_CALLBACKS() in the INTERVAL_TREE_DEFINE() macro, instead of duplicating that code in the interval tree template. - replaced vma_interval_tree_add(), which was actually handling the nonlinear and interval tree cases, with vma_interval_tree_insert_after() which handles only the interval tree case and has an API that is more consistent with the other interval tree handling functions. The nonlinear case is now handled explicitly in kernel/fork.c dup_mmap(). Signed-off-by: Michel Lespinasse <walken@google.com> Cc: Andrea Arcangeli <aarcange@redhat.com> Cc: Rik van Riel <riel@redhat.com> Cc: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Daniel Santos <daniel.santos@pobox.com> Cc: Hugh Dickins <hughd@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/interval_tree.c')
-rw-r--r--mm/interval_tree.c60
1 files changed, 29 insertions, 31 deletions
diff --git a/mm/interval_tree.c b/mm/interval_tree.c
index 7dc565660e56..4ab7b9ec3a56 100644
--- a/mm/interval_tree.c
+++ b/mm/interval_tree.c
@@ -8,40 +8,38 @@
8 8
9#include <linux/mm.h> 9#include <linux/mm.h>
10#include <linux/fs.h> 10#include <linux/fs.h>
11#include <linux/interval_tree_generic.h>
11 12
12#define ITSTRUCT struct vm_area_struct 13static inline unsigned long vma_start_pgoff(struct vm_area_struct *v)
13#define ITRB shared.linear.rb 14{
14#define ITTYPE unsigned long 15 return v->vm_pgoff;
15#define ITSUBTREE shared.linear.rb_subtree_last 16}
16#define ITSTART(n) ((n)->vm_pgoff) 17
17#define ITLAST(n) ((n)->vm_pgoff + \ 18static inline unsigned long vma_last_pgoff(struct vm_area_struct *v)
18 (((n)->vm_end - (n)->vm_start) >> PAGE_SHIFT) - 1) 19{
19#define ITSTATIC 20 return v->vm_pgoff + ((v->vm_end - v->vm_start) >> PAGE_SHIFT) - 1;
20#define ITPREFIX vma_interval_tree 21}
21 22
22#include <linux/interval_tree_tmpl.h> 23INTERVAL_TREE_DEFINE(struct vm_area_struct, shared.linear.rb,
23 24 unsigned long, shared.linear.rb_subtree_last,
24/* Insert old immediately after vma in the interval tree */ 25 vma_start_pgoff, vma_last_pgoff,, vma_interval_tree)
25void vma_interval_tree_add(struct vm_area_struct *vma, 26
26 struct vm_area_struct *old, 27/* Insert node immediately after prev in the interval tree */
27 struct address_space *mapping) 28void vma_interval_tree_insert_after(struct vm_area_struct *node,
29 struct vm_area_struct *prev,
30 struct rb_root *root)
28{ 31{
29 struct rb_node **link; 32 struct rb_node **link;
30 struct vm_area_struct *parent; 33 struct vm_area_struct *parent;
31 unsigned long last; 34 unsigned long last = vma_last_pgoff(node);
32
33 if (unlikely(vma->vm_flags & VM_NONLINEAR)) {
34 list_add(&vma->shared.nonlinear, &old->shared.nonlinear);
35 return;
36 }
37 35
38 last = ITLAST(vma); 36 VM_BUG_ON(vma_start_pgoff(node) != vma_start_pgoff(prev));
39 37
40 if (!old->shared.linear.rb.rb_right) { 38 if (!prev->shared.linear.rb.rb_right) {
41 parent = old; 39 parent = prev;
42 link = &old->shared.linear.rb.rb_right; 40 link = &prev->shared.linear.rb.rb_right;
43 } else { 41 } else {
44 parent = rb_entry(old->shared.linear.rb.rb_right, 42 parent = rb_entry(prev->shared.linear.rb.rb_right,
45 struct vm_area_struct, shared.linear.rb); 43 struct vm_area_struct, shared.linear.rb);
46 if (parent->shared.linear.rb_subtree_last < last) 44 if (parent->shared.linear.rb_subtree_last < last)
47 parent->shared.linear.rb_subtree_last = last; 45 parent->shared.linear.rb_subtree_last = last;
@@ -54,8 +52,8 @@ void vma_interval_tree_add(struct vm_area_struct *vma,
54 link = &parent->shared.linear.rb.rb_left; 52 link = &parent->shared.linear.rb.rb_left;
55 } 53 }
56 54
57 vma->shared.linear.rb_subtree_last = last; 55 node->shared.linear.rb_subtree_last = last;
58 rb_link_node(&vma->shared.linear.rb, &parent->shared.linear.rb, link); 56 rb_link_node(&node->shared.linear.rb, &parent->shared.linear.rb, link);
59 rb_insert_augmented(&vma->shared.linear.rb, &mapping->i_mmap, 57 rb_insert_augmented(&node->shared.linear.rb, root,
60 &vma_interval_tree_augment_callbacks); 58 &vma_interval_tree_augment);
61} 59}