RDMA/counter: Add set/clear per-port auto mode support
authorMark Zhang <markz@mellanox.com>
Tue, 2 Jul 2019 10:02:34 +0000 (13:02 +0300)
committerJason Gunthorpe <jgg@mellanox.com>
Fri, 5 Jul 2019 13:22:54 +0000 (10:22 -0300)
Add an API to support set/clear per-port auto mode.

Signed-off-by: Mark Zhang <markz@mellanox.com>
Reviewed-by: Majd Dibbiny <majd@mellanox.com>
Signed-off-by: Leon Romanovsky <leonro@mellanox.com>
Signed-off-by: Jason Gunthorpe <jgg@mellanox.com>
drivers/infiniband/core/Makefile
drivers/infiniband/core/counters.c [new file with mode: 0644]
drivers/infiniband/core/device.c
include/rdma/ib_verbs.h
include/rdma/rdma_counter.h
include/uapi/rdma/rdma_netlink.h

index 42f1b2a4f7466232b237b085e5e15b83d796f0c3..09881bd5f12dd030df685002d69c4731bf18abb7 100644 (file)
@@ -11,7 +11,7 @@ ib_core-y :=                  packer.o ud_header.o verbs.o cq.o rw.o sysfs.o \
                                device.o fmr_pool.o cache.o netlink.o \
                                roce_gid_mgmt.o mr_pool.o addr.o sa_query.o \
                                multicast.o mad.o smi.o agent.o mad_rmpp.o \
-                               nldev.o restrack.o
+                               nldev.o restrack.o counters.o
 
 ib_core-$(CONFIG_SECURITY_INFINIBAND) += security.o
 ib_core-$(CONFIG_CGROUP_RDMA) += cgroup.o
diff --git a/drivers/infiniband/core/counters.c b/drivers/infiniband/core/counters.c
new file mode 100644 (file)
index 0000000..6167914
--- /dev/null
@@ -0,0 +1,74 @@
+// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
+/*
+ * Copyright (c) 2019 Mellanox Technologies. All rights reserved.
+ */
+#include <rdma/ib_verbs.h>
+#include <rdma/rdma_counter.h>
+
+#include "core_priv.h"
+#include "restrack.h"
+
+#define ALL_AUTO_MODE_MASKS (RDMA_COUNTER_MASK_QP_TYPE)
+
+static int __counter_set_mode(struct rdma_counter_mode *curr,
+                             enum rdma_nl_counter_mode new_mode,
+                             enum rdma_nl_counter_mask new_mask)
+{
+       if ((new_mode == RDMA_COUNTER_MODE_AUTO) &&
+           ((new_mask & (~ALL_AUTO_MODE_MASKS)) ||
+            (curr->mode != RDMA_COUNTER_MODE_NONE)))
+               return -EINVAL;
+
+       curr->mode = new_mode;
+       curr->mask = new_mask;
+       return 0;
+}
+
+/**
+ * rdma_counter_set_auto_mode() - Turn on/off per-port auto mode
+ *
+ * When @on is true, the @mask must be set
+ */
+int rdma_counter_set_auto_mode(struct ib_device *dev, u8 port,
+                              bool on, enum rdma_nl_counter_mask mask)
+{
+       struct rdma_port_counter *port_counter;
+       int ret;
+
+       port_counter = &dev->port_data[port].port_counter;
+       mutex_lock(&port_counter->lock);
+       if (on) {
+               ret = __counter_set_mode(&port_counter->mode,
+                                        RDMA_COUNTER_MODE_AUTO, mask);
+       } else {
+               if (port_counter->mode.mode != RDMA_COUNTER_MODE_AUTO) {
+                       ret = -EINVAL;
+                       goto out;
+               }
+               ret = __counter_set_mode(&port_counter->mode,
+                                        RDMA_COUNTER_MODE_NONE, 0);
+       }
+
+out:
+       mutex_unlock(&port_counter->lock);
+       return ret;
+}
+
+void rdma_counter_init(struct ib_device *dev)
+{
+       struct rdma_port_counter *port_counter;
+       u32 port;
+
+       if (!dev->ops.alloc_hw_stats || !dev->port_data)
+               return;
+
+       rdma_for_each_port(dev, port) {
+               port_counter = &dev->port_data[port].port_counter;
+               port_counter->mode.mode = RDMA_COUNTER_MODE_NONE;
+               mutex_init(&port_counter->lock);
+       }
+}
+
+void rdma_counter_release(struct ib_device *dev)
+{
+}
index 8a6ccb936dfeb4923f34ef8c75681582961e7219..6579865e48666da273086b3385388b1fb830cd84 100644 (file)
@@ -46,6 +46,7 @@
 #include <rdma/rdma_netlink.h>
 #include <rdma/ib_addr.h>
 #include <rdma/ib_cache.h>
+#include <rdma/rdma_counter.h>
 
 #include "core_priv.h"
 #include "restrack.h"
@@ -492,10 +493,12 @@ static void ib_device_release(struct device *device)
        if (dev->port_data) {
                ib_cache_release_one(dev);
                ib_security_release_port_pkey_list(dev);
+               rdma_counter_release(dev);
                kfree_rcu(container_of(dev->port_data, struct ib_port_data_rcu,
                                       pdata[0]),
                          rcu_head);
        }
+
        xa_destroy(&dev->compat_devs);
        xa_destroy(&dev->client_data);
        kfree_rcu(dev, rcu_head);
@@ -1316,6 +1319,8 @@ int ib_register_device(struct ib_device *device, const char *name)
 
        ib_device_register_rdmacg(device);
 
+       rdma_counter_init(device);
+
        /*
         * Ensure that ADD uevent is not fired because it
         * is too early amd device is not initialized yet.
index 26e9c2594913ea473869fd3975ef089609c38270..3d19c056fbc056f30eb5b9392a7b360f7c727e0e 100644 (file)
@@ -62,6 +62,7 @@
 #include <linux/irqflags.h>
 #include <linux/preempt.h>
 #include <uapi/rdma/ib_user_verbs.h>
+#include <rdma/rdma_counter.h>
 #include <rdma/restrack.h>
 #include <rdma/signature.h>
 #include <uapi/rdma/rdma_user_ioctl.h>
@@ -2119,6 +2120,7 @@ struct ib_port_data {
        spinlock_t netdev_lock;
        struct net_device __rcu *netdev;
        struct hlist_node ndev_hash_link;
+       struct rdma_port_counter port_counter;
 };
 
 /* rdma netdev type - specifies protocol type */
index 283ac1a0cdb78350569ea813cf5bae2e06e690ae..8dd2619c015df72226049f4ad2c00d04ff797427 100644 (file)
@@ -6,8 +6,26 @@
 #ifndef _RDMA_COUNTER_H_
 #define _RDMA_COUNTER_H_
 
+#include <linux/mutex.h>
+
 #include <rdma/ib_verbs.h>
 #include <rdma/restrack.h>
+#include <rdma/rdma_netlink.h>
+
+struct auto_mode_param {
+       int qp_type;
+};
+
+struct rdma_counter_mode {
+       enum rdma_nl_counter_mode mode;
+       enum rdma_nl_counter_mask mask;
+       struct auto_mode_param param;
+};
+
+struct rdma_port_counter {
+       struct rdma_counter_mode mode;
+       struct mutex lock;
+};
 
 struct rdma_counter {
        struct rdma_restrack_entry      res;
@@ -15,4 +33,10 @@ struct rdma_counter {
        uint32_t                        id;
        u8                              port;
 };
+
+void rdma_counter_init(struct ib_device *dev);
+void rdma_counter_release(struct ib_device *dev);
+int rdma_counter_set_auto_mode(struct ib_device *dev, u8 port,
+                              bool on, enum rdma_nl_counter_mask mask);
+
 #endif /* _RDMA_COUNTER_H_ */
index 650cee8c4bf183aaa5653530116355e6dd5961e5..e3cd912e9cef59c8315765b6d4bd43944ae0e19f 100644 (file)
@@ -507,4 +507,30 @@ enum rdma_nldev_attr {
         */
        RDMA_NLDEV_ATTR_MAX
 };
+
+/*
+ * Supported counter bind modes. All modes are mutual-exclusive.
+ */
+enum rdma_nl_counter_mode {
+       RDMA_COUNTER_MODE_NONE,
+
+       /*
+        * A qp is bound with a counter automatically during initialization
+        * based on the auto mode (e.g., qp type, ...)
+        */
+       RDMA_COUNTER_MODE_AUTO,
+
+       /*
+        * Always the end
+        */
+       RDMA_COUNTER_MODE_MAX,
+};
+
+/*
+ * Supported criteria in counter auto mode.
+ * Currently only "qp type" is supported
+ */
+enum rdma_nl_counter_mask {
+       RDMA_COUNTER_MASK_QP_TYPE = 1,
+};
 #endif /* _UAPI_RDMA_NETLINK_H */