Commit df5a00f8 authored by Mitko Haralanov's avatar Mitko Haralanov Committed by Doug Ledford

IB/hfi1: Use interval RB trees

The interval RB trees can handle RB nodes which
hold ranged information. This is exactly the usage
for the buffer cache implemented in the expected
receive code path.

Convert the MMU/RB functions to use the interval RB
tree API. This will help with future users of the
caching API, as well.
Reviewed-by: default avatarDennis Dalessandro <dennis.dalessandro@intel.com>
Reviewed-by: default avatarDean Luick <dean.luick@intel.com>
Signed-off-by: default avatarMitko Haralanov <mitko.haralanov@intel.com>
Signed-off-by: default avatarJubin John <jubin.john@intel.com>
Signed-off-by: default avatarDoug Ledford <dledford@redhat.com>
parent 909e2cd0
......@@ -46,7 +46,7 @@
*/
#include <linux/list.h>
#include <linux/mmu_notifier.h>
#include <linux/rbtree.h>
#include <linux/interval_tree_generic.h>
#include "mmu_rb.h"
#include "trace.h"
......@@ -62,6 +62,8 @@ struct mmu_rb_handler {
static LIST_HEAD(mmu_rb_handlers);
static DEFINE_SPINLOCK(mmu_rb_lock); /* protect mmu_rb_handlers list */
static unsigned long mmu_node_start(struct mmu_rb_node *);
static unsigned long mmu_node_last(struct mmu_rb_node *);
static struct mmu_rb_handler *find_mmu_handler(struct rb_root *);
static inline void mmu_notifier_page(struct mmu_notifier *, struct mm_struct *,
unsigned long);
......@@ -78,6 +80,19 @@ static struct mmu_notifier_ops mn_opts = {
.invalidate_range_start = mmu_notifier_range_start,
};
INTERVAL_TREE_DEFINE(struct mmu_rb_node, node, unsigned long, __last,
mmu_node_start, mmu_node_last, static, __mmu_int_rb);
static unsigned long mmu_node_start(struct mmu_rb_node *node)
{
return node->addr & PAGE_MASK;
}
static unsigned long mmu_node_last(struct mmu_rb_node *node)
{
return ((node->addr & PAGE_MASK) + node->len);
}
int hfi1_mmu_rb_register(struct rb_root *root, struct mmu_rb_ops *ops)
{
struct mmu_rb_handler *handlr;
......@@ -133,40 +148,27 @@ void hfi1_mmu_rb_unregister(struct rb_root *root)
int hfi1_mmu_rb_insert(struct rb_root *root, struct mmu_rb_node *mnode)
{
struct rb_node **new, *parent = NULL;
struct mmu_rb_handler *handler = find_mmu_handler(root);
struct mmu_rb_node *this;
struct mmu_rb_node *node;
unsigned long flags;
int res, ret = 0;
int ret = 0;
if (!handler)
return -EINVAL;
new = &handler->root->rb_node;
spin_lock_irqsave(&handler->lock, flags);
while (*new) {
this = container_of(*new, struct mmu_rb_node, node);
res = handler->ops->compare(this, mnode->addr, mnode->len);
parent = *new;
if (res < 0) {
new = &((*new)->rb_left);
} else if (res > 0) {
new = &((*new)->rb_right);
} else {
ret = 1;
goto unlock;
}
node = __mmu_rb_search(handler, mnode->addr, mnode->len);
if (node) {
ret = -EINVAL;
goto unlock;
}
__mmu_int_rb_insert(mnode, root);
if (handler->ops->insert) {
ret = handler->ops->insert(root, mnode);
if (ret)
goto unlock;
__mmu_int_rb_remove(mnode, root);
}
rb_link_node(&mnode->node, parent, new);
rb_insert_color(&mnode->node, root);
unlock:
spin_unlock_irqrestore(&handler->lock, flags);
return ret;
......@@ -177,29 +179,17 @@ static struct mmu_rb_node *__mmu_rb_search(struct mmu_rb_handler *handler,
unsigned long addr,
unsigned long len)
{
struct rb_node *node = handler->root->rb_node;
struct mmu_rb_node *mnode;
int res;
while (node) {
mnode = container_of(node, struct mmu_rb_node, node);
res = handler->ops->compare(mnode, addr, len);
if (res < 0)
node = node->rb_left;
else if (res > 0)
node = node->rb_right;
else
return mnode;
}
return NULL;
struct mmu_rb_node *node;
node = __mmu_int_rb_iter_first(handler->root, addr, len);
return node;
}
static void __mmu_rb_remove(struct mmu_rb_handler *handler,
struct mmu_rb_node *node, bool arg)
{
/* Validity of handler and node pointers has been checked by caller. */
rb_erase(&node->node, handler->root);
__mmu_int_rb_remove(node, handler->root);
if (handler->ops->remove)
handler->ops->remove(handler->root, node, arg);
}
......@@ -271,45 +261,13 @@ static void mmu_notifier_mem_invalidate(struct mmu_notifier *mn,
container_of(mn, struct mmu_rb_handler, mn);
struct rb_root *root = handler->root;
struct mmu_rb_node *node;
unsigned long addr = start, naddr, nlen, flags;
unsigned long flags;
spin_lock_irqsave(&handler->lock, flags);
while (addr < end) {
/*
* There is no good way to provide a reasonable length to the
* search function at this point. Using the remaining length in
* the invalidation range is not the right thing to do.
* We have to rely on the fact that the insertion algorithm
* takes care of any overlap or length restrictions by using the
* actual size of each node. Therefore, we can use a page as an
* arbitrary, non-zero value.
*/
node = __mmu_rb_search(handler, addr, PAGE_SIZE);
if (!node) {
/*
* Didn't find a node at this address. However, the
* range could be bigger than what we have registered
* so we have to keep looking.
*/
addr += PAGE_SIZE;
continue;
}
naddr = node->addr;
nlen = node->len;
for (node = __mmu_int_rb_iter_first(root, start, end); node;
node = __mmu_int_rb_iter_next(node, start, end)) {
if (handler->ops->invalidate(root, node))
__mmu_rb_remove(handler, node, true);
/*
* The next address to be looked up is computed based
* on the node's starting address. This is due to the
* fact that the range where we start might be in the
* middle of the node's buffer so simply incrementing
* the address by the node's size would result is a
* bad address.
*/
addr = naddr + nlen;
}
spin_unlock_irqrestore(&handler->lock, flags);
}
......@@ -50,9 +50,10 @@
#include "hfi.h"
struct mmu_rb_node {
struct rb_node node;
unsigned long addr;
unsigned long len;
unsigned long __last;
struct rb_node node;
};
struct mmu_rb_ops {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment