mirror of
https://github.com/torvalds/linux.git
synced 2025-11-01 00:58:39 +02:00
Implement device stats operations for hw stats and qp stats. Co-developed-by: Allen Hubbe <allen.hubbe@amd.com> Signed-off-by: Allen Hubbe <allen.hubbe@amd.com> Signed-off-by: Abhijit Gangurde <abhijit.gangurde@amd.com> Link: https://patch.msgid.link/20250903061606.4139957-14-abhijit.gangurde@amd.com Signed-off-by: Leon Romanovsky <leon@kernel.org>
440 lines
12 KiB
C
440 lines
12 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Copyright (C) 2018-2025, Advanced Micro Devices, Inc. */
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/printk.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/irq.h>
|
|
#include <net/addrconf.h>
|
|
#include <rdma/ib_addr.h>
|
|
#include <rdma/ib_mad.h>
|
|
|
|
#include "ionic_ibdev.h"
|
|
|
|
#define DRIVER_DESCRIPTION "AMD Pensando RoCE HCA driver"
|
|
#define DEVICE_DESCRIPTION "AMD Pensando RoCE HCA"
|
|
|
|
MODULE_AUTHOR("Allen Hubbe <allen.hubbe@amd.com>");
|
|
MODULE_DESCRIPTION(DRIVER_DESCRIPTION);
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_IMPORT_NS("NET_IONIC");
|
|
|
|
static int ionic_query_device(struct ib_device *ibdev,
|
|
struct ib_device_attr *attr,
|
|
struct ib_udata *udata)
|
|
{
|
|
struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
|
|
struct net_device *ndev;
|
|
|
|
ndev = ib_device_get_netdev(ibdev, 1);
|
|
addrconf_ifid_eui48((u8 *)&attr->sys_image_guid, ndev);
|
|
dev_put(ndev);
|
|
attr->max_mr_size = dev->lif_cfg.npts_per_lif * PAGE_SIZE / 2;
|
|
attr->page_size_cap = dev->lif_cfg.page_size_supported;
|
|
|
|
attr->vendor_id = to_pci_dev(dev->lif_cfg.hwdev)->vendor;
|
|
attr->vendor_part_id = to_pci_dev(dev->lif_cfg.hwdev)->device;
|
|
|
|
attr->hw_ver = ionic_lif_asic_rev(dev->lif_cfg.lif);
|
|
attr->fw_ver = 0;
|
|
attr->max_qp = dev->lif_cfg.qp_count;
|
|
attr->max_qp_wr = IONIC_MAX_DEPTH;
|
|
attr->device_cap_flags =
|
|
IB_DEVICE_MEM_WINDOW |
|
|
IB_DEVICE_MEM_MGT_EXTENSIONS |
|
|
IB_DEVICE_MEM_WINDOW_TYPE_2B |
|
|
0;
|
|
attr->max_send_sge =
|
|
min(ionic_v1_send_wqe_max_sge(dev->lif_cfg.max_stride, 0, false),
|
|
IONIC_SPEC_HIGH);
|
|
attr->max_recv_sge =
|
|
min(ionic_v1_recv_wqe_max_sge(dev->lif_cfg.max_stride, 0, false),
|
|
IONIC_SPEC_HIGH);
|
|
attr->max_sge_rd = attr->max_send_sge;
|
|
attr->max_cq = dev->lif_cfg.cq_count / dev->lif_cfg.udma_count;
|
|
attr->max_cqe = IONIC_MAX_CQ_DEPTH - IONIC_CQ_GRACE;
|
|
attr->max_mr = dev->lif_cfg.nmrs_per_lif;
|
|
attr->max_pd = IONIC_MAX_PD;
|
|
attr->max_qp_rd_atom = IONIC_MAX_RD_ATOM;
|
|
attr->max_ee_rd_atom = 0;
|
|
attr->max_res_rd_atom = IONIC_MAX_RD_ATOM;
|
|
attr->max_qp_init_rd_atom = IONIC_MAX_RD_ATOM;
|
|
attr->max_ee_init_rd_atom = 0;
|
|
attr->atomic_cap = IB_ATOMIC_GLOB;
|
|
attr->masked_atomic_cap = IB_ATOMIC_GLOB;
|
|
attr->max_mw = dev->lif_cfg.nmrs_per_lif;
|
|
attr->max_mcast_grp = 0;
|
|
attr->max_mcast_qp_attach = 0;
|
|
attr->max_ah = dev->lif_cfg.nahs_per_lif;
|
|
attr->max_fast_reg_page_list_len = dev->lif_cfg.npts_per_lif / 2;
|
|
attr->max_pkeys = IONIC_PKEY_TBL_LEN;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ionic_query_port(struct ib_device *ibdev, u32 port,
|
|
struct ib_port_attr *attr)
|
|
{
|
|
struct net_device *ndev;
|
|
|
|
if (port != 1)
|
|
return -EINVAL;
|
|
|
|
ndev = ib_device_get_netdev(ibdev, port);
|
|
|
|
if (netif_running(ndev) && netif_carrier_ok(ndev)) {
|
|
attr->state = IB_PORT_ACTIVE;
|
|
attr->phys_state = IB_PORT_PHYS_STATE_LINK_UP;
|
|
} else if (netif_running(ndev)) {
|
|
attr->state = IB_PORT_DOWN;
|
|
attr->phys_state = IB_PORT_PHYS_STATE_POLLING;
|
|
} else {
|
|
attr->state = IB_PORT_DOWN;
|
|
attr->phys_state = IB_PORT_PHYS_STATE_DISABLED;
|
|
}
|
|
|
|
attr->max_mtu = iboe_get_mtu(ndev->max_mtu);
|
|
attr->active_mtu = min(attr->max_mtu, iboe_get_mtu(ndev->mtu));
|
|
attr->gid_tbl_len = IONIC_GID_TBL_LEN;
|
|
attr->ip_gids = true;
|
|
attr->port_cap_flags = 0;
|
|
attr->max_msg_sz = 0x80000000;
|
|
attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN;
|
|
attr->max_vl_num = 1;
|
|
attr->subnet_prefix = 0xfe80000000000000ull;
|
|
|
|
dev_put(ndev);
|
|
|
|
return ib_get_eth_speed(ibdev, port,
|
|
&attr->active_speed,
|
|
&attr->active_width);
|
|
}
|
|
|
|
static enum rdma_link_layer ionic_get_link_layer(struct ib_device *ibdev,
|
|
u32 port)
|
|
{
|
|
return IB_LINK_LAYER_ETHERNET;
|
|
}
|
|
|
|
static int ionic_query_pkey(struct ib_device *ibdev, u32 port, u16 index,
|
|
u16 *pkey)
|
|
{
|
|
if (port != 1)
|
|
return -EINVAL;
|
|
|
|
if (index != 0)
|
|
return -EINVAL;
|
|
|
|
*pkey = IB_DEFAULT_PKEY_FULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ionic_modify_device(struct ib_device *ibdev, int mask,
|
|
struct ib_device_modify *attr)
|
|
{
|
|
struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
|
|
|
|
if (mask & ~IB_DEVICE_MODIFY_NODE_DESC)
|
|
return -EOPNOTSUPP;
|
|
|
|
if (mask & IB_DEVICE_MODIFY_NODE_DESC)
|
|
memcpy(dev->ibdev.node_desc, attr->node_desc,
|
|
IB_DEVICE_NODE_DESC_MAX);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int ionic_get_port_immutable(struct ib_device *ibdev, u32 port,
|
|
struct ib_port_immutable *attr)
|
|
{
|
|
if (port != 1)
|
|
return -EINVAL;
|
|
|
|
attr->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE_UDP_ENCAP;
|
|
|
|
attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN;
|
|
attr->gid_tbl_len = IONIC_GID_TBL_LEN;
|
|
attr->max_mad_size = IB_MGMT_MAD_SIZE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void ionic_get_dev_fw_str(struct ib_device *ibdev, char *str)
|
|
{
|
|
struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
|
|
|
|
ionic_lif_fw_version(dev->lif_cfg.lif, str, IB_FW_VERSION_NAME_MAX);
|
|
}
|
|
|
|
static ssize_t hw_rev_show(struct device *device, struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
struct ionic_ibdev *dev =
|
|
rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev);
|
|
|
|
return sysfs_emit(buf, "0x%x\n", ionic_lif_asic_rev(dev->lif_cfg.lif));
|
|
}
|
|
static DEVICE_ATTR_RO(hw_rev);
|
|
|
|
static ssize_t hca_type_show(struct device *device,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
struct ionic_ibdev *dev =
|
|
rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev);
|
|
|
|
return sysfs_emit(buf, "%s\n", dev->ibdev.node_desc);
|
|
}
|
|
static DEVICE_ATTR_RO(hca_type);
|
|
|
|
static struct attribute *ionic_rdma_attributes[] = {
|
|
&dev_attr_hw_rev.attr,
|
|
&dev_attr_hca_type.attr,
|
|
NULL
|
|
};
|
|
|
|
static const struct attribute_group ionic_rdma_attr_group = {
|
|
.attrs = ionic_rdma_attributes,
|
|
};
|
|
|
|
static void ionic_disassociate_ucontext(struct ib_ucontext *ibcontext)
|
|
{
|
|
/*
|
|
* Dummy define disassociate_ucontext so that it does not
|
|
* wait for user context before cleaning up hw resources.
|
|
*/
|
|
}
|
|
|
|
static const struct ib_device_ops ionic_dev_ops = {
|
|
.owner = THIS_MODULE,
|
|
.driver_id = RDMA_DRIVER_IONIC,
|
|
.uverbs_abi_ver = IONIC_ABI_VERSION,
|
|
|
|
.alloc_ucontext = ionic_alloc_ucontext,
|
|
.dealloc_ucontext = ionic_dealloc_ucontext,
|
|
.mmap = ionic_mmap,
|
|
.mmap_free = ionic_mmap_free,
|
|
.alloc_pd = ionic_alloc_pd,
|
|
.dealloc_pd = ionic_dealloc_pd,
|
|
.create_ah = ionic_create_ah,
|
|
.query_ah = ionic_query_ah,
|
|
.destroy_ah = ionic_destroy_ah,
|
|
.create_user_ah = ionic_create_ah,
|
|
.get_dma_mr = ionic_get_dma_mr,
|
|
.reg_user_mr = ionic_reg_user_mr,
|
|
.reg_user_mr_dmabuf = ionic_reg_user_mr_dmabuf,
|
|
.dereg_mr = ionic_dereg_mr,
|
|
.alloc_mr = ionic_alloc_mr,
|
|
.map_mr_sg = ionic_map_mr_sg,
|
|
.alloc_mw = ionic_alloc_mw,
|
|
.dealloc_mw = ionic_dealloc_mw,
|
|
.create_cq = ionic_create_cq,
|
|
.destroy_cq = ionic_destroy_cq,
|
|
.create_qp = ionic_create_qp,
|
|
.modify_qp = ionic_modify_qp,
|
|
.query_qp = ionic_query_qp,
|
|
.destroy_qp = ionic_destroy_qp,
|
|
|
|
.post_send = ionic_post_send,
|
|
.post_recv = ionic_post_recv,
|
|
.poll_cq = ionic_poll_cq,
|
|
.req_notify_cq = ionic_req_notify_cq,
|
|
|
|
.query_device = ionic_query_device,
|
|
.query_port = ionic_query_port,
|
|
.get_link_layer = ionic_get_link_layer,
|
|
.query_pkey = ionic_query_pkey,
|
|
.modify_device = ionic_modify_device,
|
|
.get_port_immutable = ionic_get_port_immutable,
|
|
.get_dev_fw_str = ionic_get_dev_fw_str,
|
|
.device_group = &ionic_rdma_attr_group,
|
|
.disassociate_ucontext = ionic_disassociate_ucontext,
|
|
|
|
INIT_RDMA_OBJ_SIZE(ib_ucontext, ionic_ctx, ibctx),
|
|
INIT_RDMA_OBJ_SIZE(ib_pd, ionic_pd, ibpd),
|
|
INIT_RDMA_OBJ_SIZE(ib_ah, ionic_ah, ibah),
|
|
INIT_RDMA_OBJ_SIZE(ib_cq, ionic_vcq, ibcq),
|
|
INIT_RDMA_OBJ_SIZE(ib_qp, ionic_qp, ibqp),
|
|
INIT_RDMA_OBJ_SIZE(ib_mw, ionic_mr, ibmw),
|
|
};
|
|
|
|
static void ionic_init_resids(struct ionic_ibdev *dev)
|
|
{
|
|
ionic_resid_init(&dev->inuse_cqid, dev->lif_cfg.cq_count);
|
|
dev->half_cqid_udma_shift =
|
|
order_base_2(dev->lif_cfg.cq_count / dev->lif_cfg.udma_count);
|
|
ionic_resid_init(&dev->inuse_pdid, IONIC_MAX_PD);
|
|
ionic_resid_init(&dev->inuse_ahid, dev->lif_cfg.nahs_per_lif);
|
|
ionic_resid_init(&dev->inuse_mrid, dev->lif_cfg.nmrs_per_lif);
|
|
/* skip reserved lkey */
|
|
dev->next_mrkey = 1;
|
|
ionic_resid_init(&dev->inuse_qpid, dev->lif_cfg.qp_count);
|
|
/* skip reserved SMI and GSI qpids */
|
|
dev->half_qpid_udma_shift =
|
|
order_base_2(dev->lif_cfg.qp_count / dev->lif_cfg.udma_count);
|
|
ionic_resid_init(&dev->inuse_dbid, dev->lif_cfg.dbid_count);
|
|
}
|
|
|
|
static void ionic_destroy_resids(struct ionic_ibdev *dev)
|
|
{
|
|
ionic_resid_destroy(&dev->inuse_cqid);
|
|
ionic_resid_destroy(&dev->inuse_pdid);
|
|
ionic_resid_destroy(&dev->inuse_ahid);
|
|
ionic_resid_destroy(&dev->inuse_mrid);
|
|
ionic_resid_destroy(&dev->inuse_qpid);
|
|
ionic_resid_destroy(&dev->inuse_dbid);
|
|
}
|
|
|
|
static void ionic_destroy_ibdev(struct ionic_ibdev *dev)
|
|
{
|
|
ionic_kill_rdma_admin(dev, false);
|
|
ib_unregister_device(&dev->ibdev);
|
|
ionic_stats_cleanup(dev);
|
|
ionic_destroy_rdma_admin(dev);
|
|
ionic_destroy_resids(dev);
|
|
WARN_ON(!xa_empty(&dev->qp_tbl));
|
|
xa_destroy(&dev->qp_tbl);
|
|
WARN_ON(!xa_empty(&dev->cq_tbl));
|
|
xa_destroy(&dev->cq_tbl);
|
|
ib_dealloc_device(&dev->ibdev);
|
|
}
|
|
|
|
static struct ionic_ibdev *ionic_create_ibdev(struct ionic_aux_dev *ionic_adev)
|
|
{
|
|
struct ib_device *ibdev;
|
|
struct ionic_ibdev *dev;
|
|
struct net_device *ndev;
|
|
int rc;
|
|
|
|
dev = ib_alloc_device(ionic_ibdev, ibdev);
|
|
if (!dev)
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
ionic_fill_lif_cfg(ionic_adev->lif, &dev->lif_cfg);
|
|
|
|
xa_init_flags(&dev->qp_tbl, GFP_ATOMIC);
|
|
xa_init_flags(&dev->cq_tbl, GFP_ATOMIC);
|
|
|
|
ionic_init_resids(dev);
|
|
|
|
rc = ionic_rdma_reset_devcmd(dev);
|
|
if (rc)
|
|
goto err_reset;
|
|
|
|
rc = ionic_create_rdma_admin(dev);
|
|
if (rc)
|
|
goto err_admin;
|
|
|
|
ibdev = &dev->ibdev;
|
|
ibdev->dev.parent = dev->lif_cfg.hwdev;
|
|
|
|
strscpy(ibdev->name, "ionic_%d", IB_DEVICE_NAME_MAX);
|
|
strscpy(ibdev->node_desc, DEVICE_DESCRIPTION, IB_DEVICE_NODE_DESC_MAX);
|
|
|
|
ibdev->node_type = RDMA_NODE_IB_CA;
|
|
ibdev->phys_port_cnt = 1;
|
|
|
|
/* the first two eq are reserved for async events */
|
|
ibdev->num_comp_vectors = dev->lif_cfg.eq_count - 2;
|
|
|
|
ndev = ionic_lif_netdev(ionic_adev->lif);
|
|
addrconf_ifid_eui48((u8 *)&ibdev->node_guid, ndev);
|
|
rc = ib_device_set_netdev(ibdev, ndev, 1);
|
|
/* ionic_lif_netdev() returns ndev with refcount held */
|
|
dev_put(ndev);
|
|
if (rc)
|
|
goto err_admin;
|
|
|
|
ib_set_device_ops(&dev->ibdev, &ionic_dev_ops);
|
|
|
|
ionic_stats_init(dev);
|
|
|
|
rc = ib_register_device(ibdev, "ionic_%d", ibdev->dev.parent);
|
|
if (rc)
|
|
goto err_register;
|
|
|
|
return dev;
|
|
|
|
err_register:
|
|
ionic_stats_cleanup(dev);
|
|
err_admin:
|
|
ionic_kill_rdma_admin(dev, false);
|
|
ionic_destroy_rdma_admin(dev);
|
|
err_reset:
|
|
ionic_destroy_resids(dev);
|
|
xa_destroy(&dev->qp_tbl);
|
|
xa_destroy(&dev->cq_tbl);
|
|
ib_dealloc_device(&dev->ibdev);
|
|
|
|
return ERR_PTR(rc);
|
|
}
|
|
|
|
static int ionic_aux_probe(struct auxiliary_device *adev,
|
|
const struct auxiliary_device_id *id)
|
|
{
|
|
struct ionic_aux_dev *ionic_adev;
|
|
struct ionic_ibdev *dev;
|
|
|
|
ionic_adev = container_of(adev, struct ionic_aux_dev, adev);
|
|
dev = ionic_create_ibdev(ionic_adev);
|
|
if (IS_ERR(dev))
|
|
return dev_err_probe(&adev->dev, PTR_ERR(dev),
|
|
"Failed to register ibdev\n");
|
|
|
|
auxiliary_set_drvdata(adev, dev);
|
|
ibdev_dbg(&dev->ibdev, "registered\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void ionic_aux_remove(struct auxiliary_device *adev)
|
|
{
|
|
struct ionic_ibdev *dev = auxiliary_get_drvdata(adev);
|
|
|
|
dev_dbg(&adev->dev, "unregister ibdev\n");
|
|
ionic_destroy_ibdev(dev);
|
|
dev_dbg(&adev->dev, "unregistered\n");
|
|
}
|
|
|
|
static const struct auxiliary_device_id ionic_aux_id_table[] = {
|
|
{ .name = "ionic.rdma", },
|
|
{},
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(auxiliary, ionic_aux_id_table);
|
|
|
|
static struct auxiliary_driver ionic_aux_r_driver = {
|
|
.name = "rdma",
|
|
.probe = ionic_aux_probe,
|
|
.remove = ionic_aux_remove,
|
|
.id_table = ionic_aux_id_table,
|
|
};
|
|
|
|
static int __init ionic_mod_init(void)
|
|
{
|
|
int rc;
|
|
|
|
ionic_evt_workq = create_workqueue(KBUILD_MODNAME "-evt");
|
|
if (!ionic_evt_workq)
|
|
return -ENOMEM;
|
|
|
|
rc = auxiliary_driver_register(&ionic_aux_r_driver);
|
|
if (rc)
|
|
goto err_aux;
|
|
|
|
return 0;
|
|
|
|
err_aux:
|
|
destroy_workqueue(ionic_evt_workq);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void __exit ionic_mod_exit(void)
|
|
{
|
|
auxiliary_driver_unregister(&ionic_aux_r_driver);
|
|
destroy_workqueue(ionic_evt_workq);
|
|
}
|
|
|
|
module_init(ionic_mod_init);
|
|
module_exit(ionic_mod_exit);
|