18d765af5SAbhijit Gangurde // SPDX-License-Identifier: GPL-2.0
28d765af5SAbhijit Gangurde /* Copyright (C) 2018-2025, Advanced Micro Devices, Inc. */
38d765af5SAbhijit Gangurde
48d765af5SAbhijit Gangurde #include <linux/module.h>
58d765af5SAbhijit Gangurde #include <linux/printk.h>
62075bbe8SAbhijit Gangurde #include <linux/pci.h>
72075bbe8SAbhijit Gangurde #include <linux/irq.h>
88d765af5SAbhijit Gangurde #include <net/addrconf.h>
92075bbe8SAbhijit Gangurde #include <rdma/ib_addr.h>
102075bbe8SAbhijit Gangurde #include <rdma/ib_mad.h>
118d765af5SAbhijit Gangurde
128d765af5SAbhijit Gangurde #include "ionic_ibdev.h"
138d765af5SAbhijit Gangurde
148d765af5SAbhijit Gangurde #define DRIVER_DESCRIPTION "AMD Pensando RoCE HCA driver"
158d765af5SAbhijit Gangurde #define DEVICE_DESCRIPTION "AMD Pensando RoCE HCA"
168d765af5SAbhijit Gangurde
178d765af5SAbhijit Gangurde MODULE_AUTHOR("Allen Hubbe <allen.hubbe@amd.com>");
188d765af5SAbhijit Gangurde MODULE_DESCRIPTION(DRIVER_DESCRIPTION);
198d765af5SAbhijit Gangurde MODULE_LICENSE("GPL");
208d765af5SAbhijit Gangurde MODULE_IMPORT_NS("NET_IONIC");
218d765af5SAbhijit Gangurde
ionic_query_device(struct ib_device * ibdev,struct ib_device_attr * attr,struct ib_udata * udata)222075bbe8SAbhijit Gangurde static int ionic_query_device(struct ib_device *ibdev,
232075bbe8SAbhijit Gangurde struct ib_device_attr *attr,
242075bbe8SAbhijit Gangurde struct ib_udata *udata)
252075bbe8SAbhijit Gangurde {
262075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
272075bbe8SAbhijit Gangurde struct net_device *ndev;
282075bbe8SAbhijit Gangurde
292075bbe8SAbhijit Gangurde ndev = ib_device_get_netdev(ibdev, 1);
302075bbe8SAbhijit Gangurde addrconf_ifid_eui48((u8 *)&attr->sys_image_guid, ndev);
312075bbe8SAbhijit Gangurde dev_put(ndev);
322075bbe8SAbhijit Gangurde attr->max_mr_size = dev->lif_cfg.npts_per_lif * PAGE_SIZE / 2;
332075bbe8SAbhijit Gangurde attr->page_size_cap = dev->lif_cfg.page_size_supported;
342075bbe8SAbhijit Gangurde
352075bbe8SAbhijit Gangurde attr->vendor_id = to_pci_dev(dev->lif_cfg.hwdev)->vendor;
362075bbe8SAbhijit Gangurde attr->vendor_part_id = to_pci_dev(dev->lif_cfg.hwdev)->device;
372075bbe8SAbhijit Gangurde
382075bbe8SAbhijit Gangurde attr->hw_ver = ionic_lif_asic_rev(dev->lif_cfg.lif);
392075bbe8SAbhijit Gangurde attr->fw_ver = 0;
402075bbe8SAbhijit Gangurde attr->max_qp = dev->lif_cfg.qp_count;
412075bbe8SAbhijit Gangurde attr->max_qp_wr = IONIC_MAX_DEPTH;
422075bbe8SAbhijit Gangurde attr->device_cap_flags =
432075bbe8SAbhijit Gangurde IB_DEVICE_MEM_WINDOW |
442075bbe8SAbhijit Gangurde IB_DEVICE_MEM_MGT_EXTENSIONS |
452075bbe8SAbhijit Gangurde IB_DEVICE_MEM_WINDOW_TYPE_2B |
462075bbe8SAbhijit Gangurde 0;
472075bbe8SAbhijit Gangurde attr->max_send_sge =
482075bbe8SAbhijit Gangurde min(ionic_v1_send_wqe_max_sge(dev->lif_cfg.max_stride, 0, false),
492075bbe8SAbhijit Gangurde IONIC_SPEC_HIGH);
502075bbe8SAbhijit Gangurde attr->max_recv_sge =
512075bbe8SAbhijit Gangurde min(ionic_v1_recv_wqe_max_sge(dev->lif_cfg.max_stride, 0, false),
522075bbe8SAbhijit Gangurde IONIC_SPEC_HIGH);
532075bbe8SAbhijit Gangurde attr->max_sge_rd = attr->max_send_sge;
542075bbe8SAbhijit Gangurde attr->max_cq = dev->lif_cfg.cq_count / dev->lif_cfg.udma_count;
552075bbe8SAbhijit Gangurde attr->max_cqe = IONIC_MAX_CQ_DEPTH - IONIC_CQ_GRACE;
562075bbe8SAbhijit Gangurde attr->max_mr = dev->lif_cfg.nmrs_per_lif;
572075bbe8SAbhijit Gangurde attr->max_pd = IONIC_MAX_PD;
582075bbe8SAbhijit Gangurde attr->max_qp_rd_atom = IONIC_MAX_RD_ATOM;
592075bbe8SAbhijit Gangurde attr->max_ee_rd_atom = 0;
602075bbe8SAbhijit Gangurde attr->max_res_rd_atom = IONIC_MAX_RD_ATOM;
612075bbe8SAbhijit Gangurde attr->max_qp_init_rd_atom = IONIC_MAX_RD_ATOM;
622075bbe8SAbhijit Gangurde attr->max_ee_init_rd_atom = 0;
632075bbe8SAbhijit Gangurde attr->atomic_cap = IB_ATOMIC_GLOB;
642075bbe8SAbhijit Gangurde attr->masked_atomic_cap = IB_ATOMIC_GLOB;
652075bbe8SAbhijit Gangurde attr->max_mw = dev->lif_cfg.nmrs_per_lif;
662075bbe8SAbhijit Gangurde attr->max_mcast_grp = 0;
672075bbe8SAbhijit Gangurde attr->max_mcast_qp_attach = 0;
682075bbe8SAbhijit Gangurde attr->max_ah = dev->lif_cfg.nahs_per_lif;
692075bbe8SAbhijit Gangurde attr->max_fast_reg_page_list_len = dev->lif_cfg.npts_per_lif / 2;
702075bbe8SAbhijit Gangurde attr->max_pkeys = IONIC_PKEY_TBL_LEN;
712075bbe8SAbhijit Gangurde
722075bbe8SAbhijit Gangurde return 0;
732075bbe8SAbhijit Gangurde }
742075bbe8SAbhijit Gangurde
ionic_query_port(struct ib_device * ibdev,u32 port,struct ib_port_attr * attr)752075bbe8SAbhijit Gangurde static int ionic_query_port(struct ib_device *ibdev, u32 port,
762075bbe8SAbhijit Gangurde struct ib_port_attr *attr)
772075bbe8SAbhijit Gangurde {
782075bbe8SAbhijit Gangurde struct net_device *ndev;
792075bbe8SAbhijit Gangurde
802075bbe8SAbhijit Gangurde if (port != 1)
812075bbe8SAbhijit Gangurde return -EINVAL;
822075bbe8SAbhijit Gangurde
832075bbe8SAbhijit Gangurde ndev = ib_device_get_netdev(ibdev, port);
84*fd80bd71SKamal Heib if (!ndev)
85*fd80bd71SKamal Heib return -ENODEV;
862075bbe8SAbhijit Gangurde
872075bbe8SAbhijit Gangurde if (netif_running(ndev) && netif_carrier_ok(ndev)) {
882075bbe8SAbhijit Gangurde attr->state = IB_PORT_ACTIVE;
892075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_LINK_UP;
902075bbe8SAbhijit Gangurde } else if (netif_running(ndev)) {
912075bbe8SAbhijit Gangurde attr->state = IB_PORT_DOWN;
922075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_POLLING;
932075bbe8SAbhijit Gangurde } else {
942075bbe8SAbhijit Gangurde attr->state = IB_PORT_DOWN;
952075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_DISABLED;
962075bbe8SAbhijit Gangurde }
972075bbe8SAbhijit Gangurde
982075bbe8SAbhijit Gangurde attr->max_mtu = iboe_get_mtu(ndev->max_mtu);
992075bbe8SAbhijit Gangurde attr->active_mtu = min(attr->max_mtu, iboe_get_mtu(ndev->mtu));
1002075bbe8SAbhijit Gangurde attr->gid_tbl_len = IONIC_GID_TBL_LEN;
1012075bbe8SAbhijit Gangurde attr->ip_gids = true;
1022075bbe8SAbhijit Gangurde attr->port_cap_flags = 0;
1032075bbe8SAbhijit Gangurde attr->max_msg_sz = 0x80000000;
1042075bbe8SAbhijit Gangurde attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN;
1052075bbe8SAbhijit Gangurde attr->max_vl_num = 1;
1062075bbe8SAbhijit Gangurde attr->subnet_prefix = 0xfe80000000000000ull;
1072075bbe8SAbhijit Gangurde
1082075bbe8SAbhijit Gangurde dev_put(ndev);
1092075bbe8SAbhijit Gangurde
1102075bbe8SAbhijit Gangurde return ib_get_eth_speed(ibdev, port,
1112075bbe8SAbhijit Gangurde &attr->active_speed,
1122075bbe8SAbhijit Gangurde &attr->active_width);
1132075bbe8SAbhijit Gangurde }
1142075bbe8SAbhijit Gangurde
ionic_get_link_layer(struct ib_device * ibdev,u32 port)1152075bbe8SAbhijit Gangurde static enum rdma_link_layer ionic_get_link_layer(struct ib_device *ibdev,
1162075bbe8SAbhijit Gangurde u32 port)
1172075bbe8SAbhijit Gangurde {
1182075bbe8SAbhijit Gangurde return IB_LINK_LAYER_ETHERNET;
1192075bbe8SAbhijit Gangurde }
1202075bbe8SAbhijit Gangurde
ionic_query_pkey(struct ib_device * ibdev,u32 port,u16 index,u16 * pkey)1212075bbe8SAbhijit Gangurde static int ionic_query_pkey(struct ib_device *ibdev, u32 port, u16 index,
1222075bbe8SAbhijit Gangurde u16 *pkey)
1232075bbe8SAbhijit Gangurde {
1242075bbe8SAbhijit Gangurde if (port != 1)
1252075bbe8SAbhijit Gangurde return -EINVAL;
1262075bbe8SAbhijit Gangurde
1272075bbe8SAbhijit Gangurde if (index != 0)
1282075bbe8SAbhijit Gangurde return -EINVAL;
1292075bbe8SAbhijit Gangurde
1302075bbe8SAbhijit Gangurde *pkey = IB_DEFAULT_PKEY_FULL;
1312075bbe8SAbhijit Gangurde
1322075bbe8SAbhijit Gangurde return 0;
1332075bbe8SAbhijit Gangurde }
1342075bbe8SAbhijit Gangurde
ionic_modify_device(struct ib_device * ibdev,int mask,struct ib_device_modify * attr)1352075bbe8SAbhijit Gangurde static int ionic_modify_device(struct ib_device *ibdev, int mask,
1362075bbe8SAbhijit Gangurde struct ib_device_modify *attr)
1372075bbe8SAbhijit Gangurde {
1382075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
1392075bbe8SAbhijit Gangurde
1402075bbe8SAbhijit Gangurde if (mask & ~IB_DEVICE_MODIFY_NODE_DESC)
1412075bbe8SAbhijit Gangurde return -EOPNOTSUPP;
1422075bbe8SAbhijit Gangurde
1432075bbe8SAbhijit Gangurde if (mask & IB_DEVICE_MODIFY_NODE_DESC)
1442075bbe8SAbhijit Gangurde memcpy(dev->ibdev.node_desc, attr->node_desc,
1452075bbe8SAbhijit Gangurde IB_DEVICE_NODE_DESC_MAX);
1462075bbe8SAbhijit Gangurde
1472075bbe8SAbhijit Gangurde return 0;
1482075bbe8SAbhijit Gangurde }
1492075bbe8SAbhijit Gangurde
ionic_get_port_immutable(struct ib_device * ibdev,u32 port,struct ib_port_immutable * attr)1502075bbe8SAbhijit Gangurde static int ionic_get_port_immutable(struct ib_device *ibdev, u32 port,
1512075bbe8SAbhijit Gangurde struct ib_port_immutable *attr)
1522075bbe8SAbhijit Gangurde {
1532075bbe8SAbhijit Gangurde if (port != 1)
1542075bbe8SAbhijit Gangurde return -EINVAL;
1552075bbe8SAbhijit Gangurde
1562075bbe8SAbhijit Gangurde attr->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE_UDP_ENCAP;
1572075bbe8SAbhijit Gangurde
1582075bbe8SAbhijit Gangurde attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN;
1592075bbe8SAbhijit Gangurde attr->gid_tbl_len = IONIC_GID_TBL_LEN;
1602075bbe8SAbhijit Gangurde attr->max_mad_size = IB_MGMT_MAD_SIZE;
1612075bbe8SAbhijit Gangurde
1622075bbe8SAbhijit Gangurde return 0;
1632075bbe8SAbhijit Gangurde }
1642075bbe8SAbhijit Gangurde
ionic_get_dev_fw_str(struct ib_device * ibdev,char * str)1652075bbe8SAbhijit Gangurde static void ionic_get_dev_fw_str(struct ib_device *ibdev, char *str)
1662075bbe8SAbhijit Gangurde {
1672075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev);
1682075bbe8SAbhijit Gangurde
1692075bbe8SAbhijit Gangurde ionic_lif_fw_version(dev->lif_cfg.lif, str, IB_FW_VERSION_NAME_MAX);
1702075bbe8SAbhijit Gangurde }
1712075bbe8SAbhijit Gangurde
hw_rev_show(struct device * device,struct device_attribute * attr,char * buf)1722075bbe8SAbhijit Gangurde static ssize_t hw_rev_show(struct device *device, struct device_attribute *attr,
1732075bbe8SAbhijit Gangurde char *buf)
1742075bbe8SAbhijit Gangurde {
1752075bbe8SAbhijit Gangurde struct ionic_ibdev *dev =
1762075bbe8SAbhijit Gangurde rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev);
1772075bbe8SAbhijit Gangurde
1782075bbe8SAbhijit Gangurde return sysfs_emit(buf, "0x%x\n", ionic_lif_asic_rev(dev->lif_cfg.lif));
1792075bbe8SAbhijit Gangurde }
1802075bbe8SAbhijit Gangurde static DEVICE_ATTR_RO(hw_rev);
1812075bbe8SAbhijit Gangurde
hca_type_show(struct device * device,struct device_attribute * attr,char * buf)1822075bbe8SAbhijit Gangurde static ssize_t hca_type_show(struct device *device,
1832075bbe8SAbhijit Gangurde struct device_attribute *attr, char *buf)
1842075bbe8SAbhijit Gangurde {
1852075bbe8SAbhijit Gangurde struct ionic_ibdev *dev =
1862075bbe8SAbhijit Gangurde rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev);
1872075bbe8SAbhijit Gangurde
1882075bbe8SAbhijit Gangurde return sysfs_emit(buf, "%s\n", dev->ibdev.node_desc);
1892075bbe8SAbhijit Gangurde }
1902075bbe8SAbhijit Gangurde static DEVICE_ATTR_RO(hca_type);
1912075bbe8SAbhijit Gangurde
1922075bbe8SAbhijit Gangurde static struct attribute *ionic_rdma_attributes[] = {
1932075bbe8SAbhijit Gangurde &dev_attr_hw_rev.attr,
1942075bbe8SAbhijit Gangurde &dev_attr_hca_type.attr,
1952075bbe8SAbhijit Gangurde NULL
1962075bbe8SAbhijit Gangurde };
1972075bbe8SAbhijit Gangurde
1982075bbe8SAbhijit Gangurde static const struct attribute_group ionic_rdma_attr_group = {
1992075bbe8SAbhijit Gangurde .attrs = ionic_rdma_attributes,
2002075bbe8SAbhijit Gangurde };
2012075bbe8SAbhijit Gangurde
ionic_disassociate_ucontext(struct ib_ucontext * ibcontext)2022075bbe8SAbhijit Gangurde static void ionic_disassociate_ucontext(struct ib_ucontext *ibcontext)
2032075bbe8SAbhijit Gangurde {
2042075bbe8SAbhijit Gangurde /*
2052075bbe8SAbhijit Gangurde * Dummy define disassociate_ucontext so that it does not
2062075bbe8SAbhijit Gangurde * wait for user context before cleaning up hw resources.
2072075bbe8SAbhijit Gangurde */
2082075bbe8SAbhijit Gangurde }
2092075bbe8SAbhijit Gangurde
210e8521822SAbhijit Gangurde static const struct ib_device_ops ionic_dev_ops = {
211e8521822SAbhijit Gangurde .owner = THIS_MODULE,
212e8521822SAbhijit Gangurde .driver_id = RDMA_DRIVER_IONIC,
213e8521822SAbhijit Gangurde .uverbs_abi_ver = IONIC_ABI_VERSION,
214e8521822SAbhijit Gangurde
215e8521822SAbhijit Gangurde .alloc_ucontext = ionic_alloc_ucontext,
216e8521822SAbhijit Gangurde .dealloc_ucontext = ionic_dealloc_ucontext,
217e8521822SAbhijit Gangurde .mmap = ionic_mmap,
218e8521822SAbhijit Gangurde .mmap_free = ionic_mmap_free,
219e8521822SAbhijit Gangurde .alloc_pd = ionic_alloc_pd,
220e8521822SAbhijit Gangurde .dealloc_pd = ionic_dealloc_pd,
221e8521822SAbhijit Gangurde .create_ah = ionic_create_ah,
222e8521822SAbhijit Gangurde .query_ah = ionic_query_ah,
223e8521822SAbhijit Gangurde .destroy_ah = ionic_destroy_ah,
224e8521822SAbhijit Gangurde .create_user_ah = ionic_create_ah,
225e8521822SAbhijit Gangurde .get_dma_mr = ionic_get_dma_mr,
226e8521822SAbhijit Gangurde .reg_user_mr = ionic_reg_user_mr,
227e8521822SAbhijit Gangurde .reg_user_mr_dmabuf = ionic_reg_user_mr_dmabuf,
228e8521822SAbhijit Gangurde .dereg_mr = ionic_dereg_mr,
229e8521822SAbhijit Gangurde .alloc_mr = ionic_alloc_mr,
230e8521822SAbhijit Gangurde .map_mr_sg = ionic_map_mr_sg,
231e8521822SAbhijit Gangurde .alloc_mw = ionic_alloc_mw,
232e8521822SAbhijit Gangurde .dealloc_mw = ionic_dealloc_mw,
233e8521822SAbhijit Gangurde .create_cq = ionic_create_cq,
234e8521822SAbhijit Gangurde .destroy_cq = ionic_destroy_cq,
235e8521822SAbhijit Gangurde .create_qp = ionic_create_qp,
236e8521822SAbhijit Gangurde .modify_qp = ionic_modify_qp,
237e8521822SAbhijit Gangurde .query_qp = ionic_query_qp,
238e8521822SAbhijit Gangurde .destroy_qp = ionic_destroy_qp,
239e8521822SAbhijit Gangurde
240b83c6205SAbhijit Gangurde .post_send = ionic_post_send,
241b83c6205SAbhijit Gangurde .post_recv = ionic_post_recv,
242b83c6205SAbhijit Gangurde .poll_cq = ionic_poll_cq,
243b83c6205SAbhijit Gangurde .req_notify_cq = ionic_req_notify_cq,
244b83c6205SAbhijit Gangurde
2452075bbe8SAbhijit Gangurde .query_device = ionic_query_device,
2462075bbe8SAbhijit Gangurde .query_port = ionic_query_port,
2472075bbe8SAbhijit Gangurde .get_link_layer = ionic_get_link_layer,
2482075bbe8SAbhijit Gangurde .query_pkey = ionic_query_pkey,
2492075bbe8SAbhijit Gangurde .modify_device = ionic_modify_device,
2502075bbe8SAbhijit Gangurde .get_port_immutable = ionic_get_port_immutable,
2512075bbe8SAbhijit Gangurde .get_dev_fw_str = ionic_get_dev_fw_str,
2522075bbe8SAbhijit Gangurde .device_group = &ionic_rdma_attr_group,
2532075bbe8SAbhijit Gangurde .disassociate_ucontext = ionic_disassociate_ucontext,
2542075bbe8SAbhijit Gangurde
255e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_ucontext, ionic_ctx, ibctx),
256e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_pd, ionic_pd, ibpd),
257e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_ah, ionic_ah, ibah),
258e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_cq, ionic_vcq, ibcq),
259e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_qp, ionic_qp, ibqp),
260e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_mw, ionic_mr, ibmw),
261e8521822SAbhijit Gangurde };
262e8521822SAbhijit Gangurde
ionic_init_resids(struct ionic_ibdev * dev)263f3bdbd42SAbhijit Gangurde static void ionic_init_resids(struct ionic_ibdev *dev)
264f3bdbd42SAbhijit Gangurde {
265f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_cqid, dev->lif_cfg.cq_count);
266f3bdbd42SAbhijit Gangurde dev->half_cqid_udma_shift =
267f3bdbd42SAbhijit Gangurde order_base_2(dev->lif_cfg.cq_count / dev->lif_cfg.udma_count);
268f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_pdid, IONIC_MAX_PD);
269f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_ahid, dev->lif_cfg.nahs_per_lif);
270f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_mrid, dev->lif_cfg.nmrs_per_lif);
271f3bdbd42SAbhijit Gangurde /* skip reserved lkey */
272f3bdbd42SAbhijit Gangurde dev->next_mrkey = 1;
273f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_qpid, dev->lif_cfg.qp_count);
274f3bdbd42SAbhijit Gangurde /* skip reserved SMI and GSI qpids */
275f3bdbd42SAbhijit Gangurde dev->half_qpid_udma_shift =
276f3bdbd42SAbhijit Gangurde order_base_2(dev->lif_cfg.qp_count / dev->lif_cfg.udma_count);
277f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_dbid, dev->lif_cfg.dbid_count);
278f3bdbd42SAbhijit Gangurde }
279f3bdbd42SAbhijit Gangurde
ionic_destroy_resids(struct ionic_ibdev * dev)280f3bdbd42SAbhijit Gangurde static void ionic_destroy_resids(struct ionic_ibdev *dev)
281f3bdbd42SAbhijit Gangurde {
282f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_cqid);
283f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_pdid);
284f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_ahid);
285f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_mrid);
286f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_qpid);
287f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_dbid);
288f3bdbd42SAbhijit Gangurde }
289f3bdbd42SAbhijit Gangurde
ionic_destroy_ibdev(struct ionic_ibdev * dev)2908d765af5SAbhijit Gangurde static void ionic_destroy_ibdev(struct ionic_ibdev *dev)
2918d765af5SAbhijit Gangurde {
292f3bdbd42SAbhijit Gangurde ionic_kill_rdma_admin(dev, false);
2938d765af5SAbhijit Gangurde ib_unregister_device(&dev->ibdev);
294ea4c3996SAbhijit Gangurde ionic_stats_cleanup(dev);
295f3bdbd42SAbhijit Gangurde ionic_destroy_rdma_admin(dev);
296f3bdbd42SAbhijit Gangurde ionic_destroy_resids(dev);
297e8521822SAbhijit Gangurde WARN_ON(!xa_empty(&dev->qp_tbl));
298e8521822SAbhijit Gangurde xa_destroy(&dev->qp_tbl);
299f3bdbd42SAbhijit Gangurde WARN_ON(!xa_empty(&dev->cq_tbl));
300f3bdbd42SAbhijit Gangurde xa_destroy(&dev->cq_tbl);
3018d765af5SAbhijit Gangurde ib_dealloc_device(&dev->ibdev);
3028d765af5SAbhijit Gangurde }
3038d765af5SAbhijit Gangurde
ionic_create_ibdev(struct ionic_aux_dev * ionic_adev)3048d765af5SAbhijit Gangurde static struct ionic_ibdev *ionic_create_ibdev(struct ionic_aux_dev *ionic_adev)
3058d765af5SAbhijit Gangurde {
3068d765af5SAbhijit Gangurde struct ib_device *ibdev;
3078d765af5SAbhijit Gangurde struct ionic_ibdev *dev;
3088d765af5SAbhijit Gangurde struct net_device *ndev;
3098d765af5SAbhijit Gangurde int rc;
3108d765af5SAbhijit Gangurde
3118d765af5SAbhijit Gangurde dev = ib_alloc_device(ionic_ibdev, ibdev);
3128d765af5SAbhijit Gangurde if (!dev)
3138d765af5SAbhijit Gangurde return ERR_PTR(-EINVAL);
3148d765af5SAbhijit Gangurde
3158d765af5SAbhijit Gangurde ionic_fill_lif_cfg(ionic_adev->lif, &dev->lif_cfg);
3168d765af5SAbhijit Gangurde
317e8521822SAbhijit Gangurde xa_init_flags(&dev->qp_tbl, GFP_ATOMIC);
318f3bdbd42SAbhijit Gangurde xa_init_flags(&dev->cq_tbl, GFP_ATOMIC);
319f3bdbd42SAbhijit Gangurde
320f3bdbd42SAbhijit Gangurde ionic_init_resids(dev);
321f3bdbd42SAbhijit Gangurde
322f3bdbd42SAbhijit Gangurde rc = ionic_rdma_reset_devcmd(dev);
323f3bdbd42SAbhijit Gangurde if (rc)
324f3bdbd42SAbhijit Gangurde goto err_reset;
325f3bdbd42SAbhijit Gangurde
326f3bdbd42SAbhijit Gangurde rc = ionic_create_rdma_admin(dev);
327f3bdbd42SAbhijit Gangurde if (rc)
328f3bdbd42SAbhijit Gangurde goto err_admin;
329f3bdbd42SAbhijit Gangurde
3308d765af5SAbhijit Gangurde ibdev = &dev->ibdev;
3318d765af5SAbhijit Gangurde ibdev->dev.parent = dev->lif_cfg.hwdev;
3328d765af5SAbhijit Gangurde
3338d765af5SAbhijit Gangurde strscpy(ibdev->name, "ionic_%d", IB_DEVICE_NAME_MAX);
3348d765af5SAbhijit Gangurde strscpy(ibdev->node_desc, DEVICE_DESCRIPTION, IB_DEVICE_NODE_DESC_MAX);
3358d765af5SAbhijit Gangurde
3368d765af5SAbhijit Gangurde ibdev->node_type = RDMA_NODE_IB_CA;
3378d765af5SAbhijit Gangurde ibdev->phys_port_cnt = 1;
3388d765af5SAbhijit Gangurde
3398d765af5SAbhijit Gangurde /* the first two eq are reserved for async events */
3408d765af5SAbhijit Gangurde ibdev->num_comp_vectors = dev->lif_cfg.eq_count - 2;
3418d765af5SAbhijit Gangurde
3428d765af5SAbhijit Gangurde ndev = ionic_lif_netdev(ionic_adev->lif);
3438d765af5SAbhijit Gangurde addrconf_ifid_eui48((u8 *)&ibdev->node_guid, ndev);
3448d765af5SAbhijit Gangurde rc = ib_device_set_netdev(ibdev, ndev, 1);
3458d765af5SAbhijit Gangurde /* ionic_lif_netdev() returns ndev with refcount held */
3468d765af5SAbhijit Gangurde dev_put(ndev);
3478d765af5SAbhijit Gangurde if (rc)
3488d765af5SAbhijit Gangurde goto err_admin;
3498d765af5SAbhijit Gangurde
350e8521822SAbhijit Gangurde ib_set_device_ops(&dev->ibdev, &ionic_dev_ops);
351e8521822SAbhijit Gangurde
352ea4c3996SAbhijit Gangurde ionic_stats_init(dev);
353ea4c3996SAbhijit Gangurde
3548d765af5SAbhijit Gangurde rc = ib_register_device(ibdev, "ionic_%d", ibdev->dev.parent);
3558d765af5SAbhijit Gangurde if (rc)
3568d765af5SAbhijit Gangurde goto err_register;
3578d765af5SAbhijit Gangurde
3588d765af5SAbhijit Gangurde return dev;
3598d765af5SAbhijit Gangurde
3608d765af5SAbhijit Gangurde err_register:
361ea4c3996SAbhijit Gangurde ionic_stats_cleanup(dev);
3628d765af5SAbhijit Gangurde err_admin:
363f3bdbd42SAbhijit Gangurde ionic_kill_rdma_admin(dev, false);
364f3bdbd42SAbhijit Gangurde ionic_destroy_rdma_admin(dev);
365f3bdbd42SAbhijit Gangurde err_reset:
366f3bdbd42SAbhijit Gangurde ionic_destroy_resids(dev);
367e8521822SAbhijit Gangurde xa_destroy(&dev->qp_tbl);
368f3bdbd42SAbhijit Gangurde xa_destroy(&dev->cq_tbl);
3698d765af5SAbhijit Gangurde ib_dealloc_device(&dev->ibdev);
3708d765af5SAbhijit Gangurde
3718d765af5SAbhijit Gangurde return ERR_PTR(rc);
3728d765af5SAbhijit Gangurde }
3738d765af5SAbhijit Gangurde
ionic_aux_probe(struct auxiliary_device * adev,const struct auxiliary_device_id * id)3748d765af5SAbhijit Gangurde static int ionic_aux_probe(struct auxiliary_device *adev,
3758d765af5SAbhijit Gangurde const struct auxiliary_device_id *id)
3768d765af5SAbhijit Gangurde {
3778d765af5SAbhijit Gangurde struct ionic_aux_dev *ionic_adev;
3788d765af5SAbhijit Gangurde struct ionic_ibdev *dev;
3798d765af5SAbhijit Gangurde
3808d765af5SAbhijit Gangurde ionic_adev = container_of(adev, struct ionic_aux_dev, adev);
3818d765af5SAbhijit Gangurde dev = ionic_create_ibdev(ionic_adev);
3828d765af5SAbhijit Gangurde if (IS_ERR(dev))
3838d765af5SAbhijit Gangurde return dev_err_probe(&adev->dev, PTR_ERR(dev),
3848d765af5SAbhijit Gangurde "Failed to register ibdev\n");
3858d765af5SAbhijit Gangurde
3868d765af5SAbhijit Gangurde auxiliary_set_drvdata(adev, dev);
3878d765af5SAbhijit Gangurde ibdev_dbg(&dev->ibdev, "registered\n");
3888d765af5SAbhijit Gangurde
3898d765af5SAbhijit Gangurde return 0;
3908d765af5SAbhijit Gangurde }
3918d765af5SAbhijit Gangurde
ionic_aux_remove(struct auxiliary_device * adev)3928d765af5SAbhijit Gangurde static void ionic_aux_remove(struct auxiliary_device *adev)
3938d765af5SAbhijit Gangurde {
3948d765af5SAbhijit Gangurde struct ionic_ibdev *dev = auxiliary_get_drvdata(adev);
3958d765af5SAbhijit Gangurde
3968d765af5SAbhijit Gangurde dev_dbg(&adev->dev, "unregister ibdev\n");
3978d765af5SAbhijit Gangurde ionic_destroy_ibdev(dev);
3988d765af5SAbhijit Gangurde dev_dbg(&adev->dev, "unregistered\n");
3998d765af5SAbhijit Gangurde }
4008d765af5SAbhijit Gangurde
4018d765af5SAbhijit Gangurde static const struct auxiliary_device_id ionic_aux_id_table[] = {
4028d765af5SAbhijit Gangurde { .name = "ionic.rdma", },
4038d765af5SAbhijit Gangurde {},
4048d765af5SAbhijit Gangurde };
4058d765af5SAbhijit Gangurde
4068d765af5SAbhijit Gangurde MODULE_DEVICE_TABLE(auxiliary, ionic_aux_id_table);
4078d765af5SAbhijit Gangurde
4088d765af5SAbhijit Gangurde static struct auxiliary_driver ionic_aux_r_driver = {
4098d765af5SAbhijit Gangurde .name = "rdma",
4108d765af5SAbhijit Gangurde .probe = ionic_aux_probe,
4118d765af5SAbhijit Gangurde .remove = ionic_aux_remove,
4128d765af5SAbhijit Gangurde .id_table = ionic_aux_id_table,
4138d765af5SAbhijit Gangurde };
4148d765af5SAbhijit Gangurde
ionic_mod_init(void)4158d765af5SAbhijit Gangurde static int __init ionic_mod_init(void)
4168d765af5SAbhijit Gangurde {
4178d765af5SAbhijit Gangurde int rc;
4188d765af5SAbhijit Gangurde
419e8521822SAbhijit Gangurde ionic_evt_workq = create_workqueue(KBUILD_MODNAME "-evt");
420f3bdbd42SAbhijit Gangurde if (!ionic_evt_workq)
421f3bdbd42SAbhijit Gangurde return -ENOMEM;
422f3bdbd42SAbhijit Gangurde
4238d765af5SAbhijit Gangurde rc = auxiliary_driver_register(&ionic_aux_r_driver);
4248d765af5SAbhijit Gangurde if (rc)
4258d765af5SAbhijit Gangurde goto err_aux;
4268d765af5SAbhijit Gangurde
4278d765af5SAbhijit Gangurde return 0;
4288d765af5SAbhijit Gangurde
4298d765af5SAbhijit Gangurde err_aux:
430f3bdbd42SAbhijit Gangurde destroy_workqueue(ionic_evt_workq);
431f3bdbd42SAbhijit Gangurde
4328d765af5SAbhijit Gangurde return rc;
4338d765af5SAbhijit Gangurde }
4348d765af5SAbhijit Gangurde
ionic_mod_exit(void)4358d765af5SAbhijit Gangurde static void __exit ionic_mod_exit(void)
4368d765af5SAbhijit Gangurde {
4378d765af5SAbhijit Gangurde auxiliary_driver_unregister(&ionic_aux_r_driver);
438f3bdbd42SAbhijit Gangurde destroy_workqueue(ionic_evt_workq);
4398d765af5SAbhijit Gangurde }
4408d765af5SAbhijit Gangurde
4418d765af5SAbhijit Gangurde module_init(ionic_mod_init);
4428d765af5SAbhijit Gangurde module_exit(ionic_mod_exit);
443