drm/vmwgfx: Stop checking minor type directly
[cascardo/linux.git] / lib / radix-tree.c
index 1a82066..8b7d845 100644 (file)
@@ -94,9 +94,10 @@ static inline unsigned long get_slot_offset(struct radix_tree_node *parent,
        return slot - parent->slots;
 }
 
-static unsigned radix_tree_descend(struct radix_tree_node *parent,
-                               struct radix_tree_node **nodep, unsigned offset)
+static unsigned int radix_tree_descend(struct radix_tree_node *parent,
+                       struct radix_tree_node **nodep, unsigned long index)
 {
+       unsigned int offset = (index >> parent->shift) & RADIX_TREE_MAP_MASK;
        void **entry = rcu_dereference_raw(parent->slots[offset]);
 
 #ifdef CONFIG_RADIX_TREE_MULTIORDER
@@ -499,12 +500,13 @@ int __radix_tree_create(struct radix_tree_root *root, unsigned long index,
                        unsigned order, struct radix_tree_node **nodep,
                        void ***slotp)
 {
-       struct radix_tree_node *node = NULL, *slot;
+       struct radix_tree_node *node = NULL, *child;
+       void **slot = (void **)&root->rnode;
        unsigned long maxindex;
-       unsigned int shift, offset;
+       unsigned int shift, offset = 0;
        unsigned long max = index | ((1UL << order) - 1);
 
-       shift = radix_tree_load_root(root, &slot, &maxindex);
+       shift = radix_tree_load_root(root, &child, &maxindex);
 
        /* Make sure the tree is high enough.  */
        if (max > maxindex) {
@@ -512,51 +514,47 @@ int __radix_tree_create(struct radix_tree_root *root, unsigned long index,
                if (error < 0)
                        return error;
                shift = error;
-               slot = root->rnode;
+               child = root->rnode;
                if (order == shift)
                        shift += RADIX_TREE_MAP_SHIFT;
        }
 
-       offset = 0;                     /* uninitialised var warning */
        while (shift > order) {
                shift -= RADIX_TREE_MAP_SHIFT;
-               if (slot == NULL) {
+               if (child == NULL) {
                        /* Have to add a child node.  */
-                       slot = radix_tree_node_alloc(root);
-                       if (!slot)
+                       child = radix_tree_node_alloc(root);
+                       if (!child)
                                return -ENOMEM;
-                       slot->shift = shift;
-                       slot->offset = offset;
-                       slot->parent = node;
-                       if (node) {
-                               rcu_assign_pointer(node->slots[offset],
-                                                       node_to_entry(slot));
+                       child->shift = shift;
+                       child->offset = offset;
+                       child->parent = node;
+                       rcu_assign_pointer(*slot, node_to_entry(child));
+                       if (node)
                                node->count++;
-                       } else
-                               rcu_assign_pointer(root->rnode,
-                                                       node_to_entry(slot));
-               } else if (!radix_tree_is_internal_node(slot))
+               } else if (!radix_tree_is_internal_node(child))
                        break;
 
                /* Go a level down */
-               node = entry_to_node(slot);
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-               offset = radix_tree_descend(node, &slot, offset);
+               node = entry_to_node(child);
+               offset = radix_tree_descend(node, &child, index);
+               slot = &node->slots[offset];
        }
 
 #ifdef CONFIG_RADIX_TREE_MULTIORDER
        /* Insert pointers to the canonical entry */
        if (order > shift) {
-               int i, n = 1 << (order - shift);
+               unsigned i, n = 1 << (order - shift);
                offset = offset & ~(n - 1);
-               slot = node_to_entry(&node->slots[offset]);
+               slot = &node->slots[offset];
+               child = node_to_entry(slot);
                for (i = 0; i < n; i++) {
-                       if (node->slots[offset + i])
+                       if (slot[i])
                                return -EEXIST;
                }
 
                for (i = 1; i < n; i++) {
-                       rcu_assign_pointer(node->slots[offset + i], slot);
+                       rcu_assign_pointer(slot[i], child);
                        node->count++;
                }
        }
@@ -565,7 +563,7 @@ int __radix_tree_create(struct radix_tree_root *root, unsigned long index,
        if (nodep)
                *nodep = node;
        if (slotp)
-               *slotp = node ? node->slots + offset : (void **)&root->rnode;
+               *slotp = slot;
        return 0;
 }
 
@@ -627,13 +625,12 @@ void *__radix_tree_lookup(struct radix_tree_root *root, unsigned long index,
 {
        struct radix_tree_node *node, *parent;
        unsigned long maxindex;
-       unsigned int shift;
        void **slot;
 
  restart:
        parent = NULL;
        slot = (void **)&root->rnode;
-       shift = radix_tree_load_root(root, &node, &maxindex);
+       radix_tree_load_root(root, &node, &maxindex);
        if (index > maxindex)
                return NULL;
 
@@ -643,9 +640,7 @@ void *__radix_tree_lookup(struct radix_tree_root *root, unsigned long index,
                if (node == RADIX_TREE_RETRY)
                        goto restart;
                parent = entry_to_node(node);
-               shift -= RADIX_TREE_MAP_SHIFT;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-               offset = radix_tree_descend(parent, &node, offset);
+               offset = radix_tree_descend(parent, &node, index);
                slot = parent->slots + offset;
        }
 
@@ -715,19 +710,15 @@ void *radix_tree_tag_set(struct radix_tree_root *root,
 {
        struct radix_tree_node *node, *parent;
        unsigned long maxindex;
-       unsigned int shift;
 
-       shift = radix_tree_load_root(root, &node, &maxindex);
+       radix_tree_load_root(root, &node, &maxindex);
        BUG_ON(index > maxindex);
 
        while (radix_tree_is_internal_node(node)) {
                unsigned offset;
 
-               shift -= RADIX_TREE_MAP_SHIFT;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-
                parent = entry_to_node(node);
-               offset = radix_tree_descend(parent, &node, offset);
+               offset = radix_tree_descend(parent, &node, index);
                BUG_ON(!node);
 
                if (!tag_get(parent, tag, offset))
@@ -742,6 +733,26 @@ void *radix_tree_tag_set(struct radix_tree_root *root,
 }
 EXPORT_SYMBOL(radix_tree_tag_set);
 
+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);
+}
+
 /**
  *     radix_tree_tag_clear - clear a tag on a radix tree node
  *     @root:          radix tree root
@@ -761,45 +772,22 @@ void *radix_tree_tag_clear(struct radix_tree_root *root,
 {
        struct radix_tree_node *node, *parent;
        unsigned long maxindex;
-       unsigned int shift;
        int uninitialized_var(offset);
 
-       shift = radix_tree_load_root(root, &node, &maxindex);
+       radix_tree_load_root(root, &node, &maxindex);
        if (index > maxindex)
                return NULL;
 
        parent = NULL;
 
        while (radix_tree_is_internal_node(node)) {
-               shift -= RADIX_TREE_MAP_SHIFT;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-
                parent = entry_to_node(node);
-               offset = radix_tree_descend(parent, &node, offset);
+               offset = radix_tree_descend(parent, &node, index);
        }
 
-       if (node == NULL)
-               goto out;
-
-       index >>= shift;
-
-       while (parent) {
-               if (!tag_get(parent, tag, offset))
-                       goto out;
-               tag_clear(parent, tag, offset);
-               if (any_tag_set(parent, tag))
-                       goto out;
-
-               index >>= RADIX_TREE_MAP_SHIFT;
-               offset = index & RADIX_TREE_MAP_MASK;
-               parent = parent->parent;
-       }
-
-       /* clear the root's tag bit */
-       if (root_tag_get(root, tag))
-               root_tag_clear(root, tag);
+       if (node)
+               node_tag_clear(root, parent, tag, offset);
 
-out:
        return node;
 }
 EXPORT_SYMBOL(radix_tree_tag_clear);
@@ -824,25 +812,21 @@ int radix_tree_tag_get(struct radix_tree_root *root,
 {
        struct radix_tree_node *node, *parent;
        unsigned long maxindex;
-       unsigned int shift;
 
        if (!root_tag_get(root, tag))
                return 0;
 
-       shift = radix_tree_load_root(root, &node, &maxindex);
+       radix_tree_load_root(root, &node, &maxindex);
        if (index > maxindex)
                return 0;
        if (node == NULL)
                return 0;
 
        while (radix_tree_is_internal_node(node)) {
-               int offset;
-
-               shift -= RADIX_TREE_MAP_SHIFT;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
+               unsigned offset;
 
                parent = entry_to_node(node);
-               offset = radix_tree_descend(parent, &node, offset);
+               offset = radix_tree_descend(parent, &node, index);
 
                if (!node)
                        return 0;
@@ -875,7 +859,7 @@ static inline void __set_iter_shift(struct radix_tree_iter *iter,
 void **radix_tree_next_chunk(struct radix_tree_root *root,
                             struct radix_tree_iter *iter, unsigned flags)
 {
-       unsigned shift, tag = flags & RADIX_TREE_ITER_TAG_MASK;
+       unsigned tag = flags & RADIX_TREE_ITER_TAG_MASK;
        struct radix_tree_node *node, *child;
        unsigned long index, offset, maxindex;
 
@@ -896,7 +880,7 @@ void **radix_tree_next_chunk(struct radix_tree_root *root,
                return NULL;
 
  restart:
-       shift = radix_tree_load_root(root, &child, &maxindex);
+       radix_tree_load_root(root, &child, &maxindex);
        if (index > maxindex)
                return NULL;
        if (!child)
@@ -913,9 +897,7 @@ void **radix_tree_next_chunk(struct radix_tree_root *root,
 
        do {
                node = entry_to_node(child);
-               shift -= RADIX_TREE_MAP_SHIFT;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-               offset = radix_tree_descend(node, &child, offset);
+               offset = radix_tree_descend(node, &child, index);
 
                if ((flags & RADIX_TREE_ITER_TAGGED) ?
                                !tag_get(node, tag, offset) : !child) {
@@ -937,7 +919,7 @@ void **radix_tree_next_chunk(struct radix_tree_root *root,
                                                break;
                                }
                        index &= ~node_maxindex(node);
-                       index += offset << shift;
+                       index += offset << node->shift;
                        /* Overflow after ~0UL */
                        if (!index)
                                return NULL;
@@ -953,7 +935,7 @@ void **radix_tree_next_chunk(struct radix_tree_root *root,
        /* Update the iterator state */
        iter->index = (index &~ node_maxindex(node)) | (offset << node->shift);
        iter->next_index = (index | node_maxindex(node)) + 1;
-       __set_iter_shift(iter, shift);
+       __set_iter_shift(iter, node->shift);
 
        /* Construct iter->tags bit-mask from node->tags[tag] array */
        if (flags & RADIX_TREE_ITER_TAGGED) {
@@ -1011,10 +993,10 @@ unsigned long radix_tree_range_tag_if_tagged(struct radix_tree_root *root,
 {
        struct radix_tree_node *parent, *node, *child;
        unsigned long maxindex;
-       unsigned int shift = radix_tree_load_root(root, &child, &maxindex);
        unsigned long tagged = 0;
        unsigned long index = *first_indexp;
 
+       radix_tree_load_root(root, &child, &maxindex);
        last_index = min(last_index, maxindex);
        if (index > last_index)
                return 0;
@@ -1031,11 +1013,9 @@ unsigned long radix_tree_range_tag_if_tagged(struct radix_tree_root *root,
        }
 
        node = entry_to_node(child);
-       shift -= RADIX_TREE_MAP_SHIFT;
 
        for (;;) {
-               unsigned offset = (index >> shift) & RADIX_TREE_MAP_MASK;
-               offset = radix_tree_descend(node, &child, offset);
+               unsigned offset = radix_tree_descend(node, &child, index);
                if (!child)
                        goto next;
                if (!tag_get(node, iftag, offset))
@@ -1043,7 +1023,6 @@ unsigned long radix_tree_range_tag_if_tagged(struct radix_tree_root *root,
                /* Sibling slots never have tags set on them */
                if (radix_tree_is_internal_node(child)) {
                        node = entry_to_node(child);
-                       shift -= RADIX_TREE_MAP_SHIFT;
                        continue;
                }
 
@@ -1064,12 +1043,12 @@ unsigned long radix_tree_range_tag_if_tagged(struct radix_tree_root *root,
                        tag_set(parent, settag, offset);
                }
  next:
-               /* Go to next item at level determined by 'shift' */
-               index = ((index >> shift) + 1) << shift;
+               /* Go to next entry in node */
+               index = ((index >> node->shift) + 1) << node->shift;
                /* Overflow can happen when last_index is ~0UL... */
                if (index > last_index || !index)
                        break;
-               offset = (index >> shift) & RADIX_TREE_MAP_MASK;
+               offset = (index >> node->shift) & RADIX_TREE_MAP_MASK;
                while (offset == 0) {
                        /*
                         * We've fully scanned this node. Go up. Because
@@ -1077,8 +1056,7 @@ unsigned long radix_tree_range_tag_if_tagged(struct radix_tree_root *root,
                         * we do below cannot wander astray.
                         */
                        node = node->parent;
-                       shift += RADIX_TREE_MAP_SHIFT;
-                       offset = (index >> shift) & RADIX_TREE_MAP_MASK;
+                       offset = (index >> node->shift) & RADIX_TREE_MAP_MASK;
                }
                if (is_sibling_entry(node, node->slots[offset]))
                        goto next;
@@ -1276,13 +1254,10 @@ struct locate_info {
 static unsigned long __locate(struct radix_tree_node *slot, void *item,
                              unsigned long index, struct locate_info *info)
 {
-       unsigned int shift;
        unsigned long i;
 
-       shift = slot->shift + RADIX_TREE_MAP_SHIFT;
-
        do {
-               shift -= RADIX_TREE_MAP_SHIFT;
+               unsigned int shift = slot->shift;
 
                for (i = (index >> shift) & RADIX_TREE_MAP_MASK;
                     i < RADIX_TREE_MAP_SIZE;
@@ -1305,9 +1280,7 @@ static unsigned long __locate(struct radix_tree_node *slot, void *item,
                        slot = node;
                        break;
                }
-               if (i == RADIX_TREE_MAP_SIZE)
-                       break;
-       } while (shift);
+       } while (i < RADIX_TREE_MAP_SIZE);
 
 out:
        if ((index == 0) && (i == RADIX_TREE_MAP_SIZE))
@@ -1527,14 +1500,9 @@ void *radix_tree_delete_item(struct radix_tree_root *root,
 
        offset = get_slot_offset(node, slot);
 
-       /*
-        * Clear all tags associated with the item to be deleted.
-        * This way of doing it would be inefficient, but seldom is any set.
-        */
-       for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) {
-               if (tag_get(node, tag, offset))
-                       radix_tree_tag_clear(root, index, tag);
-       }
+       /* Clear all tags associated with the item to be deleted.  */
+       for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++)
+               node_tag_clear(root, node, tag, offset);
 
        delete_sibling_entries(node, node_to_entry(slot), offset);
        node->slots[offset] = NULL;
@@ -1561,6 +1529,28 @@ void *radix_tree_delete(struct radix_tree_root *root, unsigned long index)
 }
 EXPORT_SYMBOL(radix_tree_delete);
 
+struct radix_tree_node *radix_tree_replace_clear_tags(
+                       struct radix_tree_root *root,
+                       unsigned long index, void *entry)
+{
+       struct radix_tree_node *node;
+       void **slot;
+
+       __radix_tree_lookup(root, index, &node, &slot);
+
+       if (node) {
+               unsigned int tag, offset = get_slot_offset(node, slot);
+               for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++)
+                       node_tag_clear(root, node, tag, offset);
+       } else {
+               /* Clear root node tags */
+               root->gfp_mask &= __GFP_BITS_MASK;
+       }
+
+       radix_tree_replace_slot(slot, entry);
+       return node;
+}
+
 /**
  *     radix_tree_tagged - test whether any items in the tree are tagged
  *     @root:          radix tree root