diff options
Diffstat (limited to 'fs/ceph/snap.c')
-rw-r--r-- | fs/ceph/snap.c | 51 |
1 files changed, 40 insertions, 11 deletions
diff --git a/fs/ceph/snap.c b/fs/ceph/snap.c index dcf18d92130a..49d0c4c59d81 100644 --- a/fs/ceph/snap.c +++ b/fs/ceph/snap.c | |||
@@ -1,6 +1,5 @@ | |||
1 | #include "ceph_debug.h" | 1 | #include "ceph_debug.h" |
2 | 2 | ||
3 | #include <linux/radix-tree.h> | ||
4 | #include <linux/sort.h> | 3 | #include <linux/sort.h> |
5 | 4 | ||
6 | #include "super.h" | 5 | #include "super.h" |
@@ -77,6 +76,28 @@ void ceph_get_snap_realm(struct ceph_mds_client *mdsc, | |||
77 | atomic_inc(&realm->nref); | 76 | atomic_inc(&realm->nref); |
78 | } | 77 | } |
79 | 78 | ||
79 | static void __insert_snap_realm(struct rb_root *root, | ||
80 | struct ceph_snap_realm *new) | ||
81 | { | ||
82 | struct rb_node **p = &root->rb_node; | ||
83 | struct rb_node *parent = NULL; | ||
84 | struct ceph_snap_realm *r = NULL; | ||
85 | |||
86 | while (*p) { | ||
87 | parent = *p; | ||
88 | r = rb_entry(parent, struct ceph_snap_realm, node); | ||
89 | if (new->ino < r->ino) | ||
90 | p = &(*p)->rb_left; | ||
91 | else if (new->ino > r->ino) | ||
92 | p = &(*p)->rb_right; | ||
93 | else | ||
94 | BUG(); | ||
95 | } | ||
96 | |||
97 | rb_link_node(&new->node, parent, p); | ||
98 | rb_insert_color(&new->node, root); | ||
99 | } | ||
100 | |||
80 | /* | 101 | /* |
81 | * create and get the realm rooted at @ino and bump its ref count. | 102 | * create and get the realm rooted at @ino and bump its ref count. |
82 | * | 103 | * |
@@ -92,8 +113,6 @@ static struct ceph_snap_realm *ceph_create_snap_realm( | |||
92 | if (!realm) | 113 | if (!realm) |
93 | return ERR_PTR(-ENOMEM); | 114 | return ERR_PTR(-ENOMEM); |
94 | 115 | ||
95 | radix_tree_insert(&mdsc->snap_realms, ino, realm); | ||
96 | |||
97 | atomic_set(&realm->nref, 0); /* tree does not take a ref */ | 116 | atomic_set(&realm->nref, 0); /* tree does not take a ref */ |
98 | realm->ino = ino; | 117 | realm->ino = ino; |
99 | INIT_LIST_HEAD(&realm->children); | 118 | INIT_LIST_HEAD(&realm->children); |
@@ -101,24 +120,34 @@ static struct ceph_snap_realm *ceph_create_snap_realm( | |||
101 | INIT_LIST_HEAD(&realm->empty_item); | 120 | INIT_LIST_HEAD(&realm->empty_item); |
102 | INIT_LIST_HEAD(&realm->inodes_with_caps); | 121 | INIT_LIST_HEAD(&realm->inodes_with_caps); |
103 | spin_lock_init(&realm->inodes_with_caps_lock); | 122 | spin_lock_init(&realm->inodes_with_caps_lock); |
123 | __insert_snap_realm(&mdsc->snap_realms, realm); | ||
104 | dout("create_snap_realm %llx %p\n", realm->ino, realm); | 124 | dout("create_snap_realm %llx %p\n", realm->ino, realm); |
105 | return realm; | 125 | return realm; |
106 | } | 126 | } |
107 | 127 | ||
108 | /* | 128 | /* |
109 | * find and get (if found) the realm rooted at @ino and bump its ref count. | 129 | * lookup the realm rooted at @ino. |
110 | * | 130 | * |
111 | * caller must hold snap_rwsem for write. | 131 | * caller must hold snap_rwsem for write. |
112 | */ | 132 | */ |
113 | struct ceph_snap_realm *ceph_lookup_snap_realm(struct ceph_mds_client *mdsc, | 133 | struct ceph_snap_realm *ceph_lookup_snap_realm(struct ceph_mds_client *mdsc, |
114 | u64 ino) | 134 | u64 ino) |
115 | { | 135 | { |
116 | struct ceph_snap_realm *realm; | 136 | struct rb_node *n = mdsc->snap_realms.rb_node; |
117 | 137 | struct ceph_snap_realm *r; | |
118 | realm = radix_tree_lookup(&mdsc->snap_realms, ino); | 138 | |
119 | if (realm) | 139 | while (n) { |
120 | dout("lookup_snap_realm %llx %p\n", realm->ino, realm); | 140 | r = rb_entry(n, struct ceph_snap_realm, node); |
121 | return realm; | 141 | if (ino < r->ino) |
142 | n = n->rb_left; | ||
143 | else if (ino > r->ino) | ||
144 | n = n->rb_right; | ||
145 | else { | ||
146 | dout("lookup_snap_realm %llx %p\n", r->ino, r); | ||
147 | return r; | ||
148 | } | ||
149 | } | ||
150 | return NULL; | ||
122 | } | 151 | } |
123 | 152 | ||
124 | static void __put_snap_realm(struct ceph_mds_client *mdsc, | 153 | static void __put_snap_realm(struct ceph_mds_client *mdsc, |
@@ -132,7 +161,7 @@ static void __destroy_snap_realm(struct ceph_mds_client *mdsc, | |||
132 | { | 161 | { |
133 | dout("__destroy_snap_realm %p %llx\n", realm, realm->ino); | 162 | dout("__destroy_snap_realm %p %llx\n", realm, realm->ino); |
134 | 163 | ||
135 | radix_tree_delete(&mdsc->snap_realms, realm->ino); | 164 | rb_erase(&realm->node, &mdsc->snap_realms); |
136 | 165 | ||
137 | if (realm->parent) { | 166 | if (realm->parent) { |
138 | list_del_init(&realm->child_item); | 167 | list_del_init(&realm->child_item); |