mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-25 00:52:06 +00:00 
			
		
		
		
	radix-tree: use indirect bit
Rather than sign direct radix-tree pointers with a special bit, sign the indirect one that hangs off the root. This means that, given a lookup_slot operation, the invalid result will be differentiated from the valid (previously, valid results could have the bit either set or clear). This does not affect slot lookups which occur under lock -- they can never return an invalid result. Is needed in future for lockless pagecache. Signed-off-by: Nick Piggin <npiggin@suse.de> Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl> Cc: Hugh Dickins <hugh@veritas.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
		
							parent
							
								
									b55ed81623
								
							
						
					
					
						commit
						c0bc9875b7
					
				| @ -26,28 +26,31 @@ | |||||||
| #include <linux/rcupdate.h> | #include <linux/rcupdate.h> | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * A direct pointer (root->rnode pointing directly to a data item, |  * An indirect pointer (root->rnode pointing to a radix_tree_node, rather | ||||||
|  * rather than another radix_tree_node) is signalled by the low bit |  * than a data item) is signalled by the low bit set in the root->rnode | ||||||
|  * set in the root->rnode pointer. |  * pointer. | ||||||
|  * |  * | ||||||
|  * In this case root->height is also NULL, but the direct pointer tests are |  * In this case root->height is > 0, but the indirect pointer tests are | ||||||
|  * needed for RCU lookups when root->height is unreliable. |  * needed for RCU lookups (because root->height is unreliable). The only | ||||||
|  |  * time callers need worry about this is when doing a lookup_slot under | ||||||
|  |  * RCU. | ||||||
|  */ |  */ | ||||||
| #define RADIX_TREE_DIRECT_PTR	1 | #define RADIX_TREE_INDIRECT_PTR	1 | ||||||
|  | #define RADIX_TREE_RETRY ((void *)-1UL) | ||||||
| 
 | 
 | ||||||
