diff options
| author | Takashi Iwai <tiwai@suse.de> | 2025-10-16 20:14:24 +0200 | 
|---|---|---|
| committer | Takashi Iwai <tiwai@suse.de> | 2025-10-16 20:14:24 +0200 | 
| commit | ec2e0fb07d789976c601bec19ecced7a501c3705 (patch) | |
| tree | d593e7b6153618ca10855a3141404e952debe73c /drivers/infiniband/hw/ionic/ionic_ibdev.c | |
| parent | c6fceaf166479c05f7d3158ef08e78ae3e3dfa23 (diff) | |
| parent | f1a450f9e17d341f69f8fb19f6d13ef9f1aa508b (diff) | |
Merge tag 'asoc-fix-v6.18-rc1' of https://git.kernel.org/pub/scm/linux/kernel/git/broonie/sound into for-linus
ASoC: Fixes for v6.18
A moderately large collection of driver specific fixes, plus a few new
quirks and device IDs.  The NAU8821 changes are a little large but more
in mechanical ways than in ways that are complex.
Diffstat (limited to 'drivers/infiniband/hw/ionic/ionic_ibdev.c')
| -rw-r--r-- | drivers/infiniband/hw/ionic/ionic_ibdev.c | 440 | 
1 files changed, 440 insertions, 0 deletions
| diff --git a/drivers/infiniband/hw/ionic/ionic_ibdev.c b/drivers/infiniband/hw/ionic/ionic_ibdev.c new file mode 100644 index 000000000000..164046d00e5d --- /dev/null +++ b/drivers/infiniband/hw/ionic/ionic_ibdev.c @@ -0,0 +1,440 @@ +// 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); | 
