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> 6*2075bbe8SAbhijit Gangurde #include <linux/pci.h> 7*2075bbe8SAbhijit Gangurde #include <linux/irq.h> 88d765af5SAbhijit Gangurde #include <net/addrconf.h> 9*2075bbe8SAbhijit Gangurde #include <rdma/ib_addr.h> 10*2075bbe8SAbhijit 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 22*2075bbe8SAbhijit Gangurde static int ionic_query_device(struct ib_device *ibdev, 23*2075bbe8SAbhijit Gangurde struct ib_device_attr *attr, 24*2075bbe8SAbhijit Gangurde struct ib_udata *udata) 25*2075bbe8SAbhijit Gangurde { 26*2075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev); 27*2075bbe8SAbhijit Gangurde struct net_device *ndev; 28*2075bbe8SAbhijit Gangurde 29*2075bbe8SAbhijit Gangurde ndev = ib_device_get_netdev(ibdev, 1); 30*2075bbe8SAbhijit Gangurde addrconf_ifid_eui48((u8 *)&attr->sys_image_guid, ndev); 31*2075bbe8SAbhijit Gangurde dev_put(ndev); 32*2075bbe8SAbhijit Gangurde attr->max_mr_size = dev->lif_cfg.npts_per_lif * PAGE_SIZE / 2; 33*2075bbe8SAbhijit Gangurde attr->page_size_cap = dev->lif_cfg.page_size_supported; 34*2075bbe8SAbhijit Gangurde 35*2075bbe8SAbhijit Gangurde attr->vendor_id = to_pci_dev(dev->lif_cfg.hwdev)->vendor; 36*2075bbe8SAbhijit Gangurde attr->vendor_part_id = to_pci_dev(dev->lif_cfg.hwdev)->device; 37*2075bbe8SAbhijit Gangurde 38*2075bbe8SAbhijit Gangurde attr->hw_ver = ionic_lif_asic_rev(dev->lif_cfg.lif); 39*2075bbe8SAbhijit Gangurde attr->fw_ver = 0; 40*2075bbe8SAbhijit Gangurde attr->max_qp = dev->lif_cfg.qp_count; 41*2075bbe8SAbhijit Gangurde attr->max_qp_wr = IONIC_MAX_DEPTH; 42*2075bbe8SAbhijit Gangurde attr->device_cap_flags = 43*2075bbe8SAbhijit Gangurde IB_DEVICE_MEM_WINDOW | 44*2075bbe8SAbhijit Gangurde IB_DEVICE_MEM_MGT_EXTENSIONS | 45*2075bbe8SAbhijit Gangurde IB_DEVICE_MEM_WINDOW_TYPE_2B | 46*2075bbe8SAbhijit Gangurde 0; 47*2075bbe8SAbhijit Gangurde attr->max_send_sge = 48*2075bbe8SAbhijit Gangurde min(ionic_v1_send_wqe_max_sge(dev->lif_cfg.max_stride, 0, false), 49*2075bbe8SAbhijit Gangurde IONIC_SPEC_HIGH); 50*2075bbe8SAbhijit Gangurde attr->max_recv_sge = 51*2075bbe8SAbhijit Gangurde min(ionic_v1_recv_wqe_max_sge(dev->lif_cfg.max_stride, 0, false), 52*2075bbe8SAbhijit Gangurde IONIC_SPEC_HIGH); 53*2075bbe8SAbhijit Gangurde attr->max_sge_rd = attr->max_send_sge; 54*2075bbe8SAbhijit Gangurde attr->max_cq = dev->lif_cfg.cq_count / dev->lif_cfg.udma_count; 55*2075bbe8SAbhijit Gangurde attr->max_cqe = IONIC_MAX_CQ_DEPTH - IONIC_CQ_GRACE; 56*2075bbe8SAbhijit Gangurde attr->max_mr = dev->lif_cfg.nmrs_per_lif; 57*2075bbe8SAbhijit Gangurde attr->max_pd = IONIC_MAX_PD; 58*2075bbe8SAbhijit Gangurde attr->max_qp_rd_atom = IONIC_MAX_RD_ATOM; 59*2075bbe8SAbhijit Gangurde attr->max_ee_rd_atom = 0; 60*2075bbe8SAbhijit Gangurde attr->max_res_rd_atom = IONIC_MAX_RD_ATOM; 61*2075bbe8SAbhijit Gangurde attr->max_qp_init_rd_atom = IONIC_MAX_RD_ATOM; 62*2075bbe8SAbhijit Gangurde attr->max_ee_init_rd_atom = 0; 63*2075bbe8SAbhijit Gangurde attr->atomic_cap = IB_ATOMIC_GLOB; 64*2075bbe8SAbhijit Gangurde attr->masked_atomic_cap = IB_ATOMIC_GLOB; 65*2075bbe8SAbhijit Gangurde attr->max_mw = dev->lif_cfg.nmrs_per_lif; 66*2075bbe8SAbhijit Gangurde attr->max_mcast_grp = 0; 67*2075bbe8SAbhijit Gangurde attr->max_mcast_qp_attach = 0; 68*2075bbe8SAbhijit Gangurde attr->max_ah = dev->lif_cfg.nahs_per_lif; 69*2075bbe8SAbhijit Gangurde attr->max_fast_reg_page_list_len = dev->lif_cfg.npts_per_lif / 2; 70*2075bbe8SAbhijit Gangurde attr->max_pkeys = IONIC_PKEY_TBL_LEN; 71*2075bbe8SAbhijit Gangurde 72*2075bbe8SAbhijit Gangurde return 0; 73*2075bbe8SAbhijit Gangurde } 74*2075bbe8SAbhijit Gangurde 75*2075bbe8SAbhijit Gangurde static int ionic_query_port(struct ib_device *ibdev, u32 port, 76*2075bbe8SAbhijit Gangurde struct ib_port_attr *attr) 77*2075bbe8SAbhijit Gangurde { 78*2075bbe8SAbhijit Gangurde struct net_device *ndev; 79*2075bbe8SAbhijit Gangurde 80*2075bbe8SAbhijit Gangurde if (port != 1) 81*2075bbe8SAbhijit Gangurde return -EINVAL; 82*2075bbe8SAbhijit Gangurde 83*2075bbe8SAbhijit Gangurde ndev = ib_device_get_netdev(ibdev, port); 84*2075bbe8SAbhijit Gangurde 85*2075bbe8SAbhijit Gangurde if (netif_running(ndev) && netif_carrier_ok(ndev)) { 86*2075bbe8SAbhijit Gangurde attr->state = IB_PORT_ACTIVE; 87*2075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_LINK_UP; 88*2075bbe8SAbhijit Gangurde } else if (netif_running(ndev)) { 89*2075bbe8SAbhijit Gangurde attr->state = IB_PORT_DOWN; 90*2075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_POLLING; 91*2075bbe8SAbhijit Gangurde } else { 92*2075bbe8SAbhijit Gangurde attr->state = IB_PORT_DOWN; 93*2075bbe8SAbhijit Gangurde attr->phys_state = IB_PORT_PHYS_STATE_DISABLED; 94*2075bbe8SAbhijit Gangurde } 95*2075bbe8SAbhijit Gangurde 96*2075bbe8SAbhijit Gangurde attr->max_mtu = iboe_get_mtu(ndev->max_mtu); 97*2075bbe8SAbhijit Gangurde attr->active_mtu = min(attr->max_mtu, iboe_get_mtu(ndev->mtu)); 98*2075bbe8SAbhijit Gangurde attr->gid_tbl_len = IONIC_GID_TBL_LEN; 99*2075bbe8SAbhijit Gangurde attr->ip_gids = true; 100*2075bbe8SAbhijit Gangurde attr->port_cap_flags = 0; 101*2075bbe8SAbhijit Gangurde attr->max_msg_sz = 0x80000000; 102*2075bbe8SAbhijit Gangurde attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN; 103*2075bbe8SAbhijit Gangurde attr->max_vl_num = 1; 104*2075bbe8SAbhijit Gangurde attr->subnet_prefix = 0xfe80000000000000ull; 105*2075bbe8SAbhijit Gangurde 106*2075bbe8SAbhijit Gangurde dev_put(ndev); 107*2075bbe8SAbhijit Gangurde 108*2075bbe8SAbhijit Gangurde return ib_get_eth_speed(ibdev, port, 109*2075bbe8SAbhijit Gangurde &attr->active_speed, 110*2075bbe8SAbhijit Gangurde &attr->active_width); 111*2075bbe8SAbhijit Gangurde } 112*2075bbe8SAbhijit Gangurde 113*2075bbe8SAbhijit Gangurde static enum rdma_link_layer ionic_get_link_layer(struct ib_device *ibdev, 114*2075bbe8SAbhijit Gangurde u32 port) 115*2075bbe8SAbhijit Gangurde { 116*2075bbe8SAbhijit Gangurde return IB_LINK_LAYER_ETHERNET; 117*2075bbe8SAbhijit Gangurde } 118*2075bbe8SAbhijit Gangurde 119*2075bbe8SAbhijit Gangurde static int ionic_query_pkey(struct ib_device *ibdev, u32 port, u16 index, 120*2075bbe8SAbhijit Gangurde u16 *pkey) 121*2075bbe8SAbhijit Gangurde { 122*2075bbe8SAbhijit Gangurde if (port != 1) 123*2075bbe8SAbhijit Gangurde return -EINVAL; 124*2075bbe8SAbhijit Gangurde 125*2075bbe8SAbhijit Gangurde if (index != 0) 126*2075bbe8SAbhijit Gangurde return -EINVAL; 127*2075bbe8SAbhijit Gangurde 128*2075bbe8SAbhijit Gangurde *pkey = IB_DEFAULT_PKEY_FULL; 129*2075bbe8SAbhijit Gangurde 130*2075bbe8SAbhijit Gangurde return 0; 131*2075bbe8SAbhijit Gangurde } 132*2075bbe8SAbhijit Gangurde 133*2075bbe8SAbhijit Gangurde static int ionic_modify_device(struct ib_device *ibdev, int mask, 134*2075bbe8SAbhijit Gangurde struct ib_device_modify *attr) 135*2075bbe8SAbhijit Gangurde { 136*2075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev); 137*2075bbe8SAbhijit Gangurde 138*2075bbe8SAbhijit Gangurde if (mask & ~IB_DEVICE_MODIFY_NODE_DESC) 139*2075bbe8SAbhijit Gangurde return -EOPNOTSUPP; 140*2075bbe8SAbhijit Gangurde 141*2075bbe8SAbhijit Gangurde if (mask & IB_DEVICE_MODIFY_NODE_DESC) 142*2075bbe8SAbhijit Gangurde memcpy(dev->ibdev.node_desc, attr->node_desc, 143*2075bbe8SAbhijit Gangurde IB_DEVICE_NODE_DESC_MAX); 144*2075bbe8SAbhijit Gangurde 145*2075bbe8SAbhijit Gangurde return 0; 146*2075bbe8SAbhijit Gangurde } 147*2075bbe8SAbhijit Gangurde 148*2075bbe8SAbhijit Gangurde static int ionic_get_port_immutable(struct ib_device *ibdev, u32 port, 149*2075bbe8SAbhijit Gangurde struct ib_port_immutable *attr) 150*2075bbe8SAbhijit Gangurde { 151*2075bbe8SAbhijit Gangurde if (port != 1) 152*2075bbe8SAbhijit Gangurde return -EINVAL; 153*2075bbe8SAbhijit Gangurde 154*2075bbe8SAbhijit Gangurde attr->core_cap_flags = RDMA_CORE_PORT_IBA_ROCE_UDP_ENCAP; 155*2075bbe8SAbhijit Gangurde 156*2075bbe8SAbhijit Gangurde attr->pkey_tbl_len = IONIC_PKEY_TBL_LEN; 157*2075bbe8SAbhijit Gangurde attr->gid_tbl_len = IONIC_GID_TBL_LEN; 158*2075bbe8SAbhijit Gangurde attr->max_mad_size = IB_MGMT_MAD_SIZE; 159*2075bbe8SAbhijit Gangurde 160*2075bbe8SAbhijit Gangurde return 0; 161*2075bbe8SAbhijit Gangurde } 162*2075bbe8SAbhijit Gangurde 163*2075bbe8SAbhijit Gangurde static void ionic_get_dev_fw_str(struct ib_device *ibdev, char *str) 164*2075bbe8SAbhijit Gangurde { 165*2075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = to_ionic_ibdev(ibdev); 166*2075bbe8SAbhijit Gangurde 167*2075bbe8SAbhijit Gangurde ionic_lif_fw_version(dev->lif_cfg.lif, str, IB_FW_VERSION_NAME_MAX); 168*2075bbe8SAbhijit Gangurde } 169*2075bbe8SAbhijit Gangurde 170*2075bbe8SAbhijit Gangurde static ssize_t hw_rev_show(struct device *device, struct device_attribute *attr, 171*2075bbe8SAbhijit Gangurde char *buf) 172*2075bbe8SAbhijit Gangurde { 173*2075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = 174*2075bbe8SAbhijit Gangurde rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev); 175*2075bbe8SAbhijit Gangurde 176*2075bbe8SAbhijit Gangurde return sysfs_emit(buf, "0x%x\n", ionic_lif_asic_rev(dev->lif_cfg.lif)); 177*2075bbe8SAbhijit Gangurde } 178*2075bbe8SAbhijit Gangurde static DEVICE_ATTR_RO(hw_rev); 179*2075bbe8SAbhijit Gangurde 180*2075bbe8SAbhijit Gangurde static ssize_t hca_type_show(struct device *device, 181*2075bbe8SAbhijit Gangurde struct device_attribute *attr, char *buf) 182*2075bbe8SAbhijit Gangurde { 183*2075bbe8SAbhijit Gangurde struct ionic_ibdev *dev = 184*2075bbe8SAbhijit Gangurde rdma_device_to_drv_device(device, struct ionic_ibdev, ibdev); 185*2075bbe8SAbhijit Gangurde 186*2075bbe8SAbhijit Gangurde return sysfs_emit(buf, "%s\n", dev->ibdev.node_desc); 187*2075bbe8SAbhijit Gangurde } 188*2075bbe8SAbhijit Gangurde static DEVICE_ATTR_RO(hca_type); 189*2075bbe8SAbhijit Gangurde 190*2075bbe8SAbhijit Gangurde static struct attribute *ionic_rdma_attributes[] = { 191*2075bbe8SAbhijit Gangurde &dev_attr_hw_rev.attr, 192*2075bbe8SAbhijit Gangurde &dev_attr_hca_type.attr, 193*2075bbe8SAbhijit Gangurde NULL 194*2075bbe8SAbhijit Gangurde }; 195*2075bbe8SAbhijit Gangurde 196*2075bbe8SAbhijit Gangurde static const struct attribute_group ionic_rdma_attr_group = { 197*2075bbe8SAbhijit Gangurde .attrs = ionic_rdma_attributes, 198*2075bbe8SAbhijit Gangurde }; 199*2075bbe8SAbhijit Gangurde 200*2075bbe8SAbhijit Gangurde static void ionic_disassociate_ucontext(struct ib_ucontext *ibcontext) 201*2075bbe8SAbhijit Gangurde { 202*2075bbe8SAbhijit Gangurde /* 203*2075bbe8SAbhijit Gangurde * Dummy define disassociate_ucontext so that it does not 204*2075bbe8SAbhijit Gangurde * wait for user context before cleaning up hw resources. 205*2075bbe8SAbhijit Gangurde */ 206*2075bbe8SAbhijit Gangurde } 207*2075bbe8SAbhijit Gangurde 208e8521822SAbhijit Gangurde static const struct ib_device_ops ionic_dev_ops = { 209e8521822SAbhijit Gangurde .owner = THIS_MODULE, 210e8521822SAbhijit Gangurde .driver_id = RDMA_DRIVER_IONIC, 211e8521822SAbhijit Gangurde .uverbs_abi_ver = IONIC_ABI_VERSION, 212e8521822SAbhijit Gangurde 213e8521822SAbhijit Gangurde .alloc_ucontext = ionic_alloc_ucontext, 214e8521822SAbhijit Gangurde .dealloc_ucontext = ionic_dealloc_ucontext, 215e8521822SAbhijit Gangurde .mmap = ionic_mmap, 216e8521822SAbhijit Gangurde .mmap_free = ionic_mmap_free, 217e8521822SAbhijit Gangurde .alloc_pd = ionic_alloc_pd, 218e8521822SAbhijit Gangurde .dealloc_pd = ionic_dealloc_pd, 219e8521822SAbhijit Gangurde .create_ah = ionic_create_ah, 220e8521822SAbhijit Gangurde .query_ah = ionic_query_ah, 221e8521822SAbhijit Gangurde .destroy_ah = ionic_destroy_ah, 222e8521822SAbhijit Gangurde .create_user_ah = ionic_create_ah, 223e8521822SAbhijit Gangurde .get_dma_mr = ionic_get_dma_mr, 224e8521822SAbhijit Gangurde .reg_user_mr = ionic_reg_user_mr, 225e8521822SAbhijit Gangurde .reg_user_mr_dmabuf = ionic_reg_user_mr_dmabuf, 226e8521822SAbhijit Gangurde .dereg_mr = ionic_dereg_mr, 227e8521822SAbhijit Gangurde .alloc_mr = ionic_alloc_mr, 228e8521822SAbhijit Gangurde .map_mr_sg = ionic_map_mr_sg, 229e8521822SAbhijit Gangurde .alloc_mw = ionic_alloc_mw, 230e8521822SAbhijit Gangurde .dealloc_mw = ionic_dealloc_mw, 231e8521822SAbhijit Gangurde .create_cq = ionic_create_cq, 232e8521822SAbhijit Gangurde .destroy_cq = ionic_destroy_cq, 233e8521822SAbhijit Gangurde .create_qp = ionic_create_qp, 234e8521822SAbhijit Gangurde .modify_qp = ionic_modify_qp, 235e8521822SAbhijit Gangurde .query_qp = ionic_query_qp, 236e8521822SAbhijit Gangurde .destroy_qp = ionic_destroy_qp, 237e8521822SAbhijit Gangurde 238b83c6205SAbhijit Gangurde .post_send = ionic_post_send, 239b83c6205SAbhijit Gangurde .post_recv = ionic_post_recv, 240b83c6205SAbhijit Gangurde .poll_cq = ionic_poll_cq, 241b83c6205SAbhijit Gangurde .req_notify_cq = ionic_req_notify_cq, 242b83c6205SAbhijit Gangurde 243*2075bbe8SAbhijit Gangurde .query_device = ionic_query_device, 244*2075bbe8SAbhijit Gangurde .query_port = ionic_query_port, 245*2075bbe8SAbhijit Gangurde .get_link_layer = ionic_get_link_layer, 246*2075bbe8SAbhijit Gangurde .query_pkey = ionic_query_pkey, 247*2075bbe8SAbhijit Gangurde .modify_device = ionic_modify_device, 248*2075bbe8SAbhijit Gangurde .get_port_immutable = ionic_get_port_immutable, 249*2075bbe8SAbhijit Gangurde .get_dev_fw_str = ionic_get_dev_fw_str, 250*2075bbe8SAbhijit Gangurde .device_group = &ionic_rdma_attr_group, 251*2075bbe8SAbhijit Gangurde .disassociate_ucontext = ionic_disassociate_ucontext, 252*2075bbe8SAbhijit Gangurde 253e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_ucontext, ionic_ctx, ibctx), 254e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_pd, ionic_pd, ibpd), 255e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_ah, ionic_ah, ibah), 256e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_cq, ionic_vcq, ibcq), 257e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_qp, ionic_qp, ibqp), 258e8521822SAbhijit Gangurde INIT_RDMA_OBJ_SIZE(ib_mw, ionic_mr, ibmw), 259e8521822SAbhijit Gangurde }; 260e8521822SAbhijit Gangurde 261f3bdbd42SAbhijit Gangurde static void ionic_init_resids(struct ionic_ibdev *dev) 262f3bdbd42SAbhijit Gangurde { 263f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_cqid, dev->lif_cfg.cq_count); 264f3bdbd42SAbhijit Gangurde dev->half_cqid_udma_shift = 265f3bdbd42SAbhijit Gangurde order_base_2(dev->lif_cfg.cq_count / dev->lif_cfg.udma_count); 266f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_pdid, IONIC_MAX_PD); 267f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_ahid, dev->lif_cfg.nahs_per_lif); 268f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_mrid, dev->lif_cfg.nmrs_per_lif); 269f3bdbd42SAbhijit Gangurde /* skip reserved lkey */ 270f3bdbd42SAbhijit Gangurde dev->next_mrkey = 1; 271f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_qpid, dev->lif_cfg.qp_count); 272f3bdbd42SAbhijit Gangurde /* skip reserved SMI and GSI qpids */ 273f3bdbd42SAbhijit Gangurde dev->half_qpid_udma_shift = 274f3bdbd42SAbhijit Gangurde order_base_2(dev->lif_cfg.qp_count / dev->lif_cfg.udma_count); 275f3bdbd42SAbhijit Gangurde ionic_resid_init(&dev->inuse_dbid, dev->lif_cfg.dbid_count); 276f3bdbd42SAbhijit Gangurde } 277f3bdbd42SAbhijit Gangurde 278f3bdbd42SAbhijit Gangurde static void ionic_destroy_resids(struct ionic_ibdev *dev) 279f3bdbd42SAbhijit Gangurde { 280f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_cqid); 281f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_pdid); 282f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_ahid); 283f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_mrid); 284f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_qpid); 285f3bdbd42SAbhijit Gangurde ionic_resid_destroy(&dev->inuse_dbid); 286f3bdbd42SAbhijit Gangurde } 287f3bdbd42SAbhijit Gangurde 2888d765af5SAbhijit Gangurde static void ionic_destroy_ibdev(struct ionic_ibdev *dev) 2898d765af5SAbhijit Gangurde { 290f3bdbd42SAbhijit Gangurde ionic_kill_rdma_admin(dev, false); 2918d765af5SAbhijit Gangurde ib_unregister_device(&dev->ibdev); 292f3bdbd42SAbhijit Gangurde ionic_destroy_rdma_admin(dev); 293f3bdbd42SAbhijit Gangurde ionic_destroy_resids(dev); 294e8521822SAbhijit Gangurde WARN_ON(!xa_empty(&dev->qp_tbl)); 295e8521822SAbhijit Gangurde xa_destroy(&dev->qp_tbl); 296f3bdbd42SAbhijit Gangurde WARN_ON(!xa_empty(&dev->cq_tbl)); 297f3bdbd42SAbhijit Gangurde xa_destroy(&dev->cq_tbl); 2988d765af5SAbhijit Gangurde ib_dealloc_device(&dev->ibdev); 2998d765af5SAbhijit Gangurde } 3008d765af5SAbhijit Gangurde 3018d765af5SAbhijit Gangurde static struct ionic_ibdev *ionic_create_ibdev(struct ionic_aux_dev *ionic_adev) 3028d765af5SAbhijit Gangurde { 3038d765af5SAbhijit Gangurde struct ib_device *ibdev; 3048d765af5SAbhijit Gangurde struct ionic_ibdev *dev; 3058d765af5SAbhijit Gangurde struct net_device *ndev; 3068d765af5SAbhijit Gangurde int rc; 3078d765af5SAbhijit Gangurde 3088d765af5SAbhijit Gangurde dev = ib_alloc_device(ionic_ibdev, ibdev); 3098d765af5SAbhijit Gangurde if (!dev) 3108d765af5SAbhijit Gangurde return ERR_PTR(-EINVAL); 3118d765af5SAbhijit Gangurde 3128d765af5SAbhijit Gangurde ionic_fill_lif_cfg(ionic_adev->lif, &dev->lif_cfg); 3138d765af5SAbhijit Gangurde 314e8521822SAbhijit Gangurde xa_init_flags(&dev->qp_tbl, GFP_ATOMIC); 315f3bdbd42SAbhijit Gangurde xa_init_flags(&dev->cq_tbl, GFP_ATOMIC); 316f3bdbd42SAbhijit Gangurde 317f3bdbd42SAbhijit Gangurde ionic_init_resids(dev); 318f3bdbd42SAbhijit Gangurde 319f3bdbd42SAbhijit Gangurde rc = ionic_rdma_reset_devcmd(dev); 320f3bdbd42SAbhijit Gangurde if (rc) 321f3bdbd42SAbhijit Gangurde goto err_reset; 322f3bdbd42SAbhijit Gangurde 323f3bdbd42SAbhijit Gangurde rc = ionic_create_rdma_admin(dev); 324f3bdbd42SAbhijit Gangurde if (rc) 325f3bdbd42SAbhijit Gangurde goto err_admin; 326f3bdbd42SAbhijit Gangurde 3278d765af5SAbhijit Gangurde ibdev = &dev->ibdev; 3288d765af5SAbhijit Gangurde ibdev->dev.parent = dev->lif_cfg.hwdev; 3298d765af5SAbhijit Gangurde 3308d765af5SAbhijit Gangurde strscpy(ibdev->name, "ionic_%d", IB_DEVICE_NAME_MAX); 3318d765af5SAbhijit Gangurde strscpy(ibdev->node_desc, DEVICE_DESCRIPTION, IB_DEVICE_NODE_DESC_MAX); 3328d765af5SAbhijit Gangurde 3338d765af5SAbhijit Gangurde ibdev->node_type = RDMA_NODE_IB_CA; 3348d765af5SAbhijit Gangurde ibdev->phys_port_cnt = 1; 3358d765af5SAbhijit Gangurde 3368d765af5SAbhijit Gangurde /* the first two eq are reserved for async events */ 3378d765af5SAbhijit Gangurde ibdev->num_comp_vectors = dev->lif_cfg.eq_count - 2; 3388d765af5SAbhijit Gangurde 3398d765af5SAbhijit Gangurde ndev = ionic_lif_netdev(ionic_adev->lif); 3408d765af5SAbhijit Gangurde addrconf_ifid_eui48((u8 *)&ibdev->node_guid, ndev); 3418d765af5SAbhijit Gangurde rc = ib_device_set_netdev(ibdev, ndev, 1); 3428d765af5SAbhijit Gangurde /* ionic_lif_netdev() returns ndev with refcount held */ 3438d765af5SAbhijit Gangurde dev_put(ndev); 3448d765af5SAbhijit Gangurde if (rc) 3458d765af5SAbhijit Gangurde goto err_admin; 3468d765af5SAbhijit Gangurde 347e8521822SAbhijit Gangurde ib_set_device_ops(&dev->ibdev, &ionic_dev_ops); 348e8521822SAbhijit Gangurde 3498d765af5SAbhijit Gangurde rc = ib_register_device(ibdev, "ionic_%d", ibdev->dev.parent); 3508d765af5SAbhijit Gangurde if (rc) 3518d765af5SAbhijit Gangurde goto err_register; 3528d765af5SAbhijit Gangurde 3538d765af5SAbhijit Gangurde return dev; 3548d765af5SAbhijit Gangurde 3558d765af5SAbhijit Gangurde err_register: 3568d765af5SAbhijit Gangurde err_admin: 357f3bdbd42SAbhijit Gangurde ionic_kill_rdma_admin(dev, false); 358f3bdbd42SAbhijit Gangurde ionic_destroy_rdma_admin(dev); 359f3bdbd42SAbhijit Gangurde err_reset: 360f3bdbd42SAbhijit Gangurde ionic_destroy_resids(dev); 361e8521822SAbhijit Gangurde xa_destroy(&dev->qp_tbl); 362f3bdbd42SAbhijit Gangurde xa_destroy(&dev->cq_tbl); 3638d765af5SAbhijit Gangurde ib_dealloc_device(&dev->ibdev); 3648d765af5SAbhijit Gangurde 3658d765af5SAbhijit Gangurde return ERR_PTR(rc); 3668d765af5SAbhijit Gangurde } 3678d765af5SAbhijit Gangurde 3688d765af5SAbhijit Gangurde static int ionic_aux_probe(struct auxiliary_device *adev, 3698d765af5SAbhijit Gangurde const struct auxiliary_device_id *id) 3708d765af5SAbhijit Gangurde { 3718d765af5SAbhijit Gangurde struct ionic_aux_dev *ionic_adev; 3728d765af5SAbhijit Gangurde struct ionic_ibdev *dev; 3738d765af5SAbhijit Gangurde 3748d765af5SAbhijit Gangurde ionic_adev = container_of(adev, struct ionic_aux_dev, adev); 3758d765af5SAbhijit Gangurde dev = ionic_create_ibdev(ionic_adev); 3768d765af5SAbhijit Gangurde if (IS_ERR(dev)) 3778d765af5SAbhijit Gangurde return dev_err_probe(&adev->dev, PTR_ERR(dev), 3788d765af5SAbhijit Gangurde "Failed to register ibdev\n"); 3798d765af5SAbhijit Gangurde 3808d765af5SAbhijit Gangurde auxiliary_set_drvdata(adev, dev); 3818d765af5SAbhijit Gangurde ibdev_dbg(&dev->ibdev, "registered\n"); 3828d765af5SAbhijit Gangurde 3838d765af5SAbhijit Gangurde return 0; 3848d765af5SAbhijit Gangurde } 3858d765af5SAbhijit Gangurde 3868d765af5SAbhijit Gangurde static void ionic_aux_remove(struct auxiliary_device *adev) 3878d765af5SAbhijit Gangurde { 3888d765af5SAbhijit Gangurde struct ionic_ibdev *dev = auxiliary_get_drvdata(adev); 3898d765af5SAbhijit Gangurde 3908d765af5SAbhijit Gangurde dev_dbg(&adev->dev, "unregister ibdev\n"); 3918d765af5SAbhijit Gangurde ionic_destroy_ibdev(dev); 3928d765af5SAbhijit Gangurde dev_dbg(&adev->dev, "unregistered\n"); 3938d765af5SAbhijit Gangurde } 3948d765af5SAbhijit Gangurde 3958d765af5SAbhijit Gangurde static const struct auxiliary_device_id ionic_aux_id_table[] = { 3968d765af5SAbhijit Gangurde { .name = "ionic.rdma", }, 3978d765af5SAbhijit Gangurde {}, 3988d765af5SAbhijit Gangurde }; 3998d765af5SAbhijit Gangurde 4008d765af5SAbhijit Gangurde MODULE_DEVICE_TABLE(auxiliary, ionic_aux_id_table); 4018d765af5SAbhijit Gangurde 4028d765af5SAbhijit Gangurde static struct auxiliary_driver ionic_aux_r_driver = { 4038d765af5SAbhijit Gangurde .name = "rdma", 4048d765af5SAbhijit Gangurde .probe = ionic_aux_probe, 4058d765af5SAbhijit Gangurde .remove = ionic_aux_remove, 4068d765af5SAbhijit Gangurde .id_table = ionic_aux_id_table, 4078d765af5SAbhijit Gangurde }; 4088d765af5SAbhijit Gangurde 4098d765af5SAbhijit Gangurde static int __init ionic_mod_init(void) 4108d765af5SAbhijit Gangurde { 4118d765af5SAbhijit Gangurde int rc; 4128d765af5SAbhijit Gangurde 413e8521822SAbhijit Gangurde ionic_evt_workq = create_workqueue(KBUILD_MODNAME "-evt"); 414f3bdbd42SAbhijit Gangurde if (!ionic_evt_workq) 415f3bdbd42SAbhijit Gangurde return -ENOMEM; 416f3bdbd42SAbhijit Gangurde 4178d765af5SAbhijit Gangurde rc = auxiliary_driver_register(&ionic_aux_r_driver); 4188d765af5SAbhijit Gangurde if (rc) 4198d765af5SAbhijit Gangurde goto err_aux; 4208d765af5SAbhijit Gangurde 4218d765af5SAbhijit Gangurde return 0; 4228d765af5SAbhijit Gangurde 4238d765af5SAbhijit Gangurde err_aux: 424f3bdbd42SAbhijit Gangurde destroy_workqueue(ionic_evt_workq); 425f3bdbd42SAbhijit Gangurde 4268d765af5SAbhijit Gangurde return rc; 4278d765af5SAbhijit Gangurde } 4288d765af5SAbhijit Gangurde 4298d765af5SAbhijit Gangurde static void __exit ionic_mod_exit(void) 4308d765af5SAbhijit Gangurde { 4318d765af5SAbhijit Gangurde auxiliary_driver_unregister(&ionic_aux_r_driver); 432f3bdbd42SAbhijit Gangurde destroy_workqueue(ionic_evt_workq); 4338d765af5SAbhijit Gangurde } 4348d765af5SAbhijit Gangurde 4358d765af5SAbhijit Gangurde module_init(ionic_mod_init); 4368d765af5SAbhijit Gangurde module_exit(ionic_mod_exit); 437