Blame view

lib/radix-tree.c 43.3 KB
de6cc6515   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-or-later
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
  /*
   * Copyright (C) 2001 Momchil Velikov
   * Portions Copyright (C) 2001 Christoph Hellwig
cde535359   Christoph Lameter   Christoph has moved
5
   * Copyright (C) 2005 SGI, Christoph Lameter
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
6
   * Copyright (C) 2006 Nick Piggin
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
7
   * Copyright (C) 2012 Konstantin Khlebnikov
6b053b8e5   Matthew Wilcox   radix-tree: add c...
8
9
   * Copyright (C) 2016 Intel, Matthew Wilcox
   * Copyright (C) 2016 Intel, Ross Zwisler
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
10
   */
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
11
12
  #include <linux/bitmap.h>
  #include <linux/bitops.h>
460488c58   Matthew Wilcox   idr: Remove idr_a...
13
  #include <linux/bug.h>
e157b5559   Matthew Wilcox   radix-tree: add r...
14
  #include <linux/cpu.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
15
  #include <linux/errno.h>
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
16
17
  #include <linux/export.h>
  #include <linux/idr.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18
19
  #include <linux/init.h>
  #include <linux/kernel.h>
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
20
  #include <linux/kmemleak.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
21
  #include <linux/percpu.h>
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
22
23
24
  #include <linux/preempt.h>		/* in_interrupt() */
  #include <linux/radix-tree.h>
  #include <linux/rcupdate.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
  #include <linux/slab.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
26
  #include <linux/string.h>
02c02bf12   Matthew Wilcox   xarray: Change de...
27
  #include <linux/xarray.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
28

26fb1589c   Jeff Moyer   fix the max path ...
29
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
30
31
   * Radix tree node cache.
   */
58d6ea308   Matthew Wilcox   xarray: Add XArra...
32
  struct kmem_cache *radix_tree_node_cachep;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
33
34
  
  /*
553680529   Nick Piggin   radix-tree: fix p...
35
36
37
38
39
40
41
42
43
44
45
46
47
   * The radix tree is variable-height, so an insert operation not only has
   * to build the branch to its corresponding item, it also has to build the
   * branch to existing items if the size has to be increased (by
   * radix_tree_extend).
   *
   * The worst case is a zero height tree with just a single item at index 0,
   * and then inserting an item at index ULONG_MAX. This requires 2 new branches
   * of RADIX_TREE_MAX_PATH size to be created, with only the root node shared.
   * Hence:
   */
  #define RADIX_TREE_PRELOAD_SIZE (RADIX_TREE_MAX_PATH * 2 - 1)
  
  /*
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
48
49
50
51
52
53
54
55
56
   * The IDR does not have to be as high as the radix tree since it uses
   * signed integers, not unsigned longs.
   */
  #define IDR_INDEX_BITS		(8 /* CHAR_BIT */ * sizeof(int) - 1)
  #define IDR_MAX_PATH		(DIV_ROUND_UP(IDR_INDEX_BITS, \
  						RADIX_TREE_MAP_SHIFT))
  #define IDR_PRELOAD_SIZE	(IDR_MAX_PATH * 2 - 1)
  
  /*
7ad3d4d85   Matthew Wilcox   ida: Move ida_bit...
57
58
59
60
61
62
63
64
   * The IDA is even shorter since it uses a bitmap at the last level.
   */
  #define IDA_INDEX_BITS		(8 * sizeof(int) - 1 - ilog2(IDA_BITMAP_BITS))
  #define IDA_MAX_PATH		(DIV_ROUND_UP(IDA_INDEX_BITS, \
  						RADIX_TREE_MAP_SHIFT))
  #define IDA_PRELOAD_SIZE	(IDA_MAX_PATH * 2 - 1)
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
65
66
67
   * Per-cpu pool of preloaded nodes
   */
  struct radix_tree_preload {
2fcd9005c   Matthew Wilcox   radix-tree: misce...
68
  	unsigned nr;
1293d5c5f   Matthew Wilcox   radix-tree: Chain...
69
  	/* nodes->parent points to next preallocated node */
9d2a8da00   Kirill A. Shutemov   radix-tree: repla...
70
  	struct radix_tree_node *nodes;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
71
  };
8cef7d57a   Harvey Harrison   lib: radix_tree.c...
72
  static DEFINE_PER_CPU(struct radix_tree_preload, radix_tree_preloads) = { 0, };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
73

148deab22   Matthew Wilcox   radix-tree: impro...
74
75
76
77
  static inline struct radix_tree_node *entry_to_node(void *ptr)
  {
  	return (void *)((unsigned long)ptr & ~RADIX_TREE_INTERNAL_NODE);
  }
a4db4dcea   Matthew Wilcox   radix-tree: renam...
78
  static inline void *node_to_entry(void *ptr)
27d20fddc   Nick Piggin   radix-tree: fix R...
79
  {
30ff46ccb   Matthew Wilcox   radix-tree: renam...
80
  	return (void *)((unsigned long)ptr | RADIX_TREE_INTERNAL_NODE);
27d20fddc   Nick Piggin   radix-tree: fix R...
81
  }
02c02bf12   Matthew Wilcox   xarray: Change de...
82
  #define RADIX_TREE_RETRY	XA_RETRY_ENTRY
db050f292   Matthew Wilcox   radix-tree: add m...
83

d7b627277   Matthew Wilcox   radix-tree: Fix _...
84
85
  static inline unsigned long
  get_slot_offset(const struct radix_tree_node *parent, void __rcu **slot)
db050f292   Matthew Wilcox   radix-tree: add m...
86
  {
76f070b41   Matthew Wilcox   radix-tree: Fix U...
87
  	return parent ? slot - parent->slots : 0;
db050f292   Matthew Wilcox   radix-tree: add m...
88
  }
35534c869   Matthew Wilcox   radix tree: const...
89
  static unsigned int radix_tree_descend(const struct radix_tree_node *parent,
9e85d8111   Matthew Wilcox   radix-tree: make ...
90
  			struct radix_tree_node **nodep, unsigned long index)
db050f292   Matthew Wilcox   radix-tree: add m...
91
  {
9e85d8111   Matthew Wilcox   radix-tree: make ...
92
  	unsigned int offset = (index >> parent->shift) & RADIX_TREE_MAP_MASK;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
93
  	void __rcu **entry = rcu_dereference_raw(parent->slots[offset]);
db050f292   Matthew Wilcox   radix-tree: add m...
94

db050f292   Matthew Wilcox   radix-tree: add m...
95
96
97
  	*nodep = (void *)entry;
  	return offset;
  }
35534c869   Matthew Wilcox   radix tree: const...
98
  static inline gfp_t root_gfp_mask(const struct radix_tree_root *root)
612d6c19d   Nick Piggin   [PATCH] radix-tre...
99
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
100
  	return root->xa_flags & (__GFP_BITS_MASK & ~GFP_ZONEMASK);
612d6c19d   Nick Piggin   [PATCH] radix-tre...
101
  }
643b52b9c   Nick Piggin   radix-tree: fix s...
102
103
104
105
106
107
108
109
110
111
112
  static inline void tag_set(struct radix_tree_node *node, unsigned int tag,
  		int offset)
  {
  	__set_bit(offset, node->tags[tag]);
  }
  
  static inline void tag_clear(struct radix_tree_node *node, unsigned int tag,
  		int offset)
  {
  	__clear_bit(offset, node->tags[tag]);
  }
35534c869   Matthew Wilcox   radix tree: const...
113
  static inline int tag_get(const struct radix_tree_node *node, unsigned int tag,
643b52b9c   Nick Piggin   radix-tree: fix s...
114
115
116
117
  		int offset)
  {
  	return test_bit(offset, node->tags[tag]);
  }
35534c869   Matthew Wilcox   radix tree: const...
118
  static inline void root_tag_set(struct radix_tree_root *root, unsigned tag)
643b52b9c   Nick Piggin   radix-tree: fix s...
119
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
120
  	root->xa_flags |= (__force gfp_t)(1 << (tag + ROOT_TAG_SHIFT));
643b52b9c   Nick Piggin   radix-tree: fix s...
121
  }
2fcd9005c   Matthew Wilcox   radix-tree: misce...
122
  static inline void root_tag_clear(struct radix_tree_root *root, unsigned tag)
643b52b9c   Nick Piggin   radix-tree: fix s...
123
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
124
  	root->xa_flags &= (__force gfp_t)~(1 << (tag + ROOT_TAG_SHIFT));
643b52b9c   Nick Piggin   radix-tree: fix s...
125
126
127
128
  }
  
  static inline void root_tag_clear_all(struct radix_tree_root *root)
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
129
  	root->xa_flags &= (__force gfp_t)((1 << ROOT_TAG_SHIFT) - 1);
643b52b9c   Nick Piggin   radix-tree: fix s...
130
  }
35534c869   Matthew Wilcox   radix tree: const...
131
  static inline int root_tag_get(const struct radix_tree_root *root, unsigned tag)
643b52b9c   Nick Piggin   radix-tree: fix s...
132
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
133
  	return (__force int)root->xa_flags & (1 << (tag + ROOT_TAG_SHIFT));
643b52b9c   Nick Piggin   radix-tree: fix s...
134
  }
35534c869   Matthew Wilcox   radix tree: const...
135
  static inline unsigned root_tags_get(const struct radix_tree_root *root)
643b52b9c   Nick Piggin   radix-tree: fix s...
136
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
137
  	return (__force unsigned)root->xa_flags >> ROOT_TAG_SHIFT;
643b52b9c   Nick Piggin   radix-tree: fix s...
138
  }
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
139
  static inline bool is_idr(const struct radix_tree_root *root)
7b60e9ad5   Matthew Wilcox   radix-tree: fix m...
140
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
141
  	return !!(root->xa_flags & ROOT_IS_IDR);
7b60e9ad5   Matthew Wilcox   radix-tree: fix m...
142
  }
643b52b9c   Nick Piggin   radix-tree: fix s...
143
144
145
146
  /*
   * Returns 1 if any slot in the node has this tag set.
   * Otherwise returns 0.
   */
35534c869   Matthew Wilcox   radix tree: const...
147
148
  static inline int any_tag_set(const struct radix_tree_node *node,
  							unsigned int tag)
643b52b9c   Nick Piggin   radix-tree: fix s...
149
  {
2fcd9005c   Matthew Wilcox   radix-tree: misce...
150
  	unsigned idx;
643b52b9c   Nick Piggin   radix-tree: fix s...
151
152
153
154
155
156
  	for (idx = 0; idx < RADIX_TREE_TAG_LONGS; idx++) {
  		if (node->tags[tag][idx])
  			return 1;
  	}
  	return 0;
  }
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
157

0a835c4f0   Matthew Wilcox   Reimplement IDR a...
158
159
160
161
  static inline void all_tag_set(struct radix_tree_node *node, unsigned int tag)
  {
  	bitmap_fill(node->tags[tag], RADIX_TREE_MAP_SIZE);
  }
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
162
163
164
165
166
167
168
169
170
171
172
173
  /**
   * radix_tree_find_next_bit - find the next set bit in a memory region
   *
   * @addr: The address to base the search on
   * @size: The bitmap size in bits
   * @offset: The bitnumber to start searching at
   *
   * Unrollable variant of find_next_bit() for constant size arrays.
   * Tail bits starting from size to roundup(size, BITS_PER_LONG) must be zero.
   * Returns next bit offset, or size if nothing found.
   */
  static __always_inline unsigned long
bc412fca6   Matthew Wilcox   radix-tree: make ...
174
175
  radix_tree_find_next_bit(struct radix_tree_node *node, unsigned int tag,
  			 unsigned long offset)
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
176
  {
bc412fca6   Matthew Wilcox   radix-tree: make ...
177
  	const unsigned long *addr = node->tags[tag];
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
178

bc412fca6   Matthew Wilcox   radix-tree: make ...
179
  	if (offset < RADIX_TREE_MAP_SIZE) {
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
180
181
182
183
184
185
186
  		unsigned long tmp;
  
  		addr += offset / BITS_PER_LONG;
  		tmp = *addr >> (offset % BITS_PER_LONG);
  		if (tmp)
  			return __ffs(tmp) + offset;
  		offset = (offset + BITS_PER_LONG) & ~(BITS_PER_LONG - 1);
bc412fca6   Matthew Wilcox   radix-tree: make ...
187
  		while (offset < RADIX_TREE_MAP_SIZE) {
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
188
189
190
191
192
193
  			tmp = *++addr;
  			if (tmp)
  				return __ffs(tmp) + offset;
  			offset += BITS_PER_LONG;
  		}
  	}
bc412fca6   Matthew Wilcox   radix-tree: make ...
194
  	return RADIX_TREE_MAP_SIZE;
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
195
  }
268f42de7   Matthew Wilcox   radix-tree: delet...
196
197
  static unsigned int iter_offset(const struct radix_tree_iter *iter)
  {
3a08cd52c   Matthew Wilcox   radix tree: Remov...
198
  	return iter->index & RADIX_TREE_MAP_MASK;
268f42de7   Matthew Wilcox   radix-tree: delet...
199
  }
218ed7503   Matthew Wilcox   radix-tree: impro...
200
201
202
203
204
205
206
  /*
   * The maximum index which can be stored in a radix tree
   */
  static inline unsigned long shift_maxindex(unsigned int shift)
  {
  	return (RADIX_TREE_MAP_SIZE << shift) - 1;
  }
35534c869   Matthew Wilcox   radix tree: const...
207
  static inline unsigned long node_maxindex(const struct radix_tree_node *node)
218ed7503   Matthew Wilcox   radix-tree: impro...
208
209
210
  {
  	return shift_maxindex(node->shift);
  }
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
211
212
213
214
215
216
  static unsigned long next_index(unsigned long index,
  				const struct radix_tree_node *node,
  				unsigned long offset)
  {
  	return (index & ~node_maxindex(node)) + (offset << node->shift);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
217
218
219
220
221
  /*
   * This assumes that the caller has performed appropriate preallocation, and
   * that the caller has pinned this thread of control to the current CPU.
   */
  static struct radix_tree_node *
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
222
  radix_tree_node_alloc(gfp_t gfp_mask, struct radix_tree_node *parent,
d58275bc9   Matthew Wilcox   radix-tree: Store...
223
  			struct radix_tree_root *root,
e8de43407   Matthew Wilcox   radix-tree: ensur...
224
  			unsigned int shift, unsigned int offset,
01959dfe7   Matthew Wilcox   xarray: Define st...
225
  			unsigned int count, unsigned int nr_values)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
226
  {
e2848a0ef   Nick Piggin   radix-tree: avoid...
227
  	struct radix_tree_node *ret = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
228

5e4c0d974   Jan Kara   lib/radix-tree.c:...
229
  	/*
2fcd9005c   Matthew Wilcox   radix-tree: misce...
230
231
232
  	 * Preload code isn't irq safe and it doesn't make sense to use
  	 * preloading during an interrupt anyway as all the allocations have
  	 * to be atomic. So just do normal allocation when in interrupt.
5e4c0d974   Jan Kara   lib/radix-tree.c:...
233
  	 */
d0164adc8   Mel Gorman   mm, page_alloc: d...
234
  	if (!gfpflags_allow_blocking(gfp_mask) && !in_interrupt()) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
235
  		struct radix_tree_preload *rtp;
e2848a0ef   Nick Piggin   radix-tree: avoid...
236
  		/*
58e698af4   Vladimir Davydov   radix-tree: accou...
237
  		 * Even if the caller has preloaded, try to allocate from the
05eb6e726   Vladimir Davydov   radix-tree: accou...
238
239
  		 * cache first for the new node to get accounted to the memory
  		 * cgroup.
58e698af4   Vladimir Davydov   radix-tree: accou...
240
241
  		 */
  		ret = kmem_cache_alloc(radix_tree_node_cachep,
05eb6e726   Vladimir Davydov   radix-tree: accou...
242
  				       gfp_mask | __GFP_NOWARN);
58e698af4   Vladimir Davydov   radix-tree: accou...
243
244
245
246
  		if (ret)
  			goto out;
  
  		/*
e2848a0ef   Nick Piggin   radix-tree: avoid...
247
248
249
250
  		 * Provided the caller has preloaded here, we will always
  		 * succeed in getting a node here (and never reach
  		 * kmem_cache_alloc)
  		 */
7c8e0181e   Christoph Lameter   mm: replace __get...
251
  		rtp = this_cpu_ptr(&radix_tree_preloads);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
  		if (rtp->nr) {
9d2a8da00   Kirill A. Shutemov   radix-tree: repla...
253
  			ret = rtp->nodes;
1293d5c5f   Matthew Wilcox   radix-tree: Chain...
254
  			rtp->nodes = ret->parent;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
255
256
  			rtp->nr--;
  		}
ce80b067d   Catalin Marinas   lib/radix-tree.c:...
257
258
259
260
261
  		/*
  		 * Update the allocation stack trace as this is more useful
  		 * for debugging.
  		 */
  		kmemleak_update_trace(ret);
58e698af4   Vladimir Davydov   radix-tree: accou...
262
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
263
  	}
05eb6e726   Vladimir Davydov   radix-tree: accou...
264
  	ret = kmem_cache_alloc(radix_tree_node_cachep, gfp_mask);
58e698af4   Vladimir Davydov   radix-tree: accou...
265
  out:
b194d16c2   Matthew Wilcox   radix-tree: renam...
266
  	BUG_ON(radix_tree_is_internal_node(ret));
e8de43407   Matthew Wilcox   radix-tree: ensur...
267
  	if (ret) {
e8de43407   Matthew Wilcox   radix-tree: ensur...
268
269
270
  		ret->shift = shift;
  		ret->offset = offset;
  		ret->count = count;
01959dfe7   Matthew Wilcox   xarray: Define st...
271
  		ret->nr_values = nr_values;
d58275bc9   Matthew Wilcox   radix-tree: Store...
272
  		ret->parent = parent;
01959dfe7   Matthew Wilcox   xarray: Define st...
273
  		ret->array = root;
e8de43407   Matthew Wilcox   radix-tree: ensur...
274
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
275
276
  	return ret;
  }
58d6ea308   Matthew Wilcox   xarray: Add XArra...
277
  void radix_tree_node_rcu_free(struct rcu_head *head)
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
278
279
280
  {
  	struct radix_tree_node *node =
  			container_of(head, struct radix_tree_node, rcu_head);
643b52b9c   Nick Piggin   radix-tree: fix s...
281
282
  
  	/*
175542f57   Matthew Wilcox   radix-tree: add r...
283
284
285
  	 * Must only free zeroed nodes into the slab.  We can be left with
  	 * non-NULL entries by radix_tree_free_nodes, so clear the entries
  	 * and tags here.
643b52b9c   Nick Piggin   radix-tree: fix s...
286
  	 */
175542f57   Matthew Wilcox   radix-tree: add r...
287
288
  	memset(node->slots, 0, sizeof(node->slots));
  	memset(node->tags, 0, sizeof(node->tags));
91d9c05ac   Matthew Wilcox   radix-tree: move ...
289
  	INIT_LIST_HEAD(&node->private_list);
643b52b9c   Nick Piggin   radix-tree: fix s...
290

7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
291
292
  	kmem_cache_free(radix_tree_node_cachep, node);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
293
294
295
  static inline void
  radix_tree_node_free(struct radix_tree_node *node)
  {
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
296
  	call_rcu(&node->rcu_head, radix_tree_node_rcu_free);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
297
298
299
300
301
302
303
  }
  
  /*
   * Load up this CPU's radix_tree_node buffer with sufficient objects to
   * ensure that the addition of a single element in the tree cannot fail.  On
   * success, return zero, with preemption disabled.  On error, return -ENOMEM
   * with preemption not disabled.
b34df792b   David Howells   FS-Cache: Use rad...
304
305
   *
   * To make use of this facility, the radix tree must be initialised without
d0164adc8   Mel Gorman   mm, page_alloc: d...
306
   * __GFP_DIRECT_RECLAIM being passed to INIT_RADIX_TREE().
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
   */
bc9ae2247   Eric Dumazet   radix-tree: must ...
308
  static __must_check int __radix_tree_preload(gfp_t gfp_mask, unsigned nr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309
310
311
312
  {
  	struct radix_tree_preload *rtp;
  	struct radix_tree_node *node;
  	int ret = -ENOMEM;
05eb6e726   Vladimir Davydov   radix-tree: accou...
313
314
315
316
317
  	/*
  	 * Nodes preloaded by one cgroup can be be used by another cgroup, so
  	 * they should never be accounted to any particular memory cgroup.
  	 */
  	gfp_mask &= ~__GFP_ACCOUNT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
318
  	preempt_disable();
7c8e0181e   Christoph Lameter   mm: replace __get...
319
  	rtp = this_cpu_ptr(&radix_tree_preloads);
c78c66d1d   Kirill A. Shutemov   radix-tree: imple...
320
  	while (rtp->nr < nr) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
321
  		preempt_enable();
488514d17   Christoph Lameter   Remove set_migrat...
322
  		node = kmem_cache_alloc(radix_tree_node_cachep, gfp_mask);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
323
324
325
  		if (node == NULL)
  			goto out;
  		preempt_disable();
7c8e0181e   Christoph Lameter   mm: replace __get...
326
  		rtp = this_cpu_ptr(&radix_tree_preloads);
c78c66d1d   Kirill A. Shutemov   radix-tree: imple...
327
  		if (rtp->nr < nr) {
1293d5c5f   Matthew Wilcox   radix-tree: Chain...
328
  			node->parent = rtp->nodes;
9d2a8da00   Kirill A. Shutemov   radix-tree: repla...
329
330
331
  			rtp->nodes = node;
  			rtp->nr++;
  		} else {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
332
  			kmem_cache_free(radix_tree_node_cachep, node);
9d2a8da00   Kirill A. Shutemov   radix-tree: repla...
333
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
334
335
336
337
338
  	}
  	ret = 0;
  out:
  	return ret;
  }
5e4c0d974   Jan Kara   lib/radix-tree.c:...
339
340
341
342
343
344
345
346
  
  /*
   * Load up this CPU's radix_tree_node buffer with sufficient objects to
   * ensure that the addition of a single element in the tree cannot fail.  On
   * success, return zero, with preemption disabled.  On error, return -ENOMEM
   * with preemption not disabled.
   *
   * To make use of this facility, the radix tree must be initialised without
d0164adc8   Mel Gorman   mm, page_alloc: d...
347
   * __GFP_DIRECT_RECLAIM being passed to INIT_RADIX_TREE().
5e4c0d974   Jan Kara   lib/radix-tree.c:...
348
349
350
351
   */
  int radix_tree_preload(gfp_t gfp_mask)
  {
  	/* Warn on non-sensical use... */
d0164adc8   Mel Gorman   mm, page_alloc: d...
352
  	WARN_ON_ONCE(!gfpflags_allow_blocking(gfp_mask));
c78c66d1d   Kirill A. Shutemov   radix-tree: imple...
353
  	return __radix_tree_preload(gfp_mask, RADIX_TREE_PRELOAD_SIZE);
5e4c0d974   Jan Kara   lib/radix-tree.c:...
354
  }
d7f0923d8   David Chinner   [LIB]: export rad...
355
  EXPORT_SYMBOL(radix_tree_preload);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
356

6e954b9e9   Nick Piggin   [PATCH] radix tre...
357
  /*
5e4c0d974   Jan Kara   lib/radix-tree.c:...
358
359
360
361
362
363
   * The same as above function, except we don't guarantee preloading happens.
   * We do it, if we decide it helps. On success, return zero with preemption
   * disabled. On error, return -ENOMEM with preemption not disabled.
   */
  int radix_tree_maybe_preload(gfp_t gfp_mask)
  {
d0164adc8   Mel Gorman   mm, page_alloc: d...
364
  	if (gfpflags_allow_blocking(gfp_mask))
c78c66d1d   Kirill A. Shutemov   radix-tree: imple...
365
  		return __radix_tree_preload(gfp_mask, RADIX_TREE_PRELOAD_SIZE);
5e4c0d974   Jan Kara   lib/radix-tree.c:...
366
367
368
369
370
  	/* Preloading doesn't help anything with this gfp mask, skip it */
  	preempt_disable();
  	return 0;
  }
  EXPORT_SYMBOL(radix_tree_maybe_preload);
35534c869   Matthew Wilcox   radix tree: const...
371
  static unsigned radix_tree_load_root(const struct radix_tree_root *root,
1456a439f   Matthew Wilcox   radix-tree: intro...
372
373
  		struct radix_tree_node **nodep, unsigned long *maxindex)
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
374
  	struct radix_tree_node *node = rcu_dereference_raw(root->xa_head);
1456a439f   Matthew Wilcox   radix-tree: intro...
375
376
  
  	*nodep = node;
b194d16c2   Matthew Wilcox   radix-tree: renam...
377
  	if (likely(radix_tree_is_internal_node(node))) {
4dd6c0987   Matthew Wilcox   radix-tree: renam...
378
  		node = entry_to_node(node);
1456a439f   Matthew Wilcox   radix-tree: intro...
379
  		*maxindex = node_maxindex(node);
c12e51b07   Matthew Wilcox   radix-tree: repla...
380
  		return node->shift + RADIX_TREE_MAP_SHIFT;
1456a439f   Matthew Wilcox   radix-tree: intro...
381
382
383
384
385
  	}
  
  	*maxindex = 0;
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
386
387
388
  /*
   *	Extend a radix tree so it can store key @index.
   */
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
389
  static int radix_tree_extend(struct radix_tree_root *root, gfp_t gfp,
d0891265b   Matthew Wilcox   radix-tree: remov...
390
  				unsigned long index, unsigned int shift)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
391
  {
d7b627277   Matthew Wilcox   radix-tree: Fix _...
392
  	void *entry;
d0891265b   Matthew Wilcox   radix-tree: remov...
393
  	unsigned int maxshift;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
394
  	int tag;
d0891265b   Matthew Wilcox   radix-tree: remov...
395
396
397
398
  	/* Figure out what the shift should be.  */
  	maxshift = shift;
  	while (index > shift_maxindex(maxshift))
  		maxshift += RADIX_TREE_MAP_SHIFT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
399

f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
400
  	entry = rcu_dereference_raw(root->xa_head);
d7b627277   Matthew Wilcox   radix-tree: Fix _...
401
  	if (!entry && (!is_idr(root) || root_tag_get(root, IDR_FREE)))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
402
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
403

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
404
  	do {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
405
  		struct radix_tree_node *node = radix_tree_node_alloc(gfp, NULL,
d58275bc9   Matthew Wilcox   radix-tree: Store...
406
  							root, shift, 0, 1, 0);
2fcd9005c   Matthew Wilcox   radix-tree: misce...
407
  		if (!node)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
408
  			return -ENOMEM;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
409
410
411
412
413
414
415
416
417
418
419
420
  		if (is_idr(root)) {
  			all_tag_set(node, IDR_FREE);
  			if (!root_tag_get(root, IDR_FREE)) {
  				tag_clear(node, IDR_FREE, 0);
  				root_tag_set(root, IDR_FREE);
  			}
  		} else {
  			/* Propagate the aggregated tag info to the new child */
  			for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) {
  				if (root_tag_get(root, tag))
  					tag_set(node, tag, 0);
  			}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
421
  		}
d0891265b   Matthew Wilcox   radix-tree: remov...
422
  		BUG_ON(shift > BITS_PER_LONG);
d7b627277   Matthew Wilcox   radix-tree: Fix _...
423
424
  		if (radix_tree_is_internal_node(entry)) {
  			entry_to_node(entry)->parent = node;
3159f943a   Matthew Wilcox   xarray: Replace e...
425
  		} else if (xa_is_value(entry)) {
01959dfe7   Matthew Wilcox   xarray: Define st...
426
427
  			/* Moving a value entry root->xa_head to a node */
  			node->nr_values = 1;
f7942430e   Johannes Weiner   lib: radix-tree: ...
428
  		}
d7b627277   Matthew Wilcox   radix-tree: Fix _...
429
430
431
432
433
434
  		/*
  		 * entry was already in the radix tree, so we do not need
  		 * rcu_assign_pointer here
  		 */
  		node->slots[0] = (void __rcu *)entry;
  		entry = node_to_entry(node);
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
435
  		rcu_assign_pointer(root->xa_head, entry);
d0891265b   Matthew Wilcox   radix-tree: remov...
436
  		shift += RADIX_TREE_MAP_SHIFT;
d0891265b   Matthew Wilcox   radix-tree: remov...
437
  	} while (shift <= maxshift);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
438
  out:
d0891265b   Matthew Wilcox   radix-tree: remov...
439
  	return maxshift + RADIX_TREE_MAP_SHIFT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
440
441
442
  }
  
  /**
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
443
444
445
   *	radix_tree_shrink    -    shrink radix tree to minimum height
   *	@root		radix tree root
   */
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
446
  static inline bool radix_tree_shrink(struct radix_tree_root *root)
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
447
  {
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
448
  	bool shrunk = false;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
449
  	for (;;) {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
450
  		struct radix_tree_node *node = rcu_dereference_raw(root->xa_head);
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
451
452
453
454
455
456
457
458
  		struct radix_tree_node *child;
  
  		if (!radix_tree_is_internal_node(node))
  			break;
  		node = entry_to_node(node);
  
  		/*
  		 * The candidate node has more than one child, or its child
3a08cd52c   Matthew Wilcox   radix tree: Remov...
459
  		 * is not at the leftmost slot, we cannot shrink.
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
460
461
462
  		 */
  		if (node->count != 1)
  			break;
12320d0ff   Matthew Wilcox   radix-tree: Add r...
463
  		child = rcu_dereference_raw(node->slots[0]);
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
464
465
  		if (!child)
  			break;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
466

66ee620f0   Matthew Wilcox   idr: Permit any v...
467
468
469
470
471
472
473
  		/*
  		 * For an IDR, we must not shrink entry 0 into the root in
  		 * case somebody calls idr_replace() with a pointer that
  		 * appears to be an internal entry
  		 */
  		if (!node->shift && is_idr(root))
  			break;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
474
475
476
477
478
479
480
481
  		if (radix_tree_is_internal_node(child))
  			entry_to_node(child)->parent = NULL;
  
  		/*
  		 * We don't need rcu_assign_pointer(), since we are simply
  		 * moving the node from one part of the tree to another: if it
  		 * was safe to dereference the old pointer to it
  		 * (node->slots[0]), it will be safe to dereference the new
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
482
  		 * one (root->xa_head) as far as dependent read barriers go.
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
483
  		 */
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
484
  		root->xa_head = (void __rcu *)child;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
485
486
  		if (is_idr(root) && !tag_get(node, IDR_FREE, 0))
  			root_tag_clear(root, IDR_FREE);
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
  
  		/*
  		 * We have a dilemma here. The node's slot[0] must not be
  		 * NULLed in case there are concurrent lookups expecting to
  		 * find the item. However if this was a bottom-level node,
  		 * then it may be subject to the slot pointer being visible
  		 * to callers dereferencing it. If item corresponding to
  		 * slot[0] is subsequently deleted, these callers would expect
  		 * their slot to become empty sooner or later.
  		 *
  		 * For example, lockless pagecache will look up a slot, deref
  		 * the page pointer, and if the page has 0 refcount it means it
  		 * was concurrently deleted from pagecache so try the deref
  		 * again. Fortunately there is already a requirement for logic
  		 * to retry the entire slot lookup -- the indirect pointer
  		 * problem (replacing direct root node with an indirect pointer
  		 * also results in a stale slot). So tag the slot as indirect
  		 * to force callers to retry.
  		 */
4d693d086   Johannes Weiner   lib: radix-tree: ...
506
507
  		node->count = 0;
  		if (!radix_tree_is_internal_node(child)) {
d7b627277   Matthew Wilcox   radix-tree: Fix _...
508
  			node->slots[0] = (void __rcu *)RADIX_TREE_RETRY;
4d693d086   Johannes Weiner   lib: radix-tree: ...
509
  		}
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
510

ea07b862a   Johannes Weiner   mm: workingset: f...
511
  		WARN_ON_ONCE(!list_empty(&node->private_list));
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
512
  		radix_tree_node_free(node);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
513
  		shrunk = true;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
514
  	}
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
515
516
  
  	return shrunk;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
517
  }
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
518
  static bool delete_node(struct radix_tree_root *root,
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
519
  			struct radix_tree_node *node)
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
520
  {
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
521
  	bool deleted = false;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
522
523
524
525
  	do {
  		struct radix_tree_node *parent;
  
  		if (node->count) {
12320d0ff   Matthew Wilcox   radix-tree: Add r...
526
  			if (node_to_entry(node) ==
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
527
  					rcu_dereference_raw(root->xa_head))
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
528
  				deleted |= radix_tree_shrink(root);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
529
  			return deleted;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
530
531
532
533
534
535
536
  		}
  
  		parent = node->parent;
  		if (parent) {
  			parent->slots[node->offset] = NULL;
  			parent->count--;
  		} else {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
537
538
539
540
541
542
  			/*
  			 * Shouldn't the tags already have all been cleared
  			 * by the caller?
  			 */
  			if (!is_idr(root))
  				root_tag_clear_all(root);
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
543
  			root->xa_head = NULL;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
544
  		}
ea07b862a   Johannes Weiner   mm: workingset: f...
545
  		WARN_ON_ONCE(!list_empty(&node->private_list));
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
546
  		radix_tree_node_free(node);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
547
  		deleted = true;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
548
549
550
  
  		node = parent;
  	} while (node);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
551
552
  
  	return deleted;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
553
554
555
  }
  
  /**
139e56166   Johannes Weiner   lib: radix_tree: ...
556
   *	__radix_tree_create	-	create a slot in a radix tree
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
557
558
   *	@root:		radix tree root
   *	@index:		index key
139e56166   Johannes Weiner   lib: radix_tree: ...
559
560
   *	@nodep:		returns node
   *	@slotp:		returns slot
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
561
   *
139e56166   Johannes Weiner   lib: radix_tree: ...
562
563
564
565
   *	Create, if necessary, and return the node and slot for an item
   *	at position @index in the radix tree @root.
   *
   *	Until there is more than one item in the tree, no nodes are
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
566
   *	allocated and @root->xa_head is used as a direct slot instead of
139e56166   Johannes Weiner   lib: radix_tree: ...
567
568
569
   *	pointing to a node, in which case *@nodep will be NULL.
   *
   *	Returns -ENOMEM, or 0 for success.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
570
   */
74d609585   Matthew Wilcox   page cache: Add a...
571
  static int __radix_tree_create(struct radix_tree_root *root,
3a08cd52c   Matthew Wilcox   radix tree: Remov...
572
573
  		unsigned long index, struct radix_tree_node **nodep,
  		void __rcu ***slotp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
574
  {
89148aa40   Matthew Wilcox   radix-tree: tidy ...
575
  	struct radix_tree_node *node = NULL, *child;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
576
  	void __rcu **slot = (void __rcu **)&root->xa_head;
49ea6ebcd   Matthew Wilcox   radix-tree: fix e...
577
  	unsigned long maxindex;
89148aa40   Matthew Wilcox   radix-tree: tidy ...
578
  	unsigned int shift, offset = 0;
3a08cd52c   Matthew Wilcox   radix tree: Remov...
579
  	unsigned long max = index;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
580
  	gfp_t gfp = root_gfp_mask(root);
49ea6ebcd   Matthew Wilcox   radix-tree: fix e...
581

89148aa40   Matthew Wilcox   radix-tree: tidy ...
582
  	shift = radix_tree_load_root(root, &child, &maxindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
583
584
  
  	/* Make sure the tree is high enough.  */
49ea6ebcd   Matthew Wilcox   radix-tree: fix e...
585
  	if (max > maxindex) {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
586
  		int error = radix_tree_extend(root, gfp, max, shift);
49ea6ebcd   Matthew Wilcox   radix-tree: fix e...
587
  		if (error < 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
588
  			return error;
49ea6ebcd   Matthew Wilcox   radix-tree: fix e...
589
  		shift = error;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
590
  		child = rcu_dereference_raw(root->xa_head);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
591
  	}
3a08cd52c   Matthew Wilcox   radix tree: Remov...
592
  	while (shift > 0) {
c12e51b07   Matthew Wilcox   radix-tree: repla...
593
  		shift -= RADIX_TREE_MAP_SHIFT;
89148aa40   Matthew Wilcox   radix-tree: tidy ...
594
  		if (child == NULL) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
595
  			/* Have to add a child node.  */
d58275bc9   Matthew Wilcox   radix-tree: Store...
596
  			child = radix_tree_node_alloc(gfp, node, root, shift,
e8de43407   Matthew Wilcox   radix-tree: ensur...
597
  							offset, 0, 0);
89148aa40   Matthew Wilcox   radix-tree: tidy ...
598
  			if (!child)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
599
  				return -ENOMEM;
89148aa40   Matthew Wilcox   radix-tree: tidy ...
600
601
  			rcu_assign_pointer(*slot, node_to_entry(child));
  			if (node)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
602
  				node->count++;
89148aa40   Matthew Wilcox   radix-tree: tidy ...
603
  		} else if (!radix_tree_is_internal_node(child))
e61452365   Matthew Wilcox   radix_tree: add s...
604
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
605
606
  
  		/* Go a level down */
89148aa40   Matthew Wilcox   radix-tree: tidy ...
607
  		node = entry_to_node(child);
9e85d8111   Matthew Wilcox   radix-tree: make ...
608
  		offset = radix_tree_descend(node, &child, index);
89148aa40   Matthew Wilcox   radix-tree: tidy ...
609
  		slot = &node->slots[offset];
e61452365   Matthew Wilcox   radix_tree: add s...
610
  	}
175542f57   Matthew Wilcox   radix-tree: add r...
611
612
613
614
615
616
  	if (nodep)
  		*nodep = node;
  	if (slotp)
  		*slotp = slot;
  	return 0;
  }
175542f57   Matthew Wilcox   radix-tree: add r...
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
  /*
   * Free any nodes below this node.  The tree is presumed to not need
   * shrinking, and any user data in the tree is presumed to not need a
   * destructor called on it.  If we need to add a destructor, we can
   * add that functionality later.  Note that we may not clear tags or
   * slots from the tree as an RCU walker may still have a pointer into
   * this subtree.  We could replace the entries with RADIX_TREE_RETRY,
   * but we'll still have to clear those in rcu_free.
   */
  static void radix_tree_free_nodes(struct radix_tree_node *node)
  {
  	unsigned offset = 0;
  	struct radix_tree_node *child = entry_to_node(node);
  
  	for (;;) {
12320d0ff   Matthew Wilcox   radix-tree: Add r...
632
  		void *entry = rcu_dereference_raw(child->slots[offset]);
02c02bf12   Matthew Wilcox   xarray: Change de...
633
  		if (xa_is_node(entry) && child->shift) {
175542f57   Matthew Wilcox   radix-tree: add r...
634
635
636
637
638
639
640
641
642
  			child = entry_to_node(entry);
  			offset = 0;
  			continue;
  		}
  		offset++;
  		while (offset == RADIX_TREE_MAP_SIZE) {
  			struct radix_tree_node *old = child;
  			offset = child->offset + 1;
  			child = child->parent;
dd040b6f6   Matthew Wilcox   radix-tree: fix p...
643
  			WARN_ON_ONCE(!list_empty(&old->private_list));
175542f57   Matthew Wilcox   radix-tree: add r...
644
645
646
647
648
649
  			radix_tree_node_free(old);
  			if (old == entry_to_node(node))
  				return;
  		}
  	}
  }
d7b627277   Matthew Wilcox   radix-tree: Fix _...
650
  static inline int insert_entries(struct radix_tree_node *node,
3a08cd52c   Matthew Wilcox   radix tree: Remov...
651
  		void __rcu **slot, void *item, bool replace)
175542f57   Matthew Wilcox   radix-tree: add r...
652
653
654
655
656
657
  {
  	if (*slot)
  		return -EEXIST;
  	rcu_assign_pointer(*slot, item);
  	if (node) {
  		node->count++;
3159f943a   Matthew Wilcox   xarray: Replace e...
658
  		if (xa_is_value(item))
01959dfe7   Matthew Wilcox   xarray: Define st...
659
  			node->nr_values++;
175542f57   Matthew Wilcox   radix-tree: add r...
660
661
662
  	}
  	return 1;
  }
139e56166   Johannes Weiner   lib: radix_tree: ...
663
664
  
  /**
e61452365   Matthew Wilcox   radix_tree: add s...
665
   *	__radix_tree_insert    -    insert into a radix tree
139e56166   Johannes Weiner   lib: radix_tree: ...
666
667
668
669
670
671
   *	@root:		radix tree root
   *	@index:		index key
   *	@item:		item to insert
   *
   *	Insert an item into the radix tree at position @index.
   */
3a08cd52c   Matthew Wilcox   radix tree: Remov...
672
673
  int radix_tree_insert(struct radix_tree_root *root, unsigned long index,
  			void *item)
139e56166   Johannes Weiner   lib: radix_tree: ...
674
675
  {
  	struct radix_tree_node *node;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
676
  	void __rcu **slot;
139e56166   Johannes Weiner   lib: radix_tree: ...
677
  	int error;
b194d16c2   Matthew Wilcox   radix-tree: renam...
678
  	BUG_ON(radix_tree_is_internal_node(item));
139e56166   Johannes Weiner   lib: radix_tree: ...
679

3a08cd52c   Matthew Wilcox   radix tree: Remov...
680
  	error = __radix_tree_create(root, index, &node, &slot);
139e56166   Johannes Weiner   lib: radix_tree: ...
681
682
  	if (error)
  		return error;
175542f57   Matthew Wilcox   radix-tree: add r...
683

3a08cd52c   Matthew Wilcox   radix tree: Remov...
684
  	error = insert_entries(node, slot, item, false);
175542f57   Matthew Wilcox   radix-tree: add r...
685
686
  	if (error < 0)
  		return error;
201b6264f   Christoph Lameter   [PATCH] radix-tre...
687

612d6c19d   Nick Piggin   [PATCH] radix-tre...
688
  	if (node) {
7b60e9ad5   Matthew Wilcox   radix-tree: fix m...
689
  		unsigned offset = get_slot_offset(node, slot);
7b60e9ad5   Matthew Wilcox   radix-tree: fix m...
690
691
692
  		BUG_ON(tag_get(node, 0, offset));
  		BUG_ON(tag_get(node, 1, offset));
  		BUG_ON(tag_get(node, 2, offset));
612d6c19d   Nick Piggin   [PATCH] radix-tre...
693
  	} else {
7b60e9ad5   Matthew Wilcox   radix-tree: fix m...
694
  		BUG_ON(root_tags_get(root));
612d6c19d   Nick Piggin   [PATCH] radix-tre...
695
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
696

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
697
698
  	return 0;
  }
3a08cd52c   Matthew Wilcox   radix tree: Remov...
699
  EXPORT_SYMBOL(radix_tree_insert);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
700

139e56166   Johannes Weiner   lib: radix_tree: ...
701
702
703
704
705
706
707
708
709
710
711
  /**
   *	__radix_tree_lookup	-	lookup an item in a radix tree
   *	@root:		radix tree root
   *	@index:		index key
   *	@nodep:		returns node
   *	@slotp:		returns slot
   *
   *	Lookup and return the item at position @index in the radix
   *	tree @root.
   *
   *	Until there is more than one item in the tree, no nodes are
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
712
   *	allocated and @root->xa_head is used as a direct slot instead of
139e56166   Johannes Weiner   lib: radix_tree: ...
713
   *	pointing to a node, in which case *@nodep will be NULL.
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
714
   */
35534c869   Matthew Wilcox   radix tree: const...
715
716
  void *__radix_tree_lookup(const struct radix_tree_root *root,
  			  unsigned long index, struct radix_tree_node **nodep,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
717
  			  void __rcu ***slotp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
718
  {
139e56166   Johannes Weiner   lib: radix_tree: ...
719
  	struct radix_tree_node *node, *parent;
858299544   Matthew Wilcox   radix-tree: rewri...
720
  	unsigned long maxindex;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
721
  	void __rcu **slot;
612d6c19d   Nick Piggin   [PATCH] radix-tre...
722

858299544   Matthew Wilcox   radix-tree: rewri...
723
724
   restart:
  	parent = NULL;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
725
  	slot = (void __rcu **)&root->xa_head;
9e85d8111   Matthew Wilcox   radix-tree: make ...
726
  	radix_tree_load_root(root, &node, &maxindex);
858299544   Matthew Wilcox   radix-tree: rewri...
727
  	if (index > maxindex)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
728
  		return NULL;
b194d16c2   Matthew Wilcox   radix-tree: renam...
729
  	while (radix_tree_is_internal_node(node)) {
858299544   Matthew Wilcox   radix-tree: rewri...
730
  		unsigned offset;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
731

4dd6c0987   Matthew Wilcox   radix-tree: renam...
732
  		parent = entry_to_node(node);
9e85d8111   Matthew Wilcox   radix-tree: make ...
733
  		offset = radix_tree_descend(parent, &node, index);
858299544   Matthew Wilcox   radix-tree: rewri...
734
  		slot = parent->slots + offset;
eff3860bb   Matthew Wilcox   radix tree: Don't...
735
736
  		if (node == RADIX_TREE_RETRY)
  			goto restart;
66ee620f0   Matthew Wilcox   idr: Permit any v...
737
738
  		if (parent->shift == 0)
  			break;
858299544   Matthew Wilcox   radix-tree: rewri...
739
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
740

139e56166   Johannes Weiner   lib: radix_tree: ...
741
742
743
744
745
  	if (nodep)
  		*nodep = parent;
  	if (slotp)
  		*slotp = slot;
  	return node;
b72b71c6c   Huang Shijie   lib: do code opti...
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
  }
  
  /**
   *	radix_tree_lookup_slot    -    lookup a slot in a radix tree
   *	@root:		radix tree root
   *	@index:		index key
   *
   *	Returns:  the slot corresponding to the position @index in the
   *	radix tree @root. This is useful for update-if-exists operations.
   *
   *	This function can be called under rcu_read_lock iff the slot is not
   *	modified by radix_tree_replace_slot, otherwise it must be called
   *	exclusive from other writers. Any dereference of the slot must be done
   *	using radix_tree_deref_slot.
   */
d7b627277   Matthew Wilcox   radix-tree: Fix _...
761
  void __rcu **radix_tree_lookup_slot(const struct radix_tree_root *root,
35534c869   Matthew Wilcox   radix tree: const...
762
  				unsigned long index)
b72b71c6c   Huang Shijie   lib: do code opti...
763
  {
d7b627277   Matthew Wilcox   radix-tree: Fix _...
764
  	void __rcu **slot;
139e56166   Johannes Weiner   lib: radix_tree: ...
765
766
767
768
  
  	if (!__radix_tree_lookup(root, index, NULL, &slot))
  		return NULL;
  	return slot;
a43313668   Hans Reiser   [PATCH] reiser4: ...
769
  }
a43313668   Hans Reiser   [PATCH] reiser4: ...
770
771
772
773
774
775
776
777
  EXPORT_SYMBOL(radix_tree_lookup_slot);
  
  /**
   *	radix_tree_lookup    -    perform lookup operation on a radix tree
   *	@root:		radix tree root
   *	@index:		index key
   *
   *	Lookup the item at the position @index in the radix tree @root.
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
778
779
780
781
782
   *
   *	This function can be called under rcu_read_lock, however the caller
   *	must manage lifetimes of leaf nodes (eg. RCU may also be used to free
   *	them safely). No RCU barriers are required to access or modify the
   *	returned item, however.
a43313668   Hans Reiser   [PATCH] reiser4: ...
783
   */
35534c869   Matthew Wilcox   radix tree: const...
784
  void *radix_tree_lookup(const struct radix_tree_root *root, unsigned long index)
a43313668   Hans Reiser   [PATCH] reiser4: ...
785
  {
139e56166   Johannes Weiner   lib: radix_tree: ...
786
  	return __radix_tree_lookup(root, index, NULL, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
787
788
  }
  EXPORT_SYMBOL(radix_tree_lookup);
d7b627277   Matthew Wilcox   radix-tree: Fix _...
789
  static void replace_slot(void __rcu **slot, void *item,
01959dfe7   Matthew Wilcox   xarray: Define st...
790
  		struct radix_tree_node *node, int count, int values)
f7942430e   Johannes Weiner   lib: radix-tree: ...
791
  {
01959dfe7   Matthew Wilcox   xarray: Define st...
792
  	if (node && (count || values)) {
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
793
  		node->count += count;
01959dfe7   Matthew Wilcox   xarray: Define st...
794
  		node->nr_values += values;
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
795
  	}
f7942430e   Johannes Weiner   lib: radix-tree: ...
796
797
798
  
  	rcu_assign_pointer(*slot, item);
  }
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
799
800
801
  static bool node_tag_get(const struct radix_tree_root *root,
  				const struct radix_tree_node *node,
  				unsigned int tag, unsigned int offset)
a90eb3a2a   Matthew Wilcox   radix-tree: fix r...
802
  {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
803
804
805
806
  	if (node)
  		return tag_get(node, tag, offset);
  	return root_tag_get(root, tag);
  }
a90eb3a2a   Matthew Wilcox   radix-tree: fix r...
807

0a835c4f0   Matthew Wilcox   Reimplement IDR a...
808
809
810
811
812
813
814
815
  /*
   * IDR users want to be able to store NULL in the tree, so if the slot isn't
   * free, don't adjust the count, even if it's transitioning between NULL and
   * non-NULL.  For the IDA, we mark slots as being IDR_FREE while they still
   * have empty bits, but it only stores NULL in slots when they're being
   * deleted.
   */
  static int calculate_count(struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
816
  				struct radix_tree_node *node, void __rcu **slot,
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
817
818
819
820
821
822
823
824
825
  				void *item, void *old)
  {
  	if (is_idr(root)) {
  		unsigned offset = get_slot_offset(node, slot);
  		bool free = node_tag_get(root, node, IDR_FREE, offset);
  		if (!free)
  			return 0;
  		if (!old)
  			return 1;
a90eb3a2a   Matthew Wilcox   radix-tree: fix r...
826
  	}
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
827
  	return !!item - !!old;
a90eb3a2a   Matthew Wilcox   radix-tree: fix r...
828
  }
f7942430e   Johannes Weiner   lib: radix-tree: ...
829
  /**
6d75f366b   Johannes Weiner   lib: radix-tree: ...
830
   * __radix_tree_replace		- replace item in a slot
4d693d086   Johannes Weiner   lib: radix-tree: ...
831
832
833
834
   * @root:		radix tree root
   * @node:		pointer to tree node
   * @slot:		pointer to slot in @node
   * @item:		new item to store in the slot.
6d75f366b   Johannes Weiner   lib: radix-tree: ...
835
836
837
838
839
840
   *
   * For use with __radix_tree_lookup().  Caller must hold tree write locked
   * across slot lookup and replacement.
   */
  void __radix_tree_replace(struct radix_tree_root *root,
  			  struct radix_tree_node *node,
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
841
  			  void __rcu **slot, void *item)
6d75f366b   Johannes Weiner   lib: radix-tree: ...
842
  {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
843
  	void *old = rcu_dereference_raw(*slot);
01959dfe7   Matthew Wilcox   xarray: Define st...
844
  	int values = !!xa_is_value(item) - !!xa_is_value(old);
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
845
  	int count = calculate_count(root, node, slot, item, old);
6d75f366b   Johannes Weiner   lib: radix-tree: ...
846
  	/*
01959dfe7   Matthew Wilcox   xarray: Define st...
847
  	 * This function supports replacing value entries and
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
848
  	 * deleting entries, but that needs accounting against the
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
849
  	 * node unless the slot is root->xa_head.
6d75f366b   Johannes Weiner   lib: radix-tree: ...
850
  	 */
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
851
  	WARN_ON_ONCE(!node && (slot != (void __rcu **)&root->xa_head) &&
01959dfe7   Matthew Wilcox   xarray: Define st...
852
853
  			(count || values));
  	replace_slot(slot, item, node, count, values);
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
854

4d693d086   Johannes Weiner   lib: radix-tree: ...
855
856
  	if (!node)
  		return;
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
857
  	delete_node(root, node);
6d75f366b   Johannes Weiner   lib: radix-tree: ...
858
859
860
861
862
863
864
865
  }
  
  /**
   * radix_tree_replace_slot	- replace item in a slot
   * @root:	radix tree root
   * @slot:	pointer to slot
   * @item:	new item to store in the slot.
   *
7b8d046fb   Matthew Wilcox   shmem: Convert sh...
866
   * For use with radix_tree_lookup_slot() and
6d75f366b   Johannes Weiner   lib: radix-tree: ...
867
868
869
870
   * radix_tree_gang_lookup_tag_slot().  Caller must hold tree write locked
   * across slot lookup and replacement.
   *
   * NOTE: This cannot be used to switch between non-entries (empty slots),
01959dfe7   Matthew Wilcox   xarray: Define st...
871
   * regular entries, and value entries, as that requires accounting
f4b109c6d   Johannes Weiner   lib: radix-tree: ...
872
   * inside the radix tree node. When switching from one type of entry or
e157b5559   Matthew Wilcox   radix-tree: add r...
873
874
   * deleting, use __radix_tree_lookup() and __radix_tree_replace() or
   * radix_tree_iter_replace().
6d75f366b   Johannes Weiner   lib: radix-tree: ...
875
876
   */
  void radix_tree_replace_slot(struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
877
  			     void __rcu **slot, void *item)
6d75f366b   Johannes Weiner   lib: radix-tree: ...
878
  {
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
879
  	__radix_tree_replace(root, NULL, slot, item);
6d75f366b   Johannes Weiner   lib: radix-tree: ...
880
  }
10257d719   Song Liu   EXPORT_SYMBOL rad...
881
  EXPORT_SYMBOL(radix_tree_replace_slot);
6d75f366b   Johannes Weiner   lib: radix-tree: ...
882

e157b5559   Matthew Wilcox   radix-tree: add r...
883
884
885
886
887
888
  /**
   * radix_tree_iter_replace - replace item in a slot
   * @root:	radix tree root
   * @slot:	pointer to slot
   * @item:	new item to store in the slot.
   *
2956c6644   Matthew Wilcox   radix tree: Remov...
889
890
   * For use with radix_tree_for_each_slot().
   * Caller must hold tree write locked.
e157b5559   Matthew Wilcox   radix-tree: add r...
891
892
   */
  void radix_tree_iter_replace(struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
893
894
  				const struct radix_tree_iter *iter,
  				void __rcu **slot, void *item)
e157b5559   Matthew Wilcox   radix-tree: add r...
895
  {
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
896
  	__radix_tree_replace(root, iter->node, slot, item);
e157b5559   Matthew Wilcox   radix-tree: add r...
897
  }
30b888ba9   Matthew Wilcox   radix-tree: Add r...
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
  static void node_tag_set(struct radix_tree_root *root,
  				struct radix_tree_node *node,
  				unsigned int tag, unsigned int offset)
  {
  	while (node) {
  		if (tag_get(node, tag, offset))
  			return;
  		tag_set(node, tag, offset);
  		offset = node->offset;
  		node = node->parent;
  	}
  
  	if (!root_tag_get(root, tag))
  		root_tag_set(root, tag);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
913
914
915
916
  /**
   *	radix_tree_tag_set - set a tag on a radix tree node
   *	@root:		radix tree root
   *	@index:		index key
2fcd9005c   Matthew Wilcox   radix-tree: misce...
917
   *	@tag:		tag index
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
918
   *
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
919
920
   *	Set the search tag (which must be < RADIX_TREE_MAX_TAGS)
   *	corresponding to @index in the radix tree.  From
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
921
922
   *	the root all the way down to the leaf node.
   *
2fcd9005c   Matthew Wilcox   radix-tree: misce...
923
   *	Returns the address of the tagged item.  Setting a tag on a not-present
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
924
925
926
   *	item is a bug.
   */
  void *radix_tree_tag_set(struct radix_tree_root *root,
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
927
  			unsigned long index, unsigned int tag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
928
  {
fb969909d   Ross Zwisler   radix-tree: rewri...
929
930
  	struct radix_tree_node *node, *parent;
  	unsigned long maxindex;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
931

9e85d8111   Matthew Wilcox   radix-tree: make ...
932
  	radix_tree_load_root(root, &node, &maxindex);
fb969909d   Ross Zwisler   radix-tree: rewri...
933
  	BUG_ON(index > maxindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
934

b194d16c2   Matthew Wilcox   radix-tree: renam...
935
  	while (radix_tree_is_internal_node(node)) {
fb969909d   Ross Zwisler   radix-tree: rewri...
936
  		unsigned offset;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
937

4dd6c0987   Matthew Wilcox   radix-tree: renam...
938
  		parent = entry_to_node(node);
9e85d8111   Matthew Wilcox   radix-tree: make ...
939
  		offset = radix_tree_descend(parent, &node, index);
fb969909d   Ross Zwisler   radix-tree: rewri...
940
941
942
943
  		BUG_ON(!node);
  
  		if (!tag_get(parent, tag, offset))
  			tag_set(parent, tag, offset);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
944
  	}
612d6c19d   Nick Piggin   [PATCH] radix-tre...
945
  	/* set the root's tag bit */
fb969909d   Ross Zwisler   radix-tree: rewri...
946
  	if (!root_tag_get(root, tag))
612d6c19d   Nick Piggin   [PATCH] radix-tre...
947
  		root_tag_set(root, tag);
fb969909d   Ross Zwisler   radix-tree: rewri...
948
  	return node;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
949
950
  }
  EXPORT_SYMBOL(radix_tree_tag_set);
d604c3245   Matthew Wilcox   radix-tree: intro...
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
  static void node_tag_clear(struct radix_tree_root *root,
  				struct radix_tree_node *node,
  				unsigned int tag, unsigned int offset)
  {
  	while (node) {
  		if (!tag_get(node, tag, offset))
  			return;
  		tag_clear(node, tag, offset);
  		if (any_tag_set(node, tag))
  			return;
  
  		offset = node->offset;
  		node = node->parent;
  	}
  
  	/* clear the root's tag bit */
  	if (root_tag_get(root, tag))
  		root_tag_clear(root, tag);
  }
268f42de7   Matthew Wilcox   radix-tree: delet...
970
  /**
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
971
972
973
   *	radix_tree_tag_clear - clear a tag on a radix tree node
   *	@root:		radix tree root
   *	@index:		index key
2fcd9005c   Matthew Wilcox   radix-tree: misce...
974
   *	@tag:		tag index
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
975
   *
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
976
   *	Clear the search tag (which must be < RADIX_TREE_MAX_TAGS)
2fcd9005c   Matthew Wilcox   radix-tree: misce...
977
978
   *	corresponding to @index in the radix tree.  If this causes
   *	the leaf node to have no tags set then clear the tag in the
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
979
980
981
982
983
984
   *	next-to-leaf node, etc.
   *
   *	Returns the address of the tagged item on success, else NULL.  ie:
   *	has the same return value and semantics as radix_tree_lookup().
   */
  void *radix_tree_tag_clear(struct radix_tree_root *root,
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
985
  			unsigned long index, unsigned int tag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
986
  {
00f47b581   Ross Zwisler   radix-tree: rewri...
987
988
  	struct radix_tree_node *node, *parent;
  	unsigned long maxindex;
e2bdb933a   Hugh Dickins   radix_tree: take ...
989
  	int uninitialized_var(offset);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
990

9e85d8111   Matthew Wilcox   radix-tree: make ...
991
  	radix_tree_load_root(root, &node, &maxindex);
00f47b581   Ross Zwisler   radix-tree: rewri...
992
993
  	if (index > maxindex)
  		return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
994

00f47b581   Ross Zwisler   radix-tree: rewri...
995
  	parent = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
996

b194d16c2   Matthew Wilcox   radix-tree: renam...
997
  	while (radix_tree_is_internal_node(node)) {
4dd6c0987   Matthew Wilcox   radix-tree: renam...
998
  		parent = entry_to_node(node);
9e85d8111   Matthew Wilcox   radix-tree: make ...
999
  		offset = radix_tree_descend(parent, &node, index);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1000
  	}
d604c3245   Matthew Wilcox   radix-tree: intro...
1001
1002
  	if (node)
  		node_tag_clear(root, parent, tag, offset);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1003

00f47b581   Ross Zwisler   radix-tree: rewri...
1004
  	return node;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1005
1006
  }
  EXPORT_SYMBOL(radix_tree_tag_clear);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1007
  /**
30b888ba9   Matthew Wilcox   radix-tree: Add r...
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
    * radix_tree_iter_tag_clear - clear a tag on the current iterator entry
    * @root: radix tree root
    * @iter: iterator state
    * @tag: tag to clear
    */
  void radix_tree_iter_tag_clear(struct radix_tree_root *root,
  			const struct radix_tree_iter *iter, unsigned int tag)
  {
  	node_tag_clear(root, iter->node, tag, iter_offset(iter));
  }
  
  /**
32605a181   Marcelo Tosatti   [PATCH] radix_tag...
1020
1021
1022
   * radix_tree_tag_get - get a tag on a radix tree node
   * @root:		radix tree root
   * @index:		index key
2fcd9005c   Matthew Wilcox   radix-tree: misce...
1023
   * @tag:		tag index (< RADIX_TREE_MAX_TAGS)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1024
   *
32605a181   Marcelo Tosatti   [PATCH] radix_tag...
1025
   * Return values:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1026
   *
612d6c19d   Nick Piggin   [PATCH] radix-tre...
1027
1028
   *  0: tag not present or not set
   *  1: tag set
ce82653d6   David Howells   radix_tree_tag_ge...
1029
1030
1031
1032
   *
   * Note that the return value of this function may not be relied on, even if
   * the RCU lock is held, unless tag modification and node deletion are excluded
   * from concurrency.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1033
   */
35534c869   Matthew Wilcox   radix tree: const...
1034
  int radix_tree_tag_get(const struct radix_tree_root *root,
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
1035
  			unsigned long index, unsigned int tag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1036
  {
4589ba6d0   Ross Zwisler   radix-tree: rewri...
1037
1038
  	struct radix_tree_node *node, *parent;
  	unsigned long maxindex;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1039

612d6c19d   Nick Piggin   [PATCH] radix-tre...
1040
1041
  	if (!root_tag_get(root, tag))
  		return 0;
9e85d8111   Matthew Wilcox   radix-tree: make ...
1042
  	radix_tree_load_root(root, &node, &maxindex);
4589ba6d0   Ross Zwisler   radix-tree: rewri...
1043
1044
  	if (index > maxindex)
  		return 0;
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1045

b194d16c2   Matthew Wilcox   radix-tree: renam...
1046
  	while (radix_tree_is_internal_node(node)) {
9e85d8111   Matthew Wilcox   radix-tree: make ...
1047
  		unsigned offset;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1048

4dd6c0987   Matthew Wilcox   radix-tree: renam...
1049
  		parent = entry_to_node(node);
9e85d8111   Matthew Wilcox   radix-tree: make ...
1050
  		offset = radix_tree_descend(parent, &node, index);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1051

4589ba6d0   Ross Zwisler   radix-tree: rewri...
1052
  		if (!tag_get(parent, tag, offset))
3fa36acbc   Hugh Dickins   radix_tree: clean...
1053
  			return 0;
4589ba6d0   Ross Zwisler   radix-tree: rewri...
1054
1055
  		if (node == RADIX_TREE_RETRY)
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1056
  	}
4589ba6d0   Ross Zwisler   radix-tree: rewri...
1057
1058
  
  	return 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1059
1060
  }
  EXPORT_SYMBOL(radix_tree_tag_get);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1061

148deab22   Matthew Wilcox   radix-tree: impro...
1062
1063
1064
1065
1066
1067
1068
  /* Construct iter->tags bit-mask from node->tags[tag] array */
  static void set_iter_tags(struct radix_tree_iter *iter,
  				struct radix_tree_node *node, unsigned offset,
  				unsigned tag)
  {
  	unsigned tag_long = offset / BITS_PER_LONG;
  	unsigned tag_bit  = offset % BITS_PER_LONG;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1069
1070
1071
1072
  	if (!node) {
  		iter->tags = 1;
  		return;
  	}
148deab22   Matthew Wilcox   radix-tree: impro...
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
  	iter->tags = node->tags[tag][tag_long] >> tag_bit;
  
  	/* This never happens if RADIX_TREE_TAG_LONGS == 1 */
  	if (tag_long < RADIX_TREE_TAG_LONGS - 1) {
  		/* Pick tags from next element */
  		if (tag_bit)
  			iter->tags |= node->tags[tag][tag_long + 1] <<
  						(BITS_PER_LONG - tag_bit);
  		/* Clip chunk size, here only BITS_PER_LONG tags */
  		iter->next_index = __radix_tree_iter_add(iter, BITS_PER_LONG);
  	}
  }
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1085
1086
  void __rcu **radix_tree_iter_resume(void __rcu **slot,
  					struct radix_tree_iter *iter)
148deab22   Matthew Wilcox   radix-tree: impro...
1087
  {
148deab22   Matthew Wilcox   radix-tree: impro...
1088
1089
  	slot++;
  	iter->index = __radix_tree_iter_add(iter, 1);
148deab22   Matthew Wilcox   radix-tree: impro...
1090
1091
1092
1093
1094
  	iter->next_index = iter->index;
  	iter->tags = 0;
  	return NULL;
  }
  EXPORT_SYMBOL(radix_tree_iter_resume);
6df8ba4f8   Fengguang Wu   radixtree: introd...
1095
  /**
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1096
1097
1098
1099
1100
1101
1102
   * radix_tree_next_chunk - find next chunk of slots for iteration
   *
   * @root:	radix tree root
   * @iter:	iterator state
   * @flags:	RADIX_TREE_ITER_* flags and tag index
   * Returns:	pointer to chunk first slot, or NULL if iteration is over
   */
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1103
  void __rcu **radix_tree_next_chunk(const struct radix_tree_root *root,
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1104
1105
  			     struct radix_tree_iter *iter, unsigned flags)
  {
9e85d8111   Matthew Wilcox   radix-tree: make ...
1106
  	unsigned tag = flags & RADIX_TREE_ITER_TAG_MASK;
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1107
  	struct radix_tree_node *node, *child;
21ef53393   Ross Zwisler   radix-tree: add s...
1108
  	unsigned long index, offset, maxindex;
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1109
1110
1111
1112
1113
1114
1115
1116
1117
  
  	if ((flags & RADIX_TREE_ITER_TAGGED) && !root_tag_get(root, tag))
  		return NULL;
  
  	/*
  	 * Catch next_index overflow after ~0UL. iter->index never overflows
  	 * during iterating; it can be zero only at the beginning.
  	 * And we cannot overflow iter->next_index in a single step,
  	 * because RADIX_TREE_MAP_SHIFT < BITS_PER_LONG.
fffaee365   Konstantin Khlebnikov   radix-tree: fix c...
1118
1119
  	 *
  	 * This condition also used by radix_tree_next_slot() to stop
91b9677c4   Matthew Wilcox   radix-tree: fix typo
1120
  	 * contiguous iterating, and forbid switching to the next chunk.
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1121
1122
1123
1124
  	 */
  	index = iter->next_index;
  	if (!index && iter->index)
  		return NULL;
21ef53393   Ross Zwisler   radix-tree: add s...
1125
   restart:
9e85d8111   Matthew Wilcox   radix-tree: make ...
1126
  	radix_tree_load_root(root, &child, &maxindex);
21ef53393   Ross Zwisler   radix-tree: add s...
1127
1128
  	if (index > maxindex)
  		return NULL;
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1129
1130
  	if (!child)
  		return NULL;
21ef53393   Ross Zwisler   radix-tree: add s...
1131

8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1132
  	if (!radix_tree_is_internal_node(child)) {
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1133
  		/* Single-slot tree */
21ef53393   Ross Zwisler   radix-tree: add s...
1134
1135
  		iter->index = index;
  		iter->next_index = maxindex + 1;
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1136
  		iter->tags = 1;
268f42de7   Matthew Wilcox   radix-tree: delet...
1137
  		iter->node = NULL;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
1138
  		return (void __rcu **)&root->xa_head;
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1139
  	}
21ef53393   Ross Zwisler   radix-tree: add s...
1140

8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1141
1142
  	do {
  		node = entry_to_node(child);
9e85d8111   Matthew Wilcox   radix-tree: make ...
1143
  		offset = radix_tree_descend(node, &child, index);
21ef53393   Ross Zwisler   radix-tree: add s...
1144

78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1145
  		if ((flags & RADIX_TREE_ITER_TAGGED) ?
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1146
  				!tag_get(node, tag, offset) : !child) {
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1147
1148
1149
1150
1151
  			/* Hole detected */
  			if (flags & RADIX_TREE_ITER_CONTIG)
  				return NULL;
  
  			if (flags & RADIX_TREE_ITER_TAGGED)
bc412fca6   Matthew Wilcox   radix-tree: make ...
1152
  				offset = radix_tree_find_next_bit(node, tag,
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1153
1154
1155
  						offset + 1);
  			else
  				while (++offset	< RADIX_TREE_MAP_SIZE) {
12320d0ff   Matthew Wilcox   radix-tree: Add r...
1156
1157
  					void *slot = rcu_dereference_raw(
  							node->slots[offset]);
21ef53393   Ross Zwisler   radix-tree: add s...
1158
  					if (slot)
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1159
1160
  						break;
  				}
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1161
  			index &= ~node_maxindex(node);
9e85d8111   Matthew Wilcox   radix-tree: make ...
1162
  			index += offset << node->shift;
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1163
1164
1165
1166
1167
  			/* Overflow after ~0UL */
  			if (!index)
  				return NULL;
  			if (offset == RADIX_TREE_MAP_SIZE)
  				goto restart;
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1168
  			child = rcu_dereference_raw(node->slots[offset]);
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1169
  		}
e157b5559   Matthew Wilcox   radix-tree: add r...
1170
  		if (!child)
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1171
  			goto restart;
e157b5559   Matthew Wilcox   radix-tree: add r...
1172
1173
  		if (child == RADIX_TREE_RETRY)
  			break;
66ee620f0   Matthew Wilcox   idr: Permit any v...
1174
  	} while (node->shift && radix_tree_is_internal_node(child));
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1175
1176
  
  	/* Update the iterator state */
3a08cd52c   Matthew Wilcox   radix tree: Remov...
1177
  	iter->index = (index &~ node_maxindex(node)) | offset;
8c1244de0   Matthew Wilcox   radix-tree: tidy ...
1178
  	iter->next_index = (index | node_maxindex(node)) + 1;
268f42de7   Matthew Wilcox   radix-tree: delet...
1179
  	iter->node = node;
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1180

148deab22   Matthew Wilcox   radix-tree: impro...
1181
1182
  	if (flags & RADIX_TREE_ITER_TAGGED)
  		set_iter_tags(iter, node, offset, tag);
78c1d7848   Konstantin Khlebnikov   radix-tree: intro...
1183
1184
1185
1186
1187
1188
  
  	return node->slots + offset;
  }
  EXPORT_SYMBOL(radix_tree_next_chunk);
  
  /**
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
   *	radix_tree_gang_lookup - perform multiple lookup on a radix tree
   *	@root:		radix tree root
   *	@results:	where the results of the lookup are placed
   *	@first_index:	start the lookup from this key
   *	@max_items:	place up to this many items at *results
   *
   *	Performs an index-ascending scan of the tree for present items.  Places
   *	them at *@results and returns the number of items which were placed at
   *	*@results.
   *
   *	The implementation is naive.
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1200
1201
1202
   *
   *	Like radix_tree_lookup, radix_tree_gang_lookup may be called under
   *	rcu_read_lock. In this case, rather than the returned results being
2fcd9005c   Matthew Wilcox   radix-tree: misce...
1203
1204
1205
1206
   *	an atomic snapshot of the tree at a single point in time, the
   *	semantics of an RCU protected gang lookup are as though multiple
   *	radix_tree_lookups have been issued in individual locks, and results
   *	stored in 'results'.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1207
1208
   */
  unsigned int
35534c869   Matthew Wilcox   radix tree: const...
1209
  radix_tree_gang_lookup(const struct radix_tree_root *root, void **results,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1210
1211
  			unsigned long first_index, unsigned int max_items)
  {
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1212
  	struct radix_tree_iter iter;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1213
  	void __rcu **slot;
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1214
  	unsigned int ret = 0;
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1215

cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1216
  	if (unlikely(!max_items))
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1217
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1218

cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1219
  	radix_tree_for_each_slot(slot, root, &iter, first_index) {
46437f9a5   Matthew Wilcox   radix-tree: fix r...
1220
  		results[ret] = rcu_dereference_raw(*slot);
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1221
1222
  		if (!results[ret])
  			continue;
b194d16c2   Matthew Wilcox   radix-tree: renam...
1223
  		if (radix_tree_is_internal_node(results[ret])) {
46437f9a5   Matthew Wilcox   radix-tree: fix r...
1224
1225
1226
  			slot = radix_tree_iter_retry(&iter);
  			continue;
  		}
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1227
  		if (++ret == max_items)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1228
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1229
  	}
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1230

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1231
1232
1233
  	return ret;
  }
  EXPORT_SYMBOL(radix_tree_gang_lookup);
47feff2c8   Nick Piggin   radix-tree: add g...
1234
  /**
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1235
1236
1237
1238
1239
1240
   *	radix_tree_gang_lookup_tag - perform multiple lookup on a radix tree
   *	                             based on a tag
   *	@root:		radix tree root
   *	@results:	where the results of the lookup are placed
   *	@first_index:	start the lookup from this key
   *	@max_items:	place up to this many items at *results
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
1241
   *	@tag:		the tag index (< RADIX_TREE_MAX_TAGS)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1242
1243
1244
1245
1246
1247
   *
   *	Performs an index-ascending scan of the tree for present items which
   *	have the tag indexed by @tag set.  Places the items at *@results and
   *	returns the number of items which were placed at *@results.
   */
  unsigned int
35534c869   Matthew Wilcox   radix tree: const...
1248
  radix_tree_gang_lookup_tag(const struct radix_tree_root *root, void **results,
daff89f32   Jonathan Corbet   [PATCH] radix-tre...
1249
1250
  		unsigned long first_index, unsigned int max_items,
  		unsigned int tag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1251
  {
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1252
  	struct radix_tree_iter iter;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1253
  	void __rcu **slot;
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1254
  	unsigned int ret = 0;
612d6c19d   Nick Piggin   [PATCH] radix-tre...
1255

cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1256
  	if (unlikely(!max_items))
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1257
  		return 0;
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1258
  	radix_tree_for_each_tagged(slot, root, &iter, first_index, tag) {
46437f9a5   Matthew Wilcox   radix-tree: fix r...
1259
  		results[ret] = rcu_dereference_raw(*slot);
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1260
1261
  		if (!results[ret])
  			continue;
b194d16c2   Matthew Wilcox   radix-tree: renam...
1262
  		if (radix_tree_is_internal_node(results[ret])) {
46437f9a5   Matthew Wilcox   radix-tree: fix r...
1263
1264
1265
  			slot = radix_tree_iter_retry(&iter);
  			continue;
  		}
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1266
  		if (++ret == max_items)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1267
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1268
  	}
7cf9c2c76   Nick Piggin   [PATCH] radix-tre...
1269

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1270
1271
1272
1273
1274
  	return ret;
  }
  EXPORT_SYMBOL(radix_tree_gang_lookup_tag);
  
  /**
47feff2c8   Nick Piggin   radix-tree: add g...
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
   *	radix_tree_gang_lookup_tag_slot - perform multiple slot lookup on a
   *					  radix tree based on a tag
   *	@root:		radix tree root
   *	@results:	where the results of the lookup are placed
   *	@first_index:	start the lookup from this key
   *	@max_items:	place up to this many items at *results
   *	@tag:		the tag index (< RADIX_TREE_MAX_TAGS)
   *
   *	Performs an index-ascending scan of the tree for present items which
   *	have the tag indexed by @tag set.  Places the slots at *@results and
   *	returns the number of slots which were placed at *@results.
   */
  unsigned int
35534c869   Matthew Wilcox   radix tree: const...
1288
  radix_tree_gang_lookup_tag_slot(const struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1289
  		void __rcu ***results, unsigned long first_index,
35534c869   Matthew Wilcox   radix tree: const...
1290
  		unsigned int max_items, unsigned int tag)
47feff2c8   Nick Piggin   radix-tree: add g...
1291
  {
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1292
  	struct radix_tree_iter iter;
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1293
  	void __rcu **slot;
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1294
  	unsigned int ret = 0;
47feff2c8   Nick Piggin   radix-tree: add g...
1295

cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1296
  	if (unlikely(!max_items))
47feff2c8   Nick Piggin   radix-tree: add g...
1297
  		return 0;
cebbd29e1   Konstantin Khlebnikov   radix-tree: rewri...
1298
1299
1300
  	radix_tree_for_each_tagged(slot, root, &iter, first_index, tag) {
  		results[ret] = slot;
  		if (++ret == max_items)
47feff2c8   Nick Piggin   radix-tree: add g...
1301
  			break;
47feff2c8   Nick Piggin   radix-tree: add g...
1302
1303
1304
1305
1306
  	}
  
  	return ret;
  }
  EXPORT_SYMBOL(radix_tree_gang_lookup_tag_slot);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1307
  static bool __radix_tree_delete(struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1308
  				struct radix_tree_node *node, void __rcu **slot)
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1309
  {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1310
  	void *old = rcu_dereference_raw(*slot);
01959dfe7   Matthew Wilcox   xarray: Define st...
1311
  	int values = xa_is_value(old) ? -1 : 0;
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1312
1313
  	unsigned offset = get_slot_offset(node, slot);
  	int tag;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1314
1315
1316
1317
1318
  	if (is_idr(root))
  		node_tag_set(root, node, IDR_FREE, offset);
  	else
  		for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++)
  			node_tag_clear(root, node, tag, offset);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1319

01959dfe7   Matthew Wilcox   xarray: Define st...
1320
  	replace_slot(slot, NULL, node, -1, values);
1cf56f9d6   Matthew Wilcox   radix tree: Remov...
1321
  	return node && delete_node(root, node);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1322
  }
139e56166   Johannes Weiner   lib: radix_tree: ...
1323
  /**
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1324
1325
1326
1327
   * radix_tree_iter_delete - delete the entry at this iterator position
   * @root: radix tree root
   * @iter: iterator state
   * @slot: pointer to slot
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1328
   *
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1329
1330
1331
1332
1333
1334
1335
   * Delete the entry at the position currently pointed to by the iterator.
   * This may result in the current node being freed; if it is, the iterator
   * is advanced so that it will not reference the freed memory.  This
   * function may be called without any locking if there are no other threads
   * which can access this tree.
   */
  void radix_tree_iter_delete(struct radix_tree_root *root,
d7b627277   Matthew Wilcox   radix-tree: Fix _...
1336
  				struct radix_tree_iter *iter, void __rcu **slot)
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1337
1338
1339
1340
  {
  	if (__radix_tree_delete(root, iter->node, slot))
  		iter->index = iter->next_index;
  }
d1b48c1e7   Chris Wilson   drm/i915: Replace...
1341
  EXPORT_SYMBOL(radix_tree_iter_delete);
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1342
1343
1344
1345
1346
1347
  
  /**
   * radix_tree_delete_item - delete an item from a radix tree
   * @root: radix tree root
   * @index: index key
   * @item: expected item
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1348
   *
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1349
   * Remove @item at @index from the radix tree rooted at @root.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1350
   *
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1351
1352
   * Return: the deleted entry, or %NULL if it was not present
   * or the entry at the given @index was not @item.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1353
   */
53c59f262   Johannes Weiner   lib: radix-tree: ...
1354
1355
  void *radix_tree_delete_item(struct radix_tree_root *root,
  			     unsigned long index, void *item)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1356
  {
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1357
  	struct radix_tree_node *node = NULL;
7a4deea1a   Matthew Wilcox   idr: fix invalid ...
1358
  	void __rcu **slot = NULL;
139e56166   Johannes Weiner   lib: radix_tree: ...
1359
  	void *entry;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1360

139e56166   Johannes Weiner   lib: radix_tree: ...
1361
  	entry = __radix_tree_lookup(root, index, &node, &slot);
7a4deea1a   Matthew Wilcox   idr: fix invalid ...
1362
1363
  	if (!slot)
  		return NULL;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1364
1365
  	if (!entry && (!is_idr(root) || node_tag_get(root, node, IDR_FREE,
  						get_slot_offset(node, slot))))
139e56166   Johannes Weiner   lib: radix_tree: ...
1366
  		return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1367

139e56166   Johannes Weiner   lib: radix_tree: ...
1368
1369
  	if (item && entry != item)
  		return NULL;
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1370
  	__radix_tree_delete(root, node, slot);
612d6c19d   Nick Piggin   [PATCH] radix-tre...
1371

139e56166   Johannes Weiner   lib: radix_tree: ...
1372
  	return entry;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1373
  }
53c59f262   Johannes Weiner   lib: radix-tree: ...
1374
1375
1376
  EXPORT_SYMBOL(radix_tree_delete_item);
  
  /**
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1377
1378
1379
   * radix_tree_delete - delete an entry from a radix tree
   * @root: radix tree root
   * @index: index key
53c59f262   Johannes Weiner   lib: radix-tree: ...
1380
   *
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1381
   * Remove the entry at @index from the radix tree rooted at @root.
53c59f262   Johannes Weiner   lib: radix-tree: ...
1382
   *
0ac398ef3   Matthew Wilcox   radix-tree: Add r...
1383
   * Return: The deleted entry, or %NULL if it was not present.
53c59f262   Johannes Weiner   lib: radix-tree: ...
1384
1385
1386
1387
1388
   */
  void *radix_tree_delete(struct radix_tree_root *root, unsigned long index)
  {
  	return radix_tree_delete_item(root, index, NULL);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1389
1390
1391
1392
1393
1394
1395
  EXPORT_SYMBOL(radix_tree_delete);
  
  /**
   *	radix_tree_tagged - test whether any items in the tree are tagged
   *	@root:		radix tree root
   *	@tag:		tag to test
   */
35534c869   Matthew Wilcox   radix tree: const...
1396
  int radix_tree_tagged(const struct radix_tree_root *root, unsigned int tag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1397
  {
612d6c19d   Nick Piggin   [PATCH] radix-tre...
1398
  	return root_tag_get(root, tag);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1399
1400
  }
  EXPORT_SYMBOL(radix_tree_tagged);
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1401
1402
1403
1404
1405
1406
1407
1408
1409
  /**
   * idr_preload - preload for idr_alloc()
   * @gfp_mask: allocation mask to use for preloading
   *
   * Preallocate memory to use for the next call to idr_alloc().  This function
   * returns with preemption disabled.  It will be enabled by idr_preload_end().
   */
  void idr_preload(gfp_t gfp_mask)
  {
bc9ae2247   Eric Dumazet   radix-tree: must ...
1410
1411
  	if (__radix_tree_preload(gfp_mask, IDR_PRELOAD_SIZE))
  		preempt_disable();
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1412
1413
  }
  EXPORT_SYMBOL(idr_preload);
460488c58   Matthew Wilcox   idr: Remove idr_a...
1414
  void __rcu **idr_get_free(struct radix_tree_root *root,
388f79fda   Chris Mi   idr: Add new APIs...
1415
1416
  			      struct radix_tree_iter *iter, gfp_t gfp,
  			      unsigned long max)
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1417
1418
  {
  	struct radix_tree_node *node = NULL, *child;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
1419
  	void __rcu **slot = (void __rcu **)&root->xa_head;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1420
  	unsigned long maxindex, start = iter->next_index;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
  	unsigned int shift, offset = 0;
  
   grow:
  	shift = radix_tree_load_root(root, &child, &maxindex);
  	if (!radix_tree_tagged(root, IDR_FREE))
  		start = max(start, maxindex + 1);
  	if (start > max)
  		return ERR_PTR(-ENOSPC);
  
  	if (start > maxindex) {
  		int error = radix_tree_extend(root, gfp, start, shift);
  		if (error < 0)
  			return ERR_PTR(error);
  		shift = error;
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
1435
  		child = rcu_dereference_raw(root->xa_head);
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1436
  	}
66ee620f0   Matthew Wilcox   idr: Permit any v...
1437
1438
  	if (start == 0 && shift == 0)
  		shift = RADIX_TREE_MAP_SHIFT;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1439
1440
1441
1442
1443
  
  	while (shift) {
  		shift -= RADIX_TREE_MAP_SHIFT;
  		if (child == NULL) {
  			/* Have to add a child node.  */
d58275bc9   Matthew Wilcox   radix-tree: Store...
1444
1445
  			child = radix_tree_node_alloc(gfp, node, root, shift,
  							offset, 0, 0);
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
  			if (!child)
  				return ERR_PTR(-ENOMEM);
  			all_tag_set(child, IDR_FREE);
  			rcu_assign_pointer(*slot, node_to_entry(child));
  			if (node)
  				node->count++;
  		} else if (!radix_tree_is_internal_node(child))
  			break;
  
  		node = entry_to_node(child);
  		offset = radix_tree_descend(node, &child, start);
  		if (!tag_get(node, IDR_FREE, offset)) {
  			offset = radix_tree_find_next_bit(node, IDR_FREE,
  							offset + 1);
  			start = next_index(start, node, offset);
b7e9728f3   Matthew Wilcox (Oracle)   idr: Fix idr_allo...
1461
  			if (start > max || start == 0)
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
  				return ERR_PTR(-ENOSPC);
  			while (offset == RADIX_TREE_MAP_SIZE) {
  				offset = node->offset + 1;
  				node = node->parent;
  				if (!node)
  					goto grow;
  				shift = node->shift;
  			}
  			child = rcu_dereference_raw(node->slots[offset]);
  		}
  		slot = &node->slots[offset];
  	}
  
  	iter->index = start;
  	if (node)
  		iter->next_index = 1 + min(max, (start | node_maxindex(node)));
  	else
  		iter->next_index = 1;
  	iter->node = node;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
  	set_iter_tags(iter, node, offset, IDR_FREE);
  
  	return slot;
  }
  
  /**
   * idr_destroy - release all internal memory from an IDR
   * @idr: idr handle
   *
   * After this function is called, the IDR is empty, and may be reused or
   * the data structure containing it may be freed.
   *
   * A typical clean-up sequence for objects stored in an idr tree will use
   * idr_for_each() to free all objects, if necessary, then idr_destroy() to
   * free the memory used to keep track of those objects.
   */
  void idr_destroy(struct idr *idr)
  {
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
1499
  	struct radix_tree_node *node = rcu_dereference_raw(idr->idr_rt.xa_head);
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1500
1501
  	if (radix_tree_is_internal_node(node))
  		radix_tree_free_nodes(node);
f8d5d0cc1   Matthew Wilcox   xarray: Add defin...
1502
  	idr->idr_rt.xa_head = NULL;
0a835c4f0   Matthew Wilcox   Reimplement IDR a...
1503
1504
1505
  	root_tag_set(&idr->idr_rt, IDR_FREE);
  }
  EXPORT_SYMBOL(idr_destroy);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1506
  static void
449dd6984   Johannes Weiner   mm: keep page cac...
1507
  radix_tree_node_ctor(void *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1508
  {
449dd6984   Johannes Weiner   mm: keep page cac...
1509
1510
1511
1512
  	struct radix_tree_node *node = arg;
  
  	memset(node, 0, sizeof(*node));
  	INIT_LIST_HEAD(&node->private_list);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1513
  }
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1514
  static int radix_tree_cpu_dead(unsigned int cpu)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1515
  {
2fcd9005c   Matthew Wilcox   radix-tree: misce...
1516
1517
1518
1519
  	struct radix_tree_preload *rtp;
  	struct radix_tree_node *node;
  
  	/* Free per-cpu pool of preloaded nodes */
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1520
1521
1522
  	rtp = &per_cpu(radix_tree_preloads, cpu);
  	while (rtp->nr) {
  		node = rtp->nodes;
1293d5c5f   Matthew Wilcox   radix-tree: Chain...
1523
  		rtp->nodes = node->parent;
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1524
1525
  		kmem_cache_free(radix_tree_node_cachep, node);
  		rtp->nr--;
2fcd9005c   Matthew Wilcox   radix-tree: misce...
1526
  	}
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1527
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1528
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1529
1530
1531
  
  void __init radix_tree_init(void)
  {
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1532
  	int ret;
7e7844226   Michal Hocko   lockdep: allow to...
1533
1534
  
  	BUILD_BUG_ON(RADIX_TREE_MAX_TAGS + __GFP_BITS_SHIFT > 32);
fa290cda1   Matthew Wilcox   radix tree: use G...
1535
  	BUILD_BUG_ON(ROOT_IS_IDR & ~GFP_ZONEMASK);
02c02bf12   Matthew Wilcox   xarray: Change de...
1536
  	BUILD_BUG_ON(XA_CHUNK_SIZE > 255);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1537
1538
  	radix_tree_node_cachep = kmem_cache_create("radix_tree_node",
  			sizeof(struct radix_tree_node), 0,
488514d17   Christoph Lameter   Remove set_migrat...
1539
1540
  			SLAB_PANIC | SLAB_RECLAIM_ACCOUNT,
  			radix_tree_node_ctor);
d544abd5f   Sebastian Andrzej Siewior   lib/radix-tree: C...
1541
1542
1543
  	ret = cpuhp_setup_state_nocalls(CPUHP_RADIX_DEAD, "lib/radix:dead",
  					NULL, radix_tree_cpu_dead);
  	WARN_ON(ret < 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1544
  }