| static inline void *radix_tree_ptr_to_direct(void *ptr) | static inline void *radix_tree_ptr_to_indirect(void *ptr) | ||||||
| { | { | ||||||
| 	return (void *)((unsigned long)ptr | RADIX_TREE_DIRECT_PTR); | 	return (void *)((unsigned long)ptr | RADIX_TREE_INDIRECT_PTR); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline void *radix_tree_direct_to_ptr(void *ptr) | static inline void *radix_tree_indirect_to_ptr(void *ptr) | ||||||
| { | { | ||||||
| 	return (void *)((unsigned long)ptr & ~RADIX_TREE_DIRECT_PTR); | 	return (void *)((unsigned long)ptr & ~RADIX_TREE_INDIRECT_PTR); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline int radix_tree_is_direct_ptr(void *ptr) | static inline int radix_tree_is_indirect_ptr(void *ptr) | ||||||
| { | { | ||||||
| 	return (int)((unsigned long)ptr & RADIX_TREE_DIRECT_PTR); | 	return (int)((unsigned long)ptr & RADIX_TREE_INDIRECT_PTR); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*** radix-tree API starts here ***/ | /*** radix-tree API starts here ***/ | ||||||
| @ -130,7 +133,10 @@ do {									\ | |||||||
|  */ |  */ | ||||||
| static inline void *radix_tree_deref_slot(void **pslot) | static inline void *radix_tree_deref_slot(void **pslot) | ||||||
| { | { | ||||||
| 	return radix_tree_direct_to_ptr(*pslot); | 	void *ret = *pslot; | ||||||
|  | 	if (unlikely(radix_tree_is_indirect_ptr(ret))) | ||||||
|  | 		ret = RADIX_TREE_RETRY; | ||||||
|  | 	return ret; | ||||||
| } | } | ||||||
| /**
 | /**
 | ||||||
|  * radix_tree_replace_slot	- replace item in a slot |  * radix_tree_replace_slot	- replace item in a slot | ||||||
| @ -142,10 +148,8 @@ static inline void *radix_tree_deref_slot(void **pslot) | |||||||
|  */ |  */ | ||||||
| static inline void radix_tree_replace_slot(void **pslot, void *item) | static inline void radix_tree_replace_slot(void **pslot, void *item) | ||||||
| { | { | ||||||
| 	BUG_ON(radix_tree_is_direct_ptr(item)); | 	BUG_ON(radix_tree_is_indirect_ptr(item)); | ||||||
| 	rcu_assign_pointer(*pslot, | 	rcu_assign_pointer(*pslot, item); | ||||||
| 		(void *)((unsigned long)item | |  | ||||||
| 			((unsigned long)*pslot & RADIX_TREE_DIRECT_PTR))); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int radix_tree_insert(struct radix_tree_root *, unsigned long, void *); | int radix_tree_insert(struct radix_tree_root *, unsigned long, void *); | ||||||
|  | |||||||
| @ -104,7 +104,7 @@ radix_tree_node_alloc(struct radix_tree_root *root) | |||||||
| 			rtp->nr--; | 			rtp->nr--; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	BUG_ON(radix_tree_is_direct_ptr(ret)); | 	BUG_ON(radix_tree_is_indirect_ptr(ret)); | ||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -240,7 +240,7 @@ static int radix_tree_extend(struct radix_tree_root *root, unsigned long index) | |||||||
| 			return -ENOMEM; | 			return -ENOMEM; | ||||||
| 
 | 
 | ||||||
| 		/* Increase the height.  */ | 		/* Increase the height.  */ | ||||||
| 		node->slots[0] = radix_tree_direct_to_ptr(root->rnode); | 		node->slots[0] = radix_tree_indirect_to_ptr(root->rnode); | ||||||
| 
 | 
 | ||||||
| 		/* Propagate the aggregated tag info into the new root */ | 		/* Propagate the aggregated tag info into the new root */ | ||||||
| 		for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) { | 		for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) { | ||||||
| @ -251,6 +251,7 @@ static int radix_tree_extend(struct radix_tree_root *root, unsigned long index) | |||||||
| 		newheight = root->height+1; | 		newheight = root->height+1; | ||||||
| 		node->height = newheight; | 		node->height = newheight; | ||||||
| 		node->count = 1; | 		node->count = 1; | ||||||
|  | 		node = radix_tree_ptr_to_indirect(node); | ||||||
| 		rcu_assign_pointer(root->rnode, node); | 		rcu_assign_pointer(root->rnode, node); | ||||||
| 		root->height = newheight; | 		root->height = newheight; | ||||||
| 	} while (height > root->height); | 	} while (height > root->height); | ||||||
| @ -274,7 +275,7 @@ int radix_tree_insert(struct radix_tree_root *root, | |||||||
| 	int offset; | 	int offset; | ||||||
| 	int error; | 	int error; | ||||||
| 
 | 
 | ||||||
| 	BUG_ON(radix_tree_is_direct_ptr(item)); | 	BUG_ON(radix_tree_is_indirect_ptr(item)); | ||||||
| 
 | 
 | ||||||
| 	/* Make sure the tree is high enough.  */ | 	/* Make sure the tree is high enough.  */ | ||||||
| 	if (index > radix_tree_maxindex(root->height)) { | 	if (index > radix_tree_maxindex(root->height)) { | ||||||
| @ -283,7 +284,8 @@ int radix_tree_insert(struct radix_tree_root *root, | |||||||
| 			return error; | 			return error; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	slot = root->rnode; | 	slot = radix_tree_indirect_to_ptr(root->rnode); | ||||||
|  | 
 | ||||||
| 	height = root->height; | 	height = root->height; | ||||||
| 	shift = (height-1) * RADIX_TREE_MAP_SHIFT; | 	shift = (height-1) * RADIX_TREE_MAP_SHIFT; | ||||||
| 
 | 
 | ||||||
| @ -298,7 +300,8 @@ int radix_tree_insert(struct radix_tree_root *root, | |||||||
| 				rcu_assign_pointer(node->slots[offset], slot); | 				rcu_assign_pointer(node->slots[offset], slot); | ||||||
| 				node->count++; | 				node->count++; | ||||||
| 			} else | 			} else | ||||||
| 				rcu_assign_pointer(root->rnode, slot); | 				rcu_assign_pointer(root->rnode, | ||||||
|  | 					radix_tree_ptr_to_indirect(slot)); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* Go a level down */ | 		/* Go a level down */ | ||||||
| @ -318,7 +321,7 @@ int radix_tree_insert(struct radix_tree_root *root, | |||||||
| 		BUG_ON(tag_get(node, 0, offset)); | 		BUG_ON(tag_get(node, 0, offset)); | ||||||
| 		BUG_ON(tag_get(node, 1, offset)); | 		BUG_ON(tag_get(node, 1, offset)); | ||||||
| 	} else { | 	} else { | ||||||
| 		rcu_assign_pointer(root->rnode, radix_tree_ptr_to_direct(item)); | 		rcu_assign_pointer(root->rnode, item); | ||||||
| 		BUG_ON(root_tag_get(root, 0)); | 		BUG_ON(root_tag_get(root, 0)); | ||||||
| 		BUG_ON(root_tag_get(root, 1)); | 		BUG_ON(root_tag_get(root, 1)); | ||||||
| 	} | 	} | ||||||
| @ -350,11 +353,12 @@ void **radix_tree_lookup_slot(struct radix_tree_root *root, unsigned long index) | |||||||
| 	if (node == NULL) | 	if (node == NULL) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	if (radix_tree_is_direct_ptr(node)) { | 	if (!radix_tree_is_indirect_ptr(node)) { | ||||||
| 		if (index > 0) | 		if (index > 0) | ||||||
| 			return NULL; | 			return NULL; | ||||||
| 		return (void **)&root->rnode; | 		return (void **)&root->rnode; | ||||||
| 	} | 	} | ||||||
|  | 	node = radix_tree_indirect_to_ptr(node); | ||||||
| 
 | 
 | ||||||
| 	height = node->height; | 	height = node->height; | ||||||
| 	if (index > radix_tree_maxindex(height)) | 	if (index > radix_tree_maxindex(height)) | ||||||
| @ -398,11 +402,12 @@ void *radix_tree_lookup(struct radix_tree_root *root, unsigned long index) | |||||||
| 	if (node == NULL) | 	if (node == NULL) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	if (radix_tree_is_direct_ptr(node)) { | 	if (!radix_tree_is_indirect_ptr(node)) { | ||||||
| 		if (index > 0) | 		if (index > 0) | ||||||
| 			return NULL; | 			return NULL; | ||||||
| 		return radix_tree_direct_to_ptr(node); | 		return node; | ||||||
| 	} | 	} | ||||||
|  | 	node = radix_tree_indirect_to_ptr(node); | ||||||
| 
 | 
 | ||||||
| 	height = node->height; | 	height = node->height; | ||||||
| 	if (index > radix_tree_maxindex(height)) | 	if (index > radix_tree_maxindex(height)) | ||||||
| @ -447,7 +452,7 @@ void *radix_tree_tag_set(struct radix_tree_root *root, | |||||||
| 	height = root->height; | 	height = root->height; | ||||||
| 	BUG_ON(index > radix_tree_maxindex(height)); | 	BUG_ON(index > radix_tree_maxindex(height)); | ||||||
| 
 | 
 | ||||||
| 	slot = root->rnode; | 	slot = radix_tree_indirect_to_ptr(root->rnode); | ||||||
| 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | ||||||
| 
 | 
 | ||||||
| 	while (height > 0) { | 	while (height > 0) { | ||||||
| @ -497,7 +502,7 @@ void *radix_tree_tag_clear(struct radix_tree_root *root, | |||||||
| 
 | 
 | ||||||
| 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | ||||||
| 	pathp->node = NULL; | 	pathp->node = NULL; | ||||||
| 	slot = root->rnode; | 	slot = radix_tree_indirect_to_ptr(root->rnode); | ||||||
| 
 | 
 | ||||||
| 	while (height > 0) { | 	while (height > 0) { | ||||||
| 		int offset; | 		int offset; | ||||||
| @ -562,8 +567,9 @@ int radix_tree_tag_get(struct radix_tree_root *root, | |||||||
| 	if (node == NULL) | 	if (node == NULL) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (radix_tree_is_direct_ptr(node)) | 	if (!radix_tree_is_indirect_ptr(node)) | ||||||
| 		return (index == 0); | 		return (index == 0); | ||||||
|  | 	node = radix_tree_indirect_to_ptr(node); | ||||||
| 
 | 
 | ||||||
| 	height = node->height; | 	height = node->height; | ||||||
| 	if (index > radix_tree_maxindex(height)) | 	if (index > radix_tree_maxindex(height)) | ||||||
| @ -716,13 +722,13 @@ radix_tree_gang_lookup(struct radix_tree_root *root, void **results, | |||||||
| 	if (!node) | 	if (!node) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (radix_tree_is_direct_ptr(node)) { | 	if (!radix_tree_is_indirect_ptr(node)) { | ||||||
| 		if (first_index > 0) | 		if (first_index > 0) | ||||||
| 			return 0; | 			return 0; | ||||||
| 		node = radix_tree_direct_to_ptr(node); | 		results[0] = node; | ||||||
| 		results[0] = rcu_dereference(node); |  | ||||||
| 		return 1; | 		return 1; | ||||||
| 	} | 	} | ||||||
|  | 	node = radix_tree_indirect_to_ptr(node); | ||||||
| 
 | 
 | ||||||
| 	max_index = radix_tree_maxindex(node->height); | 	max_index = radix_tree_maxindex(node->height); | ||||||
| 
 | 
 | ||||||
| @ -844,13 +850,13 @@ radix_tree_gang_lookup_tag(struct radix_tree_root *root, void **results, | |||||||
| 	if (!node) | 	if (!node) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (radix_tree_is_direct_ptr(node)) { | 	if (!radix_tree_is_indirect_ptr(node)) { | ||||||
| 		if (first_index > 0) | 		if (first_index > 0) | ||||||
| 			return 0; | 			return 0; | ||||||
| 		node = radix_tree_direct_to_ptr(node); | 		results[0] = node; | ||||||
| 		results[0] = rcu_dereference(node); |  | ||||||
| 		return 1; | 		return 1; | ||||||
| 	} | 	} | ||||||
|  | 	node = radix_tree_indirect_to_ptr(node); | ||||||
| 
 | 
 | ||||||
| 	max_index = radix_tree_maxindex(node->height); | 	max_index = radix_tree_maxindex(node->height); | ||||||
| 
 | 
 | ||||||
| @ -880,12 +886,22 @@ EXPORT_SYMBOL(radix_tree_gang_lookup_tag); | |||||||
| static inline void radix_tree_shrink(struct radix_tree_root *root) | static inline void radix_tree_shrink(struct radix_tree_root *root) | ||||||
| { | { | ||||||
| 	/* try to shrink tree height */ | 	/* try to shrink tree height */ | ||||||
| 	while (root->height > 0 && | 	while (root->height > 0) { | ||||||
| 			root->rnode->count == 1 && |  | ||||||
| 			root->rnode->slots[0]) { |  | ||||||
| 		struct radix_tree_node *to_free = root->rnode; | 		struct radix_tree_node *to_free = root->rnode; | ||||||
| 		void *newptr; | 		void *newptr; | ||||||
| 
 | 
 | ||||||
|  | 		BUG_ON(!radix_tree_is_indirect_ptr(to_free)); | ||||||
|  | 		to_free = radix_tree_indirect_to_ptr(to_free); | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * The candidate node has more than one child, or its child | ||||||
|  | 		 * is not at the leftmost slot, we cannot shrink. | ||||||
|  | 		 */ | ||||||
|  | 		if (to_free->count != 1) | ||||||
|  | 			break; | ||||||
|  | 		if (!to_free->slots[0]) | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * We don't need rcu_assign_pointer(), since we are simply | 		 * We don't need rcu_assign_pointer(), since we are simply | ||||||
| 		 * moving the node from one part of the tree to another. If | 		 * moving the node from one part of the tree to another. If | ||||||
| @ -894,8 +910,8 @@ static inline void radix_tree_shrink(struct radix_tree_root *root) | |||||||
| 		 * one (root->rnode). | 		 * one (root->rnode). | ||||||
| 		 */ | 		 */ | ||||||
| 		newptr = to_free->slots[0]; | 		newptr = to_free->slots[0]; | ||||||
| 		if (root->height == 1) | 		if (root->height > 1) | ||||||
| 			newptr = radix_tree_ptr_to_direct(newptr); | 			newptr = radix_tree_ptr_to_indirect(newptr); | ||||||
| 		root->rnode = newptr; | 		root->rnode = newptr; | ||||||
| 		root->height--; | 		root->height--; | ||||||
| 		/* must only free zeroed nodes into the slab */ | 		/* must only free zeroed nodes into the slab */ | ||||||
| @ -930,12 +946,12 @@ void *radix_tree_delete(struct radix_tree_root *root, unsigned long index) | |||||||
| 		goto out; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	slot = root->rnode; | 	slot = root->rnode; | ||||||
| 	if (height == 0 && root->rnode) { | 	if (height == 0) { | ||||||
| 		slot = radix_tree_direct_to_ptr(slot); |  | ||||||
| 		root_tag_clear_all(root); | 		root_tag_clear_all(root); | ||||||
| 		root->rnode = NULL; | 		root->rnode = NULL; | ||||||
| 		goto out; | 		goto out; | ||||||
| 	} | 	} | ||||||
|  | 	slot = radix_tree_indirect_to_ptr(slot); | ||||||
| 
 | 
 | ||||||
| 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | 	shift = (height - 1) * RADIX_TREE_MAP_SHIFT; | ||||||
| 	pathp->node = NULL; | 	pathp->node = NULL; | ||||||
| @ -977,7 +993,8 @@ void *radix_tree_delete(struct radix_tree_root *root, unsigned long index) | |||||||
| 			radix_tree_node_free(to_free); | 			radix_tree_node_free(to_free); | ||||||
| 
 | 
 | ||||||
| 		if (pathp->node->count) { | 		if (pathp->node->count) { | ||||||
| 			if (pathp->node == root->rnode) | 			if (pathp->node == | ||||||
|  | 					radix_tree_indirect_to_ptr(root->rnode)) | ||||||
| 				radix_tree_shrink(root); | 				radix_tree_shrink(root); | ||||||
| 			goto out; | 			goto out; | ||||||
| 		} | 		} | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Nick Piggin
						Nick Piggin