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: Dennis Dalessandro <dennis.dalessandro@intel.com> Reviewed-by: Dean Luick <dean.luick@intel.com> Signed-off-by: Mitko Haralanov <mitko.haralanov@intel.com> Signed-off-by: Jubin John <jubin.john@intel.com> Signed-off-by: Doug Ledford <dledford@redhat.com>
This commit is contained in:
parent
909e2cd004
commit
df5a00f81d
|
@ -46,7 +46,7 @@
|
||||||
*/
|
*/
|
||||||
#include <linux/list.h>
|
#include <linux/list.h>
|
||||||
#include <linux/mmu_notifier.h>
|
#include <linux/mmu_notifier.h>
|
||||||
#include <linux/rbtree.h>
|
#include <linux/interval_tree_generic.h>
|
||||||
|
|
||||||
#include "mmu_rb.h"
|
#include "mmu_rb.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
|
@ -62,6 +62,8 @@ struct mmu_rb_handler {
|
||||||
static LIST_HEAD(mmu_rb_handlers);
|
static LIST_HEAD(mmu_rb_handlers);
|
||||||
static DEFINE_SPINLOCK(mmu_rb_lock); /* protect mmu_rb_handlers list */
|
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 struct mmu_rb_handler *find_mmu_handler(struct rb_root *);
|
||||||
static inline void mmu_notifier_page(struct mmu_notifier *, struct mm_struct *,
|
static inline void mmu_notifier_page(struct mmu_notifier *, struct mm_struct *,
|
||||||
unsigned long);
|
unsigned long);
|
||||||
|
@ -78,6 +80,19 @@ static struct mmu_notifier_ops mn_opts = {
|
||||||
.invalidate_range_start = mmu_notifier_range_start,
|
.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)
|
int hfi1_mmu_rb_register(struct rb_root *root, struct mmu_rb_ops *ops)
|
||||||
{
|
{
|
||||||
struct mmu_rb_handler *handlr;
|
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)
|
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_handler *handler = find_mmu_handler(root);
|
||||||
struct mmu_rb_node *this;
|
struct mmu_rb_node *node;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int res, ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
if (!handler)
|
if (!handler)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
new = &handler->root->rb_node;
|
|
||||||
spin_lock_irqsave(&handler->lock, flags);
|
spin_lock_irqsave(&handler->lock, flags);
|
||||||
while (*new) {
|
node = __mmu_rb_search(handler, mnode->addr, mnode->len);
|
||||||
this = container_of(*new, struct mmu_rb_node, node);
|
if (node) {
|
||||||
res = handler->ops->compare(this, mnode->addr, mnode->len);
|
ret = -EINVAL;
|
||||||
parent = *new;
|
|
||||||
|
|
||||||
if (res < 0) {
|
|
||||||
new = &((*new)->rb_left);
|
|
||||||
} else if (res > 0) {
|
|
||||||
new = &((*new)->rb_right);
|
|
||||||
} else {
|
|
||||||
ret = 1;
|
|
||||||
goto unlock;
|
goto unlock;
|
||||||
}
|
}
|
||||||
}
|
__mmu_int_rb_insert(mnode, root);
|
||||||
|
|
||||||
if (handler->ops->insert) {
|
if (handler->ops->insert) {
|
||||||
ret = handler->ops->insert(root, mnode);
|
ret = handler->ops->insert(root, mnode);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto unlock;
|
__mmu_int_rb_remove(mnode, root);
|
||||||
}
|
}
|
||||||
|
|
||||||
rb_link_node(&mnode->node, parent, new);
|
|
||||||
rb_insert_color(&mnode->node, root);
|
|
||||||
unlock:
|
unlock:
|
||||||
spin_unlock_irqrestore(&handler->lock, flags);
|
spin_unlock_irqrestore(&handler->lock, flags);
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -177,29 +179,17 @@ static struct mmu_rb_node *__mmu_rb_search(struct mmu_rb_handler *handler,
|
||||||
unsigned long addr,
|
unsigned long addr,
|
||||||
unsigned long len)
|
unsigned long len)
|
||||||
{
|
{
|
||||||
struct rb_node *node = handler->root->rb_node;
|
struct mmu_rb_node *node;
|
||||||
struct mmu_rb_node *mnode;
|
|
||||||
int res;
|
|
||||||
|
|
||||||
while (node) {
|
node = __mmu_int_rb_iter_first(handler->root, addr, len);
|
||||||
mnode = container_of(node, struct mmu_rb_node, node);
|
return 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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __mmu_rb_remove(struct mmu_rb_handler *handler,
|
static void __mmu_rb_remove(struct mmu_rb_handler *handler,
|
||||||
struct mmu_rb_node *node, bool arg)
|
struct mmu_rb_node *node, bool arg)
|
||||||
{
|
{
|
||||||
/* Validity of handler and node pointers has been checked by caller. */
|
/* 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)
|
if (handler->ops->remove)
|
||||||
handler->ops->remove(handler->root, node, arg);
|
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);
|
container_of(mn, struct mmu_rb_handler, mn);
|
||||||
struct rb_root *root = handler->root;
|
struct rb_root *root = handler->root;
|
||||||
struct mmu_rb_node *node;
|
struct mmu_rb_node *node;
|
||||||
unsigned long addr = start, naddr, nlen, flags;
|
unsigned long flags;
|
||||||
|
|
||||||
spin_lock_irqsave(&handler->lock, flags);
|
spin_lock_irqsave(&handler->lock, flags);
|
||||||
while (addr < end) {
|
for (node = __mmu_int_rb_iter_first(root, start, end); node;
|
||||||
/*
|
node = __mmu_int_rb_iter_next(node, start, 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;
|
|
||||||
if (handler->ops->invalidate(root, node))
|
if (handler->ops->invalidate(root, node))
|
||||||
__mmu_rb_remove(handler, node, true);
|
__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);
|
spin_unlock_irqrestore(&handler->lock, flags);
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,9 +50,10 @@
|
||||||
#include "hfi.h"
|
#include "hfi.h"
|
||||||
|
|
||||||
struct mmu_rb_node {
|
struct mmu_rb_node {
|
||||||
struct rb_node node;
|
|
||||||
unsigned long addr;
|
unsigned long addr;
|
||||||
unsigned long len;
|
unsigned long len;
|
||||||
|
unsigned long __last;
|
||||||
|
struct rb_node node;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mmu_rb_ops {
|
struct mmu_rb_ops {
|
||||||
|
|
Loading…
Reference in New Issue