2018-06-05 11:48:08 +07:00
|
|
|
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
|
2018-01-28 16:17:20 +07:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017-2018 Mellanox Technologies. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
2018-03-02 04:57:44 +07:00
|
|
|
#include <rdma/rdma_cm.h>
|
2018-01-28 16:17:20 +07:00
|
|
|
#include <rdma/ib_verbs.h>
|
|
|
|
#include <rdma/restrack.h>
|
|
|
|
#include <linux/mutex.h>
|
|
|
|
#include <linux/sched/task.h>
|
|
|
|
#include <linux/pid_namespace.h>
|
|
|
|
|
2018-03-02 04:57:44 +07:00
|
|
|
#include "cma_priv.h"
|
|
|
|
|
RDMA/nldev: add driver-specific resource tracking
Each driver can register a "fill entry" function with the restrack core.
This function will be called when filling out a resource, allowing the
driver to add driver-specific details. The details consist of a
nltable of nested attributes, that are in the form of <key, [print-type],
value> tuples. Both key and value attributes are mandatory. The key
nlattr must be a string, and the value nlattr can be one of the driver
attributes that are generic, but typed, allowing the attributes to be
validated. Currently the driver nlattr types include string, s32,
u32, s64, and u64. The print-type nlattr allows a driver to specify
an alternative display format for user tools displaying the attribute.
For example, a u32 attribute will default to "%u", but a print-type
attribute can be included for it to be displayed in hex. This allows
the user tool to print the number in the format desired by the driver
driver.
More attrs can be defined as they become needed by drivers.
Signed-off-by: Steve Wise <swise@opengridcomputing.com>
Reviewed-by: Leon Romanovsky <leonro@mellanox.com>
Signed-off-by: Doug Ledford <dledford@redhat.com>
2018-05-03 22:41:30 +07:00
|
|
|
static int fill_res_noop(struct sk_buff *msg,
|
|
|
|
struct rdma_restrack_entry *entry)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-30 17:48:58 +07:00
|
|
|
/**
|
|
|
|
* rdma_restrack_init() - initialize resource tracking
|
|
|
|
* @dev: IB device
|
|
|
|
*/
|
|
|
|
void rdma_restrack_init(struct ib_device *dev)
|
2018-01-28 16:17:20 +07:00
|
|
|
{
|
2019-01-30 17:48:58 +07:00
|
|
|
struct rdma_restrack_root *res = &dev->res;
|
|
|
|
|
2018-01-28 16:17:20 +07:00
|
|
|
init_rwsem(&res->rwsem);
|
RDMA/nldev: add driver-specific resource tracking
Each driver can register a "fill entry" function with the restrack core.
This function will be called when filling out a resource, allowing the
driver to add driver-specific details. The details consist of a
nltable of nested attributes, that are in the form of <key, [print-type],
value> tuples. Both key and value attributes are mandatory. The key
nlattr must be a string, and the value nlattr can be one of the driver
attributes that are generic, but typed, allowing the attributes to be
validated. Currently the driver nlattr types include string, s32,
u32, s64, and u64. The print-type nlattr allows a driver to specify
an alternative display format for user tools displaying the attribute.
For example, a u32 attribute will default to "%u", but a print-type
attribute can be included for it to be displayed in hex. This allows
the user tool to print the number in the format desired by the driver
driver.
More attrs can be defined as they become needed by drivers.
Signed-off-by: Steve Wise <swise@opengridcomputing.com>
Reviewed-by: Leon Romanovsky <leonro@mellanox.com>
Signed-off-by: Doug Ledford <dledford@redhat.com>
2018-05-03 22:41:30 +07:00
|
|
|
res->fill_res_entry = fill_res_noop;
|
2018-01-28 16:17:20 +07:00
|
|
|
}
|
|
|
|
|
2018-03-21 22:12:42 +07:00
|
|
|
static const char *type2str(enum rdma_restrack_type type)
|
|
|
|
{
|
|
|
|
static const char * const names[RDMA_RESTRACK_MAX] = {
|
|
|
|
[RDMA_RESTRACK_PD] = "PD",
|
|
|
|
[RDMA_RESTRACK_CQ] = "CQ",
|
|
|
|
[RDMA_RESTRACK_QP] = "QP",
|
|
|
|
[RDMA_RESTRACK_CM_ID] = "CM_ID",
|
|
|
|
[RDMA_RESTRACK_MR] = "MR",
|
2018-11-28 18:16:43 +07:00
|
|
|
[RDMA_RESTRACK_CTX] = "CTX",
|
2018-03-21 22:12:42 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
return names[type];
|
|
|
|
};
|
|
|
|
|
2019-01-30 17:48:58 +07:00
|
|
|
/**
|
|
|
|
* rdma_restrack_clean() - clean resource tracking
|
|
|
|
* @dev: IB device
|
|
|
|
*/
|
|
|
|
void rdma_restrack_clean(struct ib_device *dev)
|
2018-01-28 16:17:20 +07:00
|
|
|
{
|
2019-01-30 17:48:58 +07:00
|
|
|
struct rdma_restrack_root *res = &dev->res;
|
2018-03-21 22:12:42 +07:00
|
|
|
struct rdma_restrack_entry *e;
|
|
|
|
char buf[TASK_COMM_LEN];
|
|
|
|
const char *owner;
|
|
|
|
int bkt;
|
|
|
|
|
|
|
|
if (hash_empty(res->hash))
|
|
|
|
return;
|
|
|
|
|
|
|
|
dev = container_of(res, struct ib_device, res);
|
|
|
|
pr_err("restrack: %s", CUT_HERE);
|
2018-09-21 05:42:23 +07:00
|
|
|
dev_err(&dev->dev, "BUG: RESTRACK detected leak of resources\n");
|
2018-03-21 22:12:42 +07:00
|
|
|
hash_for_each(res->hash, bkt, e, node) {
|
|
|
|
if (rdma_is_kernel_res(e)) {
|
|
|
|
owner = e->kern_name;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* There is no need to call get_task_struct here,
|
|
|
|
* because we can be here only if there are more
|
|
|
|
* get_task_struct() call than put_task_struct().
|
|
|
|
*/
|
|
|
|
get_task_comm(buf, e->task);
|
|
|
|
owner = buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_err("restrack: %s %s object allocated by %s is not freed\n",
|
|
|
|
rdma_is_kernel_res(e) ? "Kernel" : "User",
|
|
|
|
type2str(e->type), owner);
|
|
|
|
}
|
|
|
|
pr_err("restrack: %s", CUT_HERE);
|
2018-01-28 16:17:20 +07:00
|
|
|
}
|
|
|
|
|
2019-01-30 17:48:58 +07:00
|
|
|
/**
|
|
|
|
* rdma_restrack_count() - the current usage of specific object
|
|
|
|
* @dev: IB device
|
|
|
|
* @type: actual type of object to operate
|
|
|
|
* @ns: PID namespace
|
|
|
|
*/
|
|
|
|
int rdma_restrack_count(struct ib_device *dev, enum rdma_restrack_type type,
|
2018-01-28 16:17:20 +07:00
|
|
|
struct pid_namespace *ns)
|
|
|
|
{
|
2019-01-30 17:48:58 +07:00
|
|
|
struct rdma_restrack_root *res = &dev->res;
|
2018-01-28 16:17:20 +07:00
|
|
|
struct rdma_restrack_entry *e;
|
|
|
|
u32 cnt = 0;
|
|
|
|
|
|
|
|
down_read(&res->rwsem);
|
|
|
|
hash_for_each_possible(res->hash, e, node, type) {
|
|
|
|
if (ns == &init_pid_ns ||
|
|
|
|
(!rdma_is_kernel_res(e) &&
|
|
|
|
ns == task_active_pid_ns(e->task)))
|
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
up_read(&res->rwsem);
|
|
|
|
return cnt;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_count);
|
|
|
|
|
|
|
|
static void set_kern_name(struct rdma_restrack_entry *res)
|
|
|
|
{
|
2018-03-02 04:58:13 +07:00
|
|
|
struct ib_pd *pd;
|
2018-01-28 16:17:20 +07:00
|
|
|
|
2018-03-02 04:58:13 +07:00
|
|
|
switch (res->type) {
|
|
|
|
case RDMA_RESTRACK_QP:
|
|
|
|
pd = container_of(res, struct ib_qp, res)->pd;
|
|
|
|
if (!pd) {
|
|
|
|
WARN_ONCE(true, "XRC QPs are not supported\n");
|
|
|
|
/* Survive, despite the programmer's error */
|
|
|
|
res->kern_name = " ";
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case RDMA_RESTRACK_MR:
|
|
|
|
pd = container_of(res, struct ib_mr, res)->pd;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Other types set kern_name directly */
|
|
|
|
pd = NULL;
|
|
|
|
break;
|
2018-01-28 16:17:20 +07:00
|
|
|
}
|
|
|
|
|
2018-03-02 04:58:13 +07:00
|
|
|
if (pd)
|
|
|
|
res->kern_name = pd->res.kern_name;
|
2018-01-28 16:17:20 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct ib_device *res_to_dev(struct rdma_restrack_entry *res)
|
|
|
|
{
|
2018-03-02 04:57:22 +07:00
|
|
|
switch (res->type) {
|
2018-01-28 16:17:20 +07:00
|
|
|
case RDMA_RESTRACK_PD:
|
2018-03-02 04:57:22 +07:00
|
|
|
return container_of(res, struct ib_pd, res)->device;
|
2018-01-28 16:17:20 +07:00
|
|
|
case RDMA_RESTRACK_CQ:
|
2018-03-02 04:57:22 +07:00
|
|
|
return container_of(res, struct ib_cq, res)->device;
|
2018-01-28 16:17:20 +07:00
|
|
|
case RDMA_RESTRACK_QP:
|
2018-03-02 04:57:22 +07:00
|
|
|
return container_of(res, struct ib_qp, res)->device;
|
2018-03-02 04:57:44 +07:00
|
|
|
case RDMA_RESTRACK_CM_ID:
|
|
|
|
return container_of(res, struct rdma_id_private,
|
|
|
|
res)->id.device;
|
2018-03-02 04:58:13 +07:00
|
|
|
case RDMA_RESTRACK_MR:
|
|
|
|
return container_of(res, struct ib_mr, res)->device;
|
2018-11-28 18:16:43 +07:00
|
|
|
case RDMA_RESTRACK_CTX:
|
|
|
|
return container_of(res, struct ib_ucontext, res)->device;
|
2018-01-28 16:17:20 +07:00
|
|
|
default:
|
2018-03-02 04:57:22 +07:00
|
|
|
WARN_ONCE(true, "Wrong resource tracking type %u\n", res->type);
|
2018-01-28 16:17:20 +07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-02 15:48:01 +07:00
|
|
|
void rdma_restrack_set_task(struct rdma_restrack_entry *res,
|
2018-10-02 15:48:02 +07:00
|
|
|
const char *caller)
|
2018-10-02 15:48:01 +07:00
|
|
|
{
|
2018-10-02 15:48:02 +07:00
|
|
|
if (caller) {
|
|
|
|
res->kern_name = caller;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-10-02 15:48:01 +07:00
|
|
|
if (res->task)
|
|
|
|
put_task_struct(res->task);
|
2018-10-02 15:48:02 +07:00
|
|
|
get_task_struct(current);
|
|
|
|
res->task = current;
|
2018-10-02 15:48:01 +07:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_set_task);
|
|
|
|
|
2018-12-17 22:15:16 +07:00
|
|
|
static void rdma_restrack_add(struct rdma_restrack_entry *res)
|
2018-01-28 16:17:20 +07:00
|
|
|
{
|
|
|
|
struct ib_device *dev = res_to_dev(res);
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
return;
|
|
|
|
|
2018-12-17 22:15:16 +07:00
|
|
|
if (res->type != RDMA_RESTRACK_CM_ID || rdma_is_kernel_res(res))
|
2018-03-15 16:10:42 +07:00
|
|
|
res->task = NULL;
|
|
|
|
|
2018-12-17 22:15:16 +07:00
|
|
|
if (!rdma_is_kernel_res(res)) {
|
2018-03-02 04:57:44 +07:00
|
|
|
if (!res->task)
|
2018-10-02 15:48:02 +07:00
|
|
|
rdma_restrack_set_task(res, NULL);
|
2018-01-28 16:17:20 +07:00
|
|
|
res->kern_name = NULL;
|
|
|
|
} else {
|
|
|
|
set_kern_name(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
kref_init(&res->kref);
|
|
|
|
init_completion(&res->comp);
|
|
|
|
res->valid = true;
|
|
|
|
|
|
|
|
down_write(&dev->res.rwsem);
|
|
|
|
hash_add(dev->res.hash, &res->node, res->type);
|
|
|
|
up_write(&dev->res.rwsem);
|
|
|
|
}
|
2018-12-17 22:15:16 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* rdma_restrack_kadd() - add kernel object to the reource tracking database
|
|
|
|
* @res: resource entry
|
|
|
|
*/
|
|
|
|
void rdma_restrack_kadd(struct rdma_restrack_entry *res)
|
|
|
|
{
|
|
|
|
res->user = false;
|
|
|
|
rdma_restrack_add(res);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_kadd);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* rdma_restrack_uadd() - add user object to the reource tracking database
|
|
|
|
* @res: resource entry
|
|
|
|
*/
|
|
|
|
void rdma_restrack_uadd(struct rdma_restrack_entry *res)
|
|
|
|
{
|
|
|
|
res->user = true;
|
|
|
|
rdma_restrack_add(res);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_uadd);
|
2018-01-28 16:17:20 +07:00
|
|
|
|
|
|
|
int __must_check rdma_restrack_get(struct rdma_restrack_entry *res)
|
|
|
|
{
|
|
|
|
return kref_get_unless_zero(&res->kref);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_get);
|
|
|
|
|
|
|
|
static void restrack_release(struct kref *kref)
|
|
|
|
{
|
|
|
|
struct rdma_restrack_entry *res;
|
|
|
|
|
|
|
|
res = container_of(kref, struct rdma_restrack_entry, kref);
|
|
|
|
complete(&res->comp);
|
|
|
|
}
|
|
|
|
|
|
|
|
int rdma_restrack_put(struct rdma_restrack_entry *res)
|
|
|
|
{
|
|
|
|
return kref_put(&res->kref, restrack_release);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_put);
|
|
|
|
|
|
|
|
void rdma_restrack_del(struct rdma_restrack_entry *res)
|
|
|
|
{
|
|
|
|
struct ib_device *dev;
|
|
|
|
|
|
|
|
if (!res->valid)
|
2018-10-02 15:48:03 +07:00
|
|
|
goto out;
|
2018-01-28 16:17:20 +07:00
|
|
|
|
|
|
|
dev = res_to_dev(res);
|
|
|
|
if (!dev)
|
|
|
|
return;
|
|
|
|
|
|
|
|
rdma_restrack_put(res);
|
|
|
|
|
|
|
|
wait_for_completion(&res->comp);
|
|
|
|
|
|
|
|
down_write(&dev->res.rwsem);
|
|
|
|
hash_del(&res->node);
|
|
|
|
res->valid = false;
|
2018-10-02 15:48:03 +07:00
|
|
|
up_write(&dev->res.rwsem);
|
|
|
|
|
|
|
|
out:
|
2018-10-12 02:10:10 +07:00
|
|
|
if (res->task) {
|
2018-01-28 16:17:20 +07:00
|
|
|
put_task_struct(res->task);
|
2018-10-12 02:10:10 +07:00
|
|
|
res->task = NULL;
|
|
|
|
}
|
2018-01-28 16:17:20 +07:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(rdma_restrack_del);
|