net: prestera: add counter HW API
authorVolodymyr Mytnyk <vmytnyk@marvell.com>
Tue, 30 Nov 2021 10:32:59 +0000 (12:32 +0200)
committerDavid S. Miller <davem@davemloft.net>
Tue, 30 Nov 2021 12:26:01 +0000 (12:26 +0000)
Add counter API for getting HW statistics.

- HW statistics gathered by this API are deleyed.
- Batch of conters is supported.
- acl stat is supported.

Co-developed-by: Serhiy Boiko <serhiy.boiko@marvell.com>
Signed-off-by: Serhiy Boiko <serhiy.boiko@marvell.com>
Signed-off-by: Volodymyr Mytnyk <vmytnyk@marvell.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/marvell/prestera/Makefile
drivers/net/ethernet/marvell/prestera/prestera.h
drivers/net/ethernet/marvell/prestera/prestera_acl.h
drivers/net/ethernet/marvell/prestera/prestera_counter.c [new file with mode: 0644]
drivers/net/ethernet/marvell/prestera/prestera_counter.h [new file with mode: 0644]
drivers/net/ethernet/marvell/prestera/prestera_hw.c
drivers/net/ethernet/marvell/prestera/prestera_hw.h
drivers/net/ethernet/marvell/prestera/prestera_main.c

index 0609df8b913db210ceb91ed18bd2770f0c027a3c..48dbcb2baf8f60710ae017db4d1cf4659c0bda1f 100644 (file)
@@ -3,6 +3,6 @@ obj-$(CONFIG_PRESTERA)  += prestera.o
 prestera-objs          := prestera_main.o prestera_hw.o prestera_dsa.o \
                           prestera_rxtx.o prestera_devlink.o prestera_ethtool.o \
                           prestera_switchdev.o prestera_acl.o prestera_flow.o \
-                          prestera_flower.o prestera_span.o
+                          prestera_flower.o prestera_span.o prestera_counter.o
 
 obj-$(CONFIG_PRESTERA_PCI)     += prestera_pci.o
index 2a4c14c704c0d91f766ac755cda7e0aa163337c3..797b2e4d355191ee6ddedf5d3cbabd72b06f9684 100644 (file)
@@ -248,6 +248,7 @@ struct prestera_switch {
        u32 mtu_max;
        u8 id;
        struct prestera_lag *lags;
+       struct prestera_counter *counter;
        u8 lag_member_max;
        u8 lag_max;
 };
index 92332eb70201d8bc8e02fbcd9c197d40800adade..8bfb7ad7307b0506e7fd8e8e5753f6ebac219ac0 100644 (file)
@@ -60,6 +60,10 @@ struct prestera_acl_match {
        __be32 mask[__PRESTERA_ACL_RULE_MATCH_TYPE_MAX];
 };
 
+struct prestera_acl_action_count {
+       u32 id;
+};
+
 struct prestera_acl_rule_entry_key {
        u32 prio;
        struct prestera_acl_match match;
@@ -67,6 +71,9 @@ struct prestera_acl_rule_entry_key {
 
 struct prestera_acl_hw_action_info {
        enum prestera_acl_rule_action id;
+       union {
+               struct prestera_acl_action_count count;
+       };
 };
 
 /* This struct (arg) used only to be passed as parameter for
diff --git a/drivers/net/ethernet/marvell/prestera/prestera_counter.c b/drivers/net/ethernet/marvell/prestera/prestera_counter.c
new file mode 100644 (file)
index 0000000..4cd53a2
--- /dev/null
@@ -0,0 +1,475 @@
+// SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
+/* Copyright (c) 2021 Marvell International Ltd. All rights reserved */
+
+#include "prestera.h"
+#include "prestera_hw.h"
+#include "prestera_acl.h"
+#include "prestera_counter.h"
+
+#define COUNTER_POLL_TIME      (msecs_to_jiffies(1000))
+#define COUNTER_RESCHED_TIME   (msecs_to_jiffies(50))
+#define COUNTER_BULK_SIZE      (256)
+
+struct prestera_counter {
+       struct prestera_switch *sw;
+       struct delayed_work stats_dw;
+       struct mutex mtx;  /* protect block_list */
+       struct prestera_counter_block **block_list;
+       u32 total_read;
+       u32 block_list_len;
+       u32 curr_idx;
+       bool is_fetching;
+};
+
+struct prestera_counter_block {
+       struct list_head list;
+       u32 id;
+       u32 offset;
+       u32 num_counters;
+       u32 client;
+       struct idr counter_idr;
+       refcount_t refcnt;
+       struct mutex mtx;  /* protect stats and counter_idr */
+       struct prestera_counter_stats *stats;
+       u8 *counter_flag;
+       bool is_updating;
+       bool full;
+};
+
+enum {
+       COUNTER_FLAG_READY = 0,
+       COUNTER_FLAG_INVALID = 1
+};
+
+static bool
+prestera_counter_is_ready(struct prestera_counter_block *block, u32 id)
+{
+       return block->counter_flag[id - block->offset] == COUNTER_FLAG_READY;
+}
+
+static void prestera_counter_lock(struct prestera_counter *counter)
+{
+       mutex_lock(&counter->mtx);
+}
+
+static void prestera_counter_unlock(struct prestera_counter *counter)
+{
+       mutex_unlock(&counter->mtx);
+}
+
+static void prestera_counter_block_lock(struct prestera_counter_block *block)
+{
+       mutex_lock(&block->mtx);
+}
+
+static void prestera_counter_block_unlock(struct prestera_counter_block *block)
+{
+       mutex_unlock(&block->mtx);
+}
+
+static bool prestera_counter_block_incref(struct prestera_counter_block *block)
+{
+       return refcount_inc_not_zero(&block->refcnt);
+}
+
+static bool prestera_counter_block_decref(struct prestera_counter_block *block)
+{
+       return refcount_dec_and_test(&block->refcnt);
+}
+
+/* must be called with prestera_counter_block_lock() */
+static void prestera_counter_stats_clear(struct prestera_counter_block *block,
+                                        u32 counter_id)
+{
+       memset(&block->stats[counter_id - block->offset], 0,
+              sizeof(*block->stats));
+}
+
+static struct prestera_counter_block *
+prestera_counter_block_lookup_not_full(struct prestera_counter *counter,
+                                      u32 client)
+{
+       u32 i;
+
+       prestera_counter_lock(counter);
+       for (i = 0; i < counter->block_list_len; i++) {
+               if (counter->block_list[i] &&
+                   counter->block_list[i]->client == client &&
+                   !counter->block_list[i]->full &&
+                   prestera_counter_block_incref(counter->block_list[i])) {
+                       prestera_counter_unlock(counter);
+                       return counter->block_list[i];
+               }
+       }
+       prestera_counter_unlock(counter);
+
+       return NULL;
+}
+
+static int prestera_counter_block_list_add(struct prestera_counter *counter,
+                                          struct prestera_counter_block *block)
+{
+       struct prestera_counter_block **arr;
+       u32 i;
+
+       prestera_counter_lock(counter);
+
+       for (i = 0; i < counter->block_list_len; i++) {
+               if (counter->block_list[i])
+                       continue;
+
+               counter->block_list[i] = block;
+               prestera_counter_unlock(counter);
+               return 0;
+       }
+
+       arr = krealloc(counter->block_list, (counter->block_list_len + 1) *
+                      sizeof(*counter->block_list), GFP_KERNEL);
+       if (!arr) {
+               prestera_counter_unlock(counter);
+               return -ENOMEM;
+       }
+
+       counter->block_list = arr;
+       counter->block_list[counter->block_list_len] = block;
+       counter->block_list_len++;
+       prestera_counter_unlock(counter);
+       return 0;
+}
+
+static struct prestera_counter_block *
+prestera_counter_block_get(struct prestera_counter *counter, u32 client)
+{
+       struct prestera_counter_block *block;
+       int err;
+
+       block = prestera_counter_block_lookup_not_full(counter, client);
+       if (block)
+               return block;
+
+       block = kzalloc(sizeof(*block), GFP_KERNEL);
+       if (!block)
+               return ERR_PTR(-ENOMEM);
+
+       err = prestera_hw_counter_block_get(counter->sw, client,
+                                           &block->id, &block->offset,
+                                           &block->num_counters);
+       if (err)
+               goto err_block;
+
+       block->stats = kcalloc(block->num_counters,
+                              sizeof(*block->stats), GFP_KERNEL);
+       if (!block->stats) {
+               err = -ENOMEM;
+               goto err_stats;
+       }
+
+       block->counter_flag = kcalloc(block->num_counters,
+                                     sizeof(*block->counter_flag),
+                                     GFP_KERNEL);
+       if (!block->counter_flag) {
+               err = -ENOMEM;
+               goto err_flag;
+       }
+
+       block->client = client;
+       mutex_init(&block->mtx);
+       refcount_set(&block->refcnt, 1);
+       idr_init_base(&block->counter_idr, block->offset);
+
+       err = prestera_counter_block_list_add(counter, block);
+       if (err)
+               goto err_list_add;
+
+       return block;
+
+err_list_add:
+       idr_destroy(&block->counter_idr);
+       mutex_destroy(&block->mtx);
+       kfree(block->counter_flag);
+err_flag:
+       kfree(block->stats);
+err_stats:
+       prestera_hw_counter_block_release(counter->sw, block->id);
+err_block:
+       kfree(block);
+       return ERR_PTR(err);
+}
+
+static void prestera_counter_block_put(struct prestera_counter *counter,
+                                      struct prestera_counter_block *block)
+{
+       u32 i;
+
+       if (!prestera_counter_block_decref(block))
+               return;
+
+       prestera_counter_lock(counter);
+       for (i = 0; i < counter->block_list_len; i++) {
+               if (counter->block_list[i] &&
+                   counter->block_list[i]->id == block->id) {
+                       counter->block_list[i] = NULL;
+                       break;
+               }
+       }
+       prestera_counter_unlock(counter);
+
+       WARN_ON(!idr_is_empty(&block->counter_idr));
+
+       prestera_hw_counter_block_release(counter->sw, block->id);
+       idr_destroy(&block->counter_idr);
+       mutex_destroy(&block->mtx);
+       kfree(block->stats);
+       kfree(block);
+}
+
+static int prestera_counter_get_vacant(struct prestera_counter_block *block,
+                                      u32 *id)
+{
+       int free_id;
+
+       if (block->full)
+               return -ENOSPC;
+
+       prestera_counter_block_lock(block);
+       free_id = idr_alloc_cyclic(&block->counter_idr, NULL, block->offset,
+                                  block->offset + block->num_counters,
+                                  GFP_KERNEL);
+       if (free_id < 0) {
+               if (free_id == -ENOSPC)
+                       block->full = true;
+
+               prestera_counter_block_unlock(block);
+               return free_id;
+       }
+       *id = free_id;
+       prestera_counter_block_unlock(block);
+
+       return 0;
+}
+
+int prestera_counter_get(struct prestera_counter *counter, u32 client,
+                        struct prestera_counter_block **bl, u32 *counter_id)
+{
+       struct prestera_counter_block *block;
+       int err;
+       u32 id;
+
+get_next_block:
+       block = prestera_counter_block_get(counter, client);
+       if (IS_ERR(block))
+               return PTR_ERR(block);
+
+       err = prestera_counter_get_vacant(block, &id);
+       if (err) {
+               prestera_counter_block_put(counter, block);
+
+               if (err == -ENOSPC)
+                       goto get_next_block;
+
+               return err;
+       }
+
+       prestera_counter_block_lock(block);
+       if (block->is_updating)
+               block->counter_flag[id - block->offset] = COUNTER_FLAG_INVALID;
+       prestera_counter_block_unlock(block);
+
+       *counter_id = id;
+       *bl = block;
+
+       return 0;
+}
+
+void prestera_counter_put(struct prestera_counter *counter,
+                         struct prestera_counter_block *block, u32 counter_id)
+{
+       if (!block)
+               return;
+
+       prestera_counter_block_lock(block);
+       idr_remove(&block->counter_idr, counter_id);
+       block->full = false;
+       prestera_counter_stats_clear(block, counter_id);
+       prestera_counter_block_unlock(block);
+
+       prestera_hw_counter_clear(counter->sw, block->id, counter_id);
+       prestera_counter_block_put(counter, block);
+}
+
+static u32 prestera_counter_block_idx_next(struct prestera_counter *counter,
+                                          u32 curr_idx)
+{
+       u32 idx, i, start = curr_idx + 1;
+
+       prestera_counter_lock(counter);
+       for (i = 0; i < counter->block_list_len; i++) {
+               idx = (start + i) % counter->block_list_len;
+               if (!counter->block_list[idx])
+                       continue;
+
+               prestera_counter_unlock(counter);
+               return idx;
+       }
+       prestera_counter_unlock(counter);
+
+       return 0;
+}
+
+static struct prestera_counter_block *
+prestera_counter_block_get_by_idx(struct prestera_counter *counter, u32 idx)
+{
+       if (idx >= counter->block_list_len)
+               return NULL;
+
+       prestera_counter_lock(counter);
+
+       if (!counter->block_list[idx] ||
+           !prestera_counter_block_incref(counter->block_list[idx])) {
+               prestera_counter_unlock(counter);
+               return NULL;
+       }
+
+       prestera_counter_unlock(counter);
+       return counter->block_list[idx];
+}
+
+static void prestera_counter_stats_work(struct work_struct *work)
+{
+       struct delayed_work *dl_work =
+               container_of(work, struct delayed_work, work);
+       struct prestera_counter *counter =
+               container_of(dl_work, struct prestera_counter, stats_dw);
+       struct prestera_counter_block *block;
+       u32 resched_time = COUNTER_POLL_TIME;
+       u32 count = COUNTER_BULK_SIZE;
+       bool done = false;
+       int err;
+       u32 i;
+
+       block = prestera_counter_block_get_by_idx(counter, counter->curr_idx);
+       if (!block) {
+               if (counter->is_fetching)
+                       goto abort;
+
+               goto next;
+       }
+
+       if (!counter->is_fetching) {
+               err = prestera_hw_counter_trigger(counter->sw, block->id);
+               if (err)
+                       goto abort;
+
+               prestera_counter_block_lock(block);
+               block->is_updating = true;
+               prestera_counter_block_unlock(block);
+
+               counter->is_fetching = true;
+               counter->total_read = 0;
+               resched_time = COUNTER_RESCHED_TIME;
+               goto resched;
+       }
+
+       prestera_counter_block_lock(block);
+       err = prestera_hw_counters_get(counter->sw, counter->total_read,
+                                      &count, &done,
+                                      &block->stats[counter->total_read]);
+       prestera_counter_block_unlock(block);
+       if (err)
+               goto abort;
+
+       counter->total_read += count;
+       if (!done || counter->total_read < block->num_counters) {
+               resched_time = COUNTER_RESCHED_TIME;
+               goto resched;
+       }
+
+       for (i = 0; i < block->num_counters; i++) {
+               if (block->counter_flag[i] == COUNTER_FLAG_INVALID) {
+                       prestera_counter_block_lock(block);
+                       block->counter_flag[i] = COUNTER_FLAG_READY;
+                       memset(&block->stats[i], 0, sizeof(*block->stats));
+                       prestera_counter_block_unlock(block);
+               }
+       }
+
+       prestera_counter_block_lock(block);
+       block->is_updating = false;
+       prestera_counter_block_unlock(block);
+
+       goto next;
+abort:
+       prestera_hw_counter_abort(counter->sw);
+next:
+       counter->is_fetching = false;
+       counter->curr_idx =
+               prestera_counter_block_idx_next(counter, counter->curr_idx);
+resched:
+       if (block)
+               prestera_counter_block_put(counter, block);
+
+       schedule_delayed_work(&counter->stats_dw, resched_time);
+}
+
+/* Can be executed without rtnl_lock().
+ * So pay attention when something changing.
+ */
+int prestera_counter_stats_get(struct prestera_counter *counter,
+                              struct prestera_counter_block *block,
+                              u32 counter_id, u64 *packets, u64 *bytes)
+{
+       if (!block || !prestera_counter_is_ready(block, counter_id)) {
+               *packets = 0;
+               *bytes = 0;
+               return 0;
+       }
+
+       prestera_counter_block_lock(block);
+       *packets = block->stats[counter_id - block->offset].packets;
+       *bytes = block->stats[counter_id - block->offset].bytes;
+
+       prestera_counter_stats_clear(block, counter_id);
+       prestera_counter_block_unlock(block);
+
+       return 0;
+}
+
+int prestera_counter_init(struct prestera_switch *sw)
+{
+       struct prestera_counter *counter;
+
+       counter = kzalloc(sizeof(*counter), GFP_KERNEL);
+       if (!counter)
+               return -ENOMEM;
+
+       counter->block_list = kzalloc(sizeof(*counter->block_list), GFP_KERNEL);
+       if (!counter->block_list) {
+               kfree(counter);
+               return -ENOMEM;
+       }
+
+       mutex_init(&counter->mtx);
+       counter->block_list_len = 1;
+       counter->sw = sw;
+       sw->counter = counter;
+
+       INIT_DELAYED_WORK(&counter->stats_dw, prestera_counter_stats_work);
+       schedule_delayed_work(&counter->stats_dw, COUNTER_POLL_TIME);
+
+       return 0;
+}
+
+void prestera_counter_fini(struct prestera_switch *sw)
+{
+       struct prestera_counter *counter = sw->counter;
+       u32 i;
+
+       cancel_delayed_work_sync(&counter->stats_dw);
+
+       for (i = 0; i < counter->block_list_len; i++)
+               WARN_ON(counter->block_list[i]);
+
+       mutex_destroy(&counter->mtx);
+       kfree(counter->block_list);
+       kfree(counter);
+}
diff --git a/drivers/net/ethernet/marvell/prestera/prestera_counter.h b/drivers/net/ethernet/marvell/prestera/prestera_counter.h
new file mode 100644 (file)
index 0000000..ad6b739
--- /dev/null
@@ -0,0 +1,30 @@
+/* SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0 */
+/* Copyright (c) 2021 Marvell International Ltd. All rights reserved. */
+
+#ifndef _PRESTERA_COUNTER_H_
+#define _PRESTERA_COUNTER_H_
+
+#include <linux/types.h>
+
+struct prestera_counter_stats {
+       u64 packets;
+       u64 bytes;
+};
+
+struct prestera_switch;
+struct prestera_counter;
+struct prestera_counter_block;
+
+int prestera_counter_init(struct prestera_switch *sw);
+void prestera_counter_fini(struct prestera_switch *sw);
+
+int prestera_counter_get(struct prestera_counter *counter, u32 client,
+                        struct prestera_counter_block **block,
+                        u32 *counter_id);
+void prestera_counter_put(struct prestera_counter *counter,
+                         struct prestera_counter_block *block, u32 counter_id);
+int prestera_counter_stats_get(struct prestera_counter *counter,
+                              struct prestera_counter_block *block,
+                              u32 counter_id, u64 *packets, u64 *bytes);
+
+#endif /* _PRESTERA_COUNTER_H_ */
index d6c425b651cba067b0be585d2454497ee5345ec0..92cb5e9099c6c54e008f495318544f20ad3147b0 100644 (file)
@@ -9,6 +9,7 @@
 #include "prestera.h"
 #include "prestera_hw.h"
 #include "prestera_acl.h"
+#include "prestera_counter.h"
 
 #define PRESTERA_SWITCH_INIT_TIMEOUT_MS (30 * 1000)
 
@@ -38,6 +39,13 @@ enum prestera_cmd_type_t {
        PRESTERA_CMD_TYPE_BRIDGE_PORT_ADD = 0x402,
        PRESTERA_CMD_TYPE_BRIDGE_PORT_DELETE = 0x403,
 
+       PRESTERA_CMD_TYPE_COUNTER_GET = 0x510,
+       PRESTERA_CMD_TYPE_COUNTER_ABORT = 0x511,
+       PRESTERA_CMD_TYPE_COUNTER_TRIGGER = 0x512,
+       PRESTERA_CMD_TYPE_COUNTER_BLOCK_GET = 0x513,
+       PRESTERA_CMD_TYPE_COUNTER_BLOCK_RELEASE = 0x514,
+       PRESTERA_CMD_TYPE_COUNTER_CLEAR = 0x515,
+
        PRESTERA_CMD_TYPE_VTCAM_CREATE = 0x540,
        PRESTERA_CMD_TYPE_VTCAM_DESTROY = 0x541,
        PRESTERA_CMD_TYPE_VTCAM_RULE_ADD = 0x550,
@@ -408,7 +416,34 @@ struct prestera_msg_vtcam_resp {
 
 struct prestera_msg_acl_action {
        __le32 id;
-       __le32 __reserved[7];
+       __le32 __reserved;
+       union {
+               struct {
+                       __le32 id;
+               } count;
+               __le32 reserved[6];
+       };
+};
+
+struct prestera_msg_counter_req {
+       struct prestera_msg_cmd cmd;
+       __le32 client;
+       __le32 block_id;
+       __le32 num_counters;
+};
+
+struct prestera_msg_counter_stats {
+       __le64 packets;
+       __le64 bytes;
+};
+
+struct prestera_msg_counter_resp {
+       struct prestera_msg_ret ret;
+       __le32 block_id;
+       __le32 offset;
+       __le32 num_counters;
+       __le32 done;
+       struct prestera_msg_counter_stats stats[0];
 };
 
 struct prestera_msg_span_req {
@@ -512,6 +547,8 @@ static void prestera_hw_build_tests(void)
        BUILD_BUG_ON(sizeof(struct prestera_msg_vtcam_rule_del_req) != 12);
        BUILD_BUG_ON(sizeof(struct prestera_msg_vtcam_bind_req) != 20);
        BUILD_BUG_ON(sizeof(struct prestera_msg_acl_action) != 32);
+       BUILD_BUG_ON(sizeof(struct prestera_msg_counter_req) != 16);
+       BUILD_BUG_ON(sizeof(struct prestera_msg_counter_stats) != 16);
 
        /* check responses */
        BUILD_BUG_ON(sizeof(struct prestera_msg_common_resp) != 8);
@@ -523,6 +560,7 @@ static void prestera_hw_build_tests(void)
        BUILD_BUG_ON(sizeof(struct prestera_msg_span_resp) != 12);
        BUILD_BUG_ON(sizeof(struct prestera_msg_rxtx_resp) != 12);
        BUILD_BUG_ON(sizeof(struct prestera_msg_vtcam_resp) != 16);
+       BUILD_BUG_ON(sizeof(struct prestera_msg_counter_resp) != 24);
 
        /* check events */
        BUILD_BUG_ON(sizeof(struct prestera_msg_event_port) != 20);
@@ -1072,6 +1110,9 @@ prestera_acl_rule_add_put_action(struct prestera_msg_acl_action *action,
        case PRESTERA_ACL_RULE_ACTION_TRAP:
                /* just rule action id, no specific data */
                break;
+       case PRESTERA_ACL_RULE_ACTION_COUNT:
+               action->count.id = __cpu_to_le32(info->count.id);
+               break;
        default:
                return -EINVAL;
        }
@@ -1831,3 +1872,100 @@ void prestera_hw_event_handler_unregister(struct prestera_switch *sw,
        list_del_rcu(&eh->list);
        kfree_rcu(eh, rcu);
 }
+
+int prestera_hw_counter_trigger(struct prestera_switch *sw, u32 block_id)
+{
+       struct prestera_msg_counter_req req = {
+               .block_id = __cpu_to_le32(block_id)
+       };
+
+       return prestera_cmd(sw, PRESTERA_CMD_TYPE_COUNTER_TRIGGER,
+                           &req.cmd, sizeof(req));
+}
+
+int prestera_hw_counter_abort(struct prestera_switch *sw)
+{
+       struct prestera_msg_counter_req req;
+
+       return prestera_cmd(sw, PRESTERA_CMD_TYPE_COUNTER_ABORT,
+                           &req.cmd, sizeof(req));
+}
+
+int prestera_hw_counters_get(struct prestera_switch *sw, u32 idx,
+                            u32 *len, bool *done,
+                            struct prestera_counter_stats *stats)
+{
+       struct prestera_msg_counter_resp *resp;
+       struct prestera_msg_counter_req req = {
+               .block_id = __cpu_to_le32(idx),
+               .num_counters = __cpu_to_le32(*len),
+       };
+       size_t size = sizeof(*resp) + sizeof(*resp->stats) * (*len);
+       int err, i;
+
+       resp = kmalloc(size, GFP_KERNEL);
+       if (!resp)
+               return -ENOMEM;
+
+       err = prestera_cmd_ret(sw, PRESTERA_CMD_TYPE_COUNTER_GET,
+                              &req.cmd, sizeof(req), &resp->ret, size);
+       if (err)
+               goto free_buff;
+
+       for (i = 0; i < __le32_to_cpu(resp->num_counters); i++) {
+               stats[i].packets += __le64_to_cpu(resp->stats[i].packets);
+               stats[i].bytes += __le64_to_cpu(resp->stats[i].bytes);
+       }
+
+       *len = __le32_to_cpu(resp->num_counters);
+       *done = __le32_to_cpu(resp->done);
+
+free_buff:
+       kfree(resp);
+       return err;
+}
+
+int prestera_hw_counter_block_get(struct prestera_switch *sw,
+                                 u32 client, u32 *block_id, u32 *offset,
+                                 u32 *num_counters)
+{
+       struct prestera_msg_counter_resp resp;
+       struct prestera_msg_counter_req req = {
+               .client = __cpu_to_le32(client)
+       };
+       int err;
+
+       err = prestera_cmd_ret(sw, PRESTERA_CMD_TYPE_COUNTER_BLOCK_GET,
+                              &req.cmd, sizeof(req), &resp.ret, sizeof(resp));
+       if (err)
+               return err;
+
+       *block_id = __le32_to_cpu(resp.block_id);
+       *offset = __le32_to_cpu(resp.offset);
+       *num_counters = __le32_to_cpu(resp.num_counters);
+
+       return 0;
+}
+
+int prestera_hw_counter_block_release(struct prestera_switch *sw,
+                                     u32 block_id)
+{
+       struct prestera_msg_counter_req req = {
+               .block_id = __cpu_to_le32(block_id)
+       };
+
+       return prestera_cmd(sw, PRESTERA_CMD_TYPE_COUNTER_BLOCK_RELEASE,
+                           &req.cmd, sizeof(req));
+}
+
+int prestera_hw_counter_clear(struct prestera_switch *sw, u32 block_id,
+                             u32 counter_id)
+{
+       struct prestera_msg_counter_req req = {
+               .block_id = __cpu_to_le32(block_id),
+               .num_counters = __cpu_to_le32(counter_id)
+       };
+
+       return prestera_cmd(sw, PRESTERA_CMD_TYPE_COUNTER_CLEAR,
+                           &req.cmd, sizeof(req));
+}
index 6b7a9f8e2ea26631ad0e8d9f9601264708ac60c2..0496e454e148513b6f5e7032bcc5ff4686a62cc7 100644 (file)
@@ -117,6 +117,12 @@ enum prestera_hw_vtcam_direction_t {
        PRESTERA_HW_VTCAM_DIR_EGRESS = 1,
 };
 
+enum {
+       PRESTERA_HW_COUNTER_CLIENT_LOOKUP_0 = 0,
+       PRESTERA_HW_COUNTER_CLIENT_LOOKUP_1 = 1,
+       PRESTERA_HW_COUNTER_CLIENT_LOOKUP_2 = 2,
+};
+
 struct prestera_switch;
 struct prestera_port;
 struct prestera_port_stats;
@@ -130,6 +136,7 @@ typedef void (*prestera_event_cb_t)
 struct prestera_rxtx_params;
 struct prestera_acl_hw_action_info;
 struct prestera_acl_iface;
+struct prestera_counter_stats;
 
 /* Switch API */
 int prestera_hw_switch_init(struct prestera_switch *sw);
@@ -211,6 +218,20 @@ int prestera_hw_vtcam_iface_unbind(struct prestera_switch *sw,
                                   struct prestera_acl_iface *iface,
                                   u32 vtcam_id);
 
+/* Counter API */
+int prestera_hw_counter_trigger(struct prestera_switch *sw, u32 block_id);
+int prestera_hw_counter_abort(struct prestera_switch *sw);
+int prestera_hw_counters_get(struct prestera_switch *sw, u32 idx,
+                            u32 *len, bool *done,
+                            struct prestera_counter_stats *stats);
+int prestera_hw_counter_block_get(struct prestera_switch *sw,
+                                 u32 client, u32 *block_id, u32 *offset,
+                                 u32 *num_counters);
+int prestera_hw_counter_block_release(struct prestera_switch *sw,
+                                     u32 block_id);
+int prestera_hw_counter_clear(struct prestera_switch *sw, u32 block_id,
+                             u32 counter_id);
+
 /* SPAN API */
 int prestera_hw_span_get(const struct prestera_port *port, u8 *span_id);
 int prestera_hw_span_bind(const struct prestera_port *port, u8 span_id);
index 4369a3ffad45b7c660c5a93ecb20225e5afa7f3c..a0dbad5cb88ddc630eaaf7be588b9db708559d4c 100644 (file)
@@ -18,6 +18,7 @@
 #include "prestera_rxtx.h"
 #include "prestera_devlink.h"
 #include "prestera_ethtool.h"
+#include "prestera_counter.h"
 #include "prestera_switchdev.h"
 
 #define PRESTERA_MTU_DEFAULT   1536
@@ -904,6 +905,10 @@ static int prestera_switch_init(struct prestera_switch *sw)
        if (err)
                goto err_handlers_register;
 
+       err = prestera_counter_init(sw);
+       if (err)
+               goto err_counter_init;
+
        err = prestera_acl_init(sw);
        if (err)
                goto err_acl_init;
@@ -936,6 +941,8 @@ err_dl_register:
 err_span_init:
        prestera_acl_fini(sw);
 err_acl_init:
+       prestera_counter_fini(sw);
+err_counter_init:
        prestera_event_handlers_unregister(sw);
 err_handlers_register:
        prestera_rxtx_switch_fini(sw);
@@ -956,6 +963,7 @@ static void prestera_switch_fini(struct prestera_switch *sw)
        prestera_devlink_traps_unregister(sw);
        prestera_span_fini(sw);
        prestera_acl_fini(sw);
+       prestera_counter_fini(sw);
        prestera_event_handlers_unregister(sw);
        prestera_rxtx_switch_fini(sw);
        prestera_switchdev_fini(sw);