xref: /linux/drivers/infiniband/hw/ionic/ionic_ibdev.c (revision 2075bbe8ef03914aa2211035eec45d1d3a5c4ff2)
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