sonic-buildimage/platform/broadcom/saibcm-modules/sdklt/linux/knet/ngknet_procfs.c

665 lines
21 KiB
C

/*! \file ngknet_procfs.c
*
* <description>
*
*/
/*
* $Copyright: Copyright 2018-2022 Broadcom. All rights reserved.
* The term 'Broadcom' refers to Broadcom Inc. and/or its subsidiaries.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* A copy of the GNU General Public License version 2 (GPLv2) can
* be found in the LICENSES folder.$
*/
#include <lkm/lkm.h>
#include <lkm/ngknet_ioctl.h>
#include "ngknet_main.h"
#include "ngknet_extra.h"
extern struct ngknet_dev ngknet_devices[];
static struct proc_dir_entry *proc_root = NULL;
static void
proc_data_show(struct seq_file *m, const unsigned char *buf, size_t len)
{
uint32_t i;
if (!buf || !len) {
seq_printf(m, "\n");
return;
}
for (i = 0; i < len; i++) {
seq_printf(m, "%02x ", buf[i]);
if ((i + 1) % 32 == 0 || (i + 1) == len) {
seq_printf(m, "\n");
if ((i + 1) < len) {
seq_printf(m, " ");
}
}
}
}
static int
proc_debug_level_show(struct seq_file *m, void *v)
{
seq_printf(m, "Debug level: 0x%x\n", ngknet_debug_level_get());
return 0;
}
static int
proc_debug_level_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_debug_level_show, NULL);
}
static ssize_t
proc_debug_level_write(struct file *file, const char *buf,
size_t count, loff_t *loff)
{
char level_str[11] = {0};
int debug_level;
if (copy_from_user(level_str, buf, sizeof(level_str) - 1)) {
return -EFAULT;
}
debug_level = simple_strtol(level_str, NULL, 16);
ngknet_debug_level_set(debug_level);
printk("Debug level set to: 0x%x\n", debug_level);
return count;
}
static int
proc_debug_level_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_debug_level_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_debug_level_open,
.proc_read = seq_read,
.proc_write = proc_debug_level_write,
.proc_lseek = seq_lseek,
.proc_release = proc_debug_level_release,
};
static int
proc_device_info_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
struct bcmcnet_dev_info *info;
int di, qi, ai = 0;
int rv;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
ai++;
rv = bcmcnet_pdma_dev_info_get(&dev->pdma_dev);
if (SHR_FAILURE(rv)) {
printk("ngknet: get device%d info failed\n", di);
break;
}
info = &dev->pdma_dev.info;
seq_printf(m, "dev_no: %d\n", di);
seq_printf(m, "dev_name: %s\n", info->dev_name);
seq_printf(m, "dev_id: 0x%x\n", info->dev_id);
seq_printf(m, "dev_type: %d\n", info->dev_type);
seq_printf(m, "max_groups: %d\n", info->max_groups);
seq_printf(m, "max_queues: %d\n", info->max_queues);
seq_printf(m, "bm_groups: 0x%x\n", info->bm_groups);
seq_printf(m, "bm_rx_queues: 0x%x\n", info->bm_rx_queues);
seq_printf(m, "bm_tx_queues: 0x%x\n", info->bm_tx_queues);
seq_printf(m, "nb_groups: %d\n", info->nb_groups);
seq_printf(m, "nb_rx_queues: %d\n", info->nb_rx_queues);
seq_printf(m, "nb_tx_queues: %d\n", info->nb_tx_queues);
seq_printf(m, "rx_desc_size: %d\n", info->rx_desc_size);
seq_printf(m, "tx_desc_size: %d\n", info->tx_desc_size);
seq_printf(m, "rx_ph_size: %d\n", info->rx_ph_size);
seq_printf(m, "tx_ph_size: %d\n", info->tx_ph_size);
for (qi = 0; qi < info->nb_rx_queues; qi++) {
seq_printf(m, "rx_buf_sz[%d]: %d\n", qi, info->rx_buf_size[qi]);
}
for (qi = 0; qi < info->nb_rx_queues; qi++) {
seq_printf(m, "nb_rx_desc[%d]: %d\n", qi, info->nb_rx_desc[qi]);
}
for (qi = 0; qi < info->nb_rx_queues; qi++) {
seq_printf(m, "rxq_state[%d]: 0x%x\n", qi, info->rxq_state[qi]);
}
for (qi = 0; qi < info->nb_tx_queues; qi++) {
seq_printf(m, "nb_tx_desc[%d]: %d\n", qi, info->nb_tx_desc[qi]);
}
for (qi = 0; qi < info->nb_tx_queues; qi++) {
seq_printf(m, "txq_state[%d]: 0x%x\n", qi, info->txq_state[qi]);
}
}
if (!ai) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "------------------------\n");
seq_printf(m, "Total %d devices\n", ai);
}
return 0;
}
static int
proc_device_info_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_device_info_show, NULL);
}
static int
proc_device_info_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_device_info_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_device_info_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_device_info_release,
};
static int
proc_filter_info_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
ngknet_filter_t filt = {0};
int di, dn = 0, fn = 0;
int rv;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
dn++;
do {
rv = ngknet_filter_get_next(dev, &filt);
if (SHR_FAILURE(rv)) {
printk("ngknet: get device%d filter failed\n", di);
break;
}
fn++;
seq_printf(m, "\n");
seq_printf(m, "dev_no: %d\n", di);
seq_printf(m, "id: %d\n", filt.id);
seq_printf(m, "next: %d\n", filt.next);
seq_printf(m, "type: %d\n", filt.type);
seq_printf(m, "flags: 0x%x\n", filt.flags);
seq_printf(m, "prio: %d\n", filt.priority);
seq_printf(m, "chan: %d\n", filt.chan);
seq_printf(m, "desc: %s\n", filt.desc);
seq_printf(m, "dest_type: %d\n", filt.dest_type);
seq_printf(m, "dest_id: %d\n", filt.dest_id);
seq_printf(m, "dest_proto: 0x%x\n", filt.dest_proto);
seq_printf(m, "mirror_type: %d\n", filt.mirror_type);
seq_printf(m, "mirror_id: %d\n", filt.mirror_id);
seq_printf(m, "mirror_proto: 0x%x\n", filt.mirror_proto);
seq_printf(m, "oob_offset: %d\n", filt.oob_data_offset);
seq_printf(m, "oob_size: %d\n", filt.oob_data_size);
seq_printf(m, "pkt_offset: %d\n", filt.pkt_data_offset);
seq_printf(m, "pkt_size: %d\n", filt.pkt_data_size);
seq_printf(m, "filt_data: ");
proc_data_show(m, filt.data.b, filt.oob_data_size + filt.pkt_data_size);
seq_printf(m, "filt_mask: ");
proc_data_show(m, filt.mask.b, filt.oob_data_size + filt.pkt_data_size);
seq_printf(m, "user_data: ");
proc_data_show(m, filt.user_data, NGKNET_FILTER_USER_DATA);
seq_printf(m, "hits: %llu\n", ((struct filt_ctrl *)dev->fc[filt.id])->hits);
} while (filt.next);
}
if (!dn) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "--------------------------------\n");
seq_printf(m, "Total %d devices, %d filters\n", dn, fn);
}
return 0;
}
static int
proc_filter_info_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_filter_info_show, NULL);
}
static int
proc_filter_info_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_filter_info_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_filter_info_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_filter_info_release,
};
static int
proc_netif_info_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
struct net_device *ndev;
struct ngknet_private *priv;
ngknet_netif_t netif = {0};
int di, ma, dn = 0, nn = 0;
int rv;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
dn++;
do {
rv = ngknet_netif_get_next(dev, &netif);
if (SHR_FAILURE(rv)) {
printk("ngknet: get device%d netif failed\n", di);
break;
}
nn++;
ndev = netif.id == 0 ? dev->net_dev : dev->vdev[netif.id];
priv = netdev_priv(ndev);
seq_printf(m, "\n");
seq_printf(m, "dev_no: %d\n", di);
seq_printf(m, "id: %d\n", netif.id);
seq_printf(m, "next: %d\n", netif.next);
seq_printf(m, "type: %d\n", netif.type);
seq_printf(m, "flags: 0x%x\n", netif.flags);
seq_printf(m, "vlan: %d\n", netif.vlan);
seq_printf(m, "mac: ");
for (ma = 0; ma < 6; ma++) {
if (ma == 5) {
seq_printf(m, "%02x\n", netif.macaddr[ma]);
} else {
seq_printf(m, "%02x:", netif.macaddr[ma]);
}
}
seq_printf(m, "mtu: %d\n", netif.mtu);
seq_printf(m, "chan: %d\n", netif.chan);
seq_printf(m, "name: %s\n", netif.name);
seq_printf(m, "meta_off: %d\n", netif.meta_off);
seq_printf(m, "meta_len: %d\n", netif.meta_len);
seq_printf(m, "meta_data: ");
proc_data_show(m, netif.meta_data, netif.meta_len);
seq_printf(m, "user_data: ");
proc_data_show(m, netif.user_data, NGKNET_NETIF_USER_DATA);
seq_printf(m, "rx_packets: %lu\n", priv->stats.rx_packets);
seq_printf(m, "rx_bytes: %lu\n", priv->stats.rx_bytes);
seq_printf(m, "rx_dropped: %lu\n", priv->stats.rx_dropped);
seq_printf(m, "rx_errors: %lu\n", priv->stats.rx_errors);
seq_printf(m, "tx_packets: %lu\n", priv->stats.tx_packets);
seq_printf(m, "tx_bytes: %lu\n", priv->stats.tx_bytes);
seq_printf(m, "tx_dropped: %lu\n", priv->stats.tx_dropped);
seq_printf(m, "tx_errors: %lu\n", priv->stats.tx_errors);
} while (netif.next);
}
if (!dn) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "--------------------------------\n");
seq_printf(m, "Total %d devices, %d netifs\n", dn, nn);
}
return 0;
}
static int
proc_netif_info_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_netif_info_show, NULL);
}
static int
proc_netif_info_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_netif_info_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_netif_info_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_netif_info_release,
};
static int
proc_pkt_stats_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
struct bcmcnet_dev_stats *stats;
int di, qi, ai = 0;
int rv;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
ai++;
rv = bcmcnet_pdma_dev_stats_get(&dev->pdma_dev);
if (SHR_FAILURE(rv)) {
printk("ngknet: get device%d stats failed\n", di);
break;
}
stats = &dev->pdma_dev.stats;
seq_printf(m, "rx_packets: %llu\n", (unsigned long long)stats->rx_packets);
seq_printf(m, "rx_bytes: %llu\n", (unsigned long long)stats->rx_bytes);
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_rxq; qi++) {
seq_printf(m, "rx_packets[%d]: %llu\n", qi, (unsigned long long)stats->rxq_packets[qi]);
seq_printf(m, "rx_bytes[%d]: %llu\n", qi, (unsigned long long)stats->rxq_bytes[qi]);
}
seq_printf(m, "rx_dropped: %llu\n", (unsigned long long)stats->rx_dropped);
seq_printf(m, "rx_errors: %llu\n", (unsigned long long)stats->rx_errors);
seq_printf(m, "rx_head_errors: %llu\n", (unsigned long long)stats->rx_head_errors);
seq_printf(m, "rx_data_errors: %llu\n", (unsigned long long)stats->rx_data_errors);
seq_printf(m, "rx_cell_errors: %llu\n", (unsigned long long)stats->rx_cell_errors);
seq_printf(m, "rx_nomems: %llu\n", (unsigned long long)stats->rx_nomems);
seq_printf(m, "tx_packets: %llu\n", (unsigned long long)stats->tx_packets);
seq_printf(m, "tx_bytes: %llu\n", (unsigned long long)stats->tx_bytes);
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_txq; qi++) {
seq_printf(m, "tx_packets[%d]: %llu\n", qi, (unsigned long long)stats->txq_packets[qi]);
seq_printf(m, "tx_bytes[%d]: %llu\n", qi, (unsigned long long)stats->txq_bytes[qi]);
}
seq_printf(m, "tx_dropped: %llu\n", (unsigned long long)stats->tx_dropped);
seq_printf(m, "tx_errors: %llu\n", (unsigned long long)stats->tx_errors);
seq_printf(m, "tx_xoffs: %llu\n", (unsigned long long)stats->tx_xoffs);
seq_printf(m, "interrupts: %llu\n", (unsigned long long)stats->intrs);
}
if (!ai) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "------------------------\n");
seq_printf(m, "Total %d devices\n", ai);
}
return 0;
}
static int
proc_pkt_stats_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_pkt_stats_show, NULL);
}
static int
proc_pkt_stats_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_pkt_stats_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_pkt_stats_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_pkt_stats_release,
};
static int
proc_rate_limit_show(struct seq_file *m, void *v)
{
seq_printf(m, "Rx rate limit: %d pps\n", ngknet_rx_rate_limit_get());
return 0;
}
static int
proc_rate_limit_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_rate_limit_show, NULL);
}
static ssize_t
proc_rate_limit_write(struct file *file, const char *buf,
size_t count, loff_t *loff)
{
char limit_str[9] = {0};
int rate_limit;
if (copy_from_user(limit_str, buf, sizeof(limit_str) - 1)) {
return -EFAULT;
}
rate_limit = simple_strtol(limit_str, NULL, 10);
ngknet_rx_rate_limit_set(rate_limit);
printk("Rx rate limit set to: %d pps\n", rate_limit);
return count;
}
static int
proc_rate_limit_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_rate_limit_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_rate_limit_open,
.proc_read = seq_read,
.proc_write = proc_rate_limit_write,
.proc_lseek = seq_lseek,
.proc_release = proc_rate_limit_release,
};
static int
proc_reg_status_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
int di, qi, ai = 0;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
ai++;
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_rxq; qi++) {
bcmcnet_pdma_rx_queue_reg_dump(&dev->pdma_dev, qi);
}
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_txq; qi++) {
bcmcnet_pdma_tx_queue_reg_dump(&dev->pdma_dev, qi);
}
}
if (!ai) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "------------------------\n");
seq_printf(m, "Total %d devices\n", ai);
}
return 0;
}
static int
proc_reg_status_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_reg_status_show, NULL);
}
static int
proc_reg_status_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_reg_status_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_reg_status_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_reg_status_release,
};
static int
proc_ring_status_show(struct seq_file *m, void *v)
{
struct ngknet_dev *dev;
int di, qi, ai = 0;
for (di = 0; di < NUM_PDMA_DEV_MAX; di++) {
dev = &ngknet_devices[di];
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
continue;
}
ai++;
seq_printf(m, "%s-%d, ", "Unit", di);
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_rxq; qi++) {
bcmcnet_pdma_rx_ring_dump(&dev->pdma_dev, qi);
}
seq_printf(m, "%s%d, ", "Rx queues: ", qi);
for (qi = 0; qi < dev->pdma_dev.ctrl.nb_txq; qi++) {
bcmcnet_pdma_tx_ring_dump(&dev->pdma_dev, qi);
}
seq_printf(m, "%s%d. ", "Tx queues: ", qi);
seq_printf(m, "\n");
}
if (!ai) {
seq_printf(m, "%s\n", "No active device");
} else {
seq_printf(m, "------------------------\n");
seq_printf(m, "Total %d devices\n", ai);
}
return 0;
}
static int
proc_ring_status_open(struct inode *inode, struct file *file)
{
return single_open(file, proc_ring_status_show, NULL);
}
static int
proc_ring_status_release(struct inode *inode, struct file *file)
{
return single_release(inode, file);
}
static struct proc_ops proc_ring_status_fops = {
PROC_OWNER(THIS_MODULE)
.proc_open = proc_ring_status_open,
.proc_read = seq_read,
.proc_lseek = seq_lseek,
.proc_release = proc_ring_status_release,
};
int
ngknet_procfs_init(void)
{
struct proc_dir_entry *entry = NULL;
proc_root = proc_mkdir(NGKNET_MODULE_NAME, NULL);
if (proc_root == NULL) {
printk(KERN_ERR "ngknet: proc_mkdir failed\n");
return -1;
}
PROC_CREATE(entry, "debug_level", 0666, proc_root, &proc_debug_level_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "device_info", 0444, proc_root, &proc_device_info_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "filter_info", 0444, proc_root, &proc_filter_info_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "netif_info", 0444, proc_root, &proc_netif_info_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "pkt_stats", 0444, proc_root, &proc_pkt_stats_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "rate_limit", 0666, proc_root, &proc_rate_limit_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "reg_status", 0444, proc_root, &proc_reg_status_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
PROC_CREATE(entry, "ring_status", 0444, proc_root, &proc_ring_status_fops);
if (entry == NULL) {
printk(KERN_ERR "ngknet: proc_create failed\n");
return -1;
}
return 0;
}
int
ngknet_procfs_cleanup(void)
{
remove_proc_entry("debug_level", proc_root);
remove_proc_entry("device_info", proc_root);
remove_proc_entry("filter_info", proc_root);
remove_proc_entry("netif_info", proc_root);
remove_proc_entry("pkt_stats", proc_root);
remove_proc_entry("rate_limit", proc_root);
remove_proc_entry("reg_status", proc_root);
remove_proc_entry("ring_status", proc_root);
remove_proc_entry(NGKNET_MODULE_NAME, NULL);
return 0;
}