xref: /openbmc/linux/drivers/infiniband/hw/mana/qp.c (revision 21453285)
10266a177SLong Li // SPDX-License-Identifier: GPL-2.0-only
20266a177SLong Li /*
30266a177SLong Li  * Copyright (c) 2022, Microsoft Corporation. All rights reserved.
40266a177SLong Li  */
50266a177SLong Li 
60266a177SLong Li #include "mana_ib.h"
70266a177SLong Li 
mana_ib_cfg_vport_steering(struct mana_ib_dev * dev,struct net_device * ndev,mana_handle_t default_rxobj,mana_handle_t ind_table[],u32 log_ind_tbl_size,u32 rx_hash_key_len,u8 * rx_hash_key)80266a177SLong Li static int mana_ib_cfg_vport_steering(struct mana_ib_dev *dev,
90266a177SLong Li 				      struct net_device *ndev,
100266a177SLong Li 				      mana_handle_t default_rxobj,
110266a177SLong Li 				      mana_handle_t ind_table[],
120266a177SLong Li 				      u32 log_ind_tbl_size, u32 rx_hash_key_len,
130266a177SLong Li 				      u8 *rx_hash_key)
140266a177SLong Li {
150266a177SLong Li 	struct mana_port_context *mpc = netdev_priv(ndev);
16*21453285SLong Li 	struct mana_cfg_rx_steer_req_v2 *req;
170266a177SLong Li 	struct mana_cfg_rx_steer_resp resp = {};
180266a177SLong Li 	mana_handle_t *req_indir_tab;
190266a177SLong Li 	struct gdma_context *gc;
200266a177SLong Li 	struct gdma_dev *mdev;
210266a177SLong Li 	u32 req_buf_size;
220266a177SLong Li 	int i, err;
230266a177SLong Li 
240266a177SLong Li 	mdev = dev->gdma_dev;
250266a177SLong Li 	gc = mdev->gdma_context;
260266a177SLong Li 
270266a177SLong Li 	req_buf_size =
280266a177SLong Li 		sizeof(*req) + sizeof(mana_handle_t) * MANA_INDIRECT_TABLE_SIZE;
290266a177SLong Li 	req = kzalloc(req_buf_size, GFP_KERNEL);
300266a177SLong Li 	if (!req)
310266a177SLong Li 		return -ENOMEM;
320266a177SLong Li 
330266a177SLong Li 	mana_gd_init_req_hdr(&req->hdr, MANA_CONFIG_VPORT_RX, req_buf_size,
340266a177SLong Li 			     sizeof(resp));
350266a177SLong Li 
36*21453285SLong Li 	req->hdr.req.msg_version = GDMA_MESSAGE_V2;
37*21453285SLong Li 
380266a177SLong Li 	req->vport = mpc->port_handle;
390266a177SLong Li 	req->rx_enable = 1;
400266a177SLong Li 	req->update_default_rxobj = 1;
410266a177SLong Li 	req->default_rxobj = default_rxobj;
420266a177SLong Li 	req->hdr.dev_id = mdev->dev_id;
430266a177SLong Li 
440266a177SLong Li 	/* If there are more than 1 entries in indirection table, enable RSS */
450266a177SLong Li 	if (log_ind_tbl_size)
460266a177SLong Li 		req->rss_enable = true;
470266a177SLong Li 
480266a177SLong Li 	req->num_indir_entries = MANA_INDIRECT_TABLE_SIZE;
490266a177SLong Li 	req->indir_tab_offset = sizeof(*req);
500266a177SLong Li 	req->update_indir_tab = true;
51*21453285SLong Li 	req->cqe_coalescing_enable = 1;
520266a177SLong Li 
530266a177SLong Li 	req_indir_tab = (mana_handle_t *)(req + 1);
540266a177SLong Li 	/* The ind table passed to the hardware must have
550266a177SLong Li 	 * MANA_INDIRECT_TABLE_SIZE entries. Adjust the verb
560266a177SLong Li 	 * ind_table to MANA_INDIRECT_TABLE_SIZE if required
570266a177SLong Li 	 */
580266a177SLong Li 	ibdev_dbg(&dev->ib_dev, "ind table size %u\n", 1 << log_ind_tbl_size);
590266a177SLong Li 	for (i = 0; i < MANA_INDIRECT_TABLE_SIZE; i++) {
600266a177SLong Li 		req_indir_tab[i] = ind_table[i % (1 << log_ind_tbl_size)];
610266a177SLong Li 		ibdev_dbg(&dev->ib_dev, "index %u handle 0x%llx\n", i,
620266a177SLong Li 			  req_indir_tab[i]);
630266a177SLong Li 	}
640266a177SLong Li 
650266a177SLong Li 	req->update_hashkey = true;
660266a177SLong Li 	if (rx_hash_key_len)
670266a177SLong Li 		memcpy(req->hashkey, rx_hash_key, rx_hash_key_len);
680266a177SLong Li 	else
690266a177SLong Li 		netdev_rss_key_fill(req->hashkey, MANA_HASH_KEY_SIZE);
700266a177SLong Li 
710266a177SLong Li 	ibdev_dbg(&dev->ib_dev, "vport handle %llu default_rxobj 0x%llx\n",
720266a177SLong Li 		  req->vport, default_rxobj);
730266a177SLong Li 
740266a177SLong Li 	err = mana_gd_send_request(gc, req_buf_size, req, sizeof(resp), &resp);
750266a177SLong Li 	if (err) {
760266a177SLong Li 		netdev_err(ndev, "Failed to configure vPort RX: %d\n", err);
770266a177SLong Li 		goto out;
780266a177SLong Li 	}
790266a177SLong Li 
800266a177SLong Li 	if (resp.hdr.status) {
810266a177SLong Li 		netdev_err(ndev, "vPort RX configuration failed: 0x%x\n",
820266a177SLong Li 			   resp.hdr.status);
830266a177SLong Li 		err = -EPROTO;
840266a177SLong Li 		goto out;
850266a177SLong Li 	}
860266a177SLong Li 
870266a177SLong Li 	netdev_info(ndev, "Configured steering vPort %llu log_entries %u\n",
880266a177SLong Li 		    mpc->port_handle, log_ind_tbl_size);
890266a177SLong Li 
900266a177SLong Li out:
910266a177SLong Li 	kfree(req);
920266a177SLong Li 	return err;
930266a177SLong Li }
940266a177SLong Li 
mana_ib_create_qp_rss(struct ib_qp * ibqp,struct ib_pd * pd,struct ib_qp_init_attr * attr,struct ib_udata * udata)950266a177SLong Li static int mana_ib_create_qp_rss(struct ib_qp *ibqp, struct ib_pd *pd,
960266a177SLong Li 				 struct ib_qp_init_attr *attr,
970266a177SLong Li 				 struct ib_udata *udata)
980266a177SLong Li {
990266a177SLong Li 	struct mana_ib_qp *qp = container_of(ibqp, struct mana_ib_qp, ibqp);
1000266a177SLong Li 	struct mana_ib_dev *mdev =
1010266a177SLong Li 		container_of(pd->device, struct mana_ib_dev, ib_dev);
1020266a177SLong Li 	struct ib_rwq_ind_table *ind_tbl = attr->rwq_ind_tbl;
1030266a177SLong Li 	struct mana_ib_create_qp_rss_resp resp = {};
1040266a177SLong Li 	struct mana_ib_create_qp_rss ucmd = {};
1050266a177SLong Li 	struct gdma_dev *gd = mdev->gdma_dev;
1060266a177SLong Li 	mana_handle_t *mana_ind_table;
1070266a177SLong Li 	struct mana_port_context *mpc;
1080266a177SLong Li 	struct mana_context *mc;
1090266a177SLong Li 	struct net_device *ndev;
1100266a177SLong Li 	struct mana_ib_cq *cq;
1110266a177SLong Li 	struct mana_ib_wq *wq;
1120266a177SLong Li 	unsigned int ind_tbl_size;
1130266a177SLong Li 	struct ib_cq *ibcq;
1140266a177SLong Li 	struct ib_wq *ibwq;
1150266a177SLong Li 	int i = 0;
1160266a177SLong Li 	u32 port;
1170266a177SLong Li 	int ret;
1180266a177SLong Li 
1190266a177SLong Li 	mc = gd->driver_data;
1200266a177SLong Li 
1210266a177SLong Li 	if (!udata || udata->inlen < sizeof(ucmd))
1220266a177SLong Li 		return -EINVAL;
1230266a177SLong Li 
1240266a177SLong Li 	ret = ib_copy_from_udata(&ucmd, udata, min(sizeof(ucmd), udata->inlen));
1250266a177SLong Li 	if (ret) {
1260266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
1270266a177SLong Li 			  "Failed copy from udata for create rss-qp, err %d\n",
1280266a177SLong Li 			  ret);
1290266a177SLong Li 		return ret;
1300266a177SLong Li 	}
1310266a177SLong Li 
1320266a177SLong Li 	if (attr->cap.max_recv_wr > MAX_SEND_BUFFERS_PER_QUEUE) {
1330266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
1340266a177SLong Li 			  "Requested max_recv_wr %d exceeding limit\n",
1350266a177SLong Li 			  attr->cap.max_recv_wr);
1360266a177SLong Li 		return -EINVAL;
1370266a177SLong Li 	}
1380266a177SLong Li 
1390266a177SLong Li 	if (attr->cap.max_recv_sge > MAX_RX_WQE_SGL_ENTRIES) {
1400266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
1410266a177SLong Li 			  "Requested max_recv_sge %d exceeding limit\n",
1420266a177SLong Li 			  attr->cap.max_recv_sge);
1430266a177SLong Li 		return -EINVAL;
1440266a177SLong Li 	}
1450266a177SLong Li 
1460266a177SLong Li 	ind_tbl_size = 1 << ind_tbl->log_ind_tbl_size;
1470266a177SLong Li 	if (ind_tbl_size > MANA_INDIRECT_TABLE_SIZE) {
1480266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
1490266a177SLong Li 			  "Indirect table size %d exceeding limit\n",
1500266a177SLong Li 			  ind_tbl_size);
1510266a177SLong Li 		return -EINVAL;
1520266a177SLong Li 	}
1530266a177SLong Li 
1540266a177SLong Li 	if (ucmd.rx_hash_function != MANA_IB_RX_HASH_FUNC_TOEPLITZ) {
1550266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
1560266a177SLong Li 			  "RX Hash function is not supported, %d\n",
1570266a177SLong Li 			  ucmd.rx_hash_function);
1580266a177SLong Li 		return -EINVAL;
1590266a177SLong Li 	}
1600266a177SLong Li 
1610266a177SLong Li 	/* IB ports start with 1, MANA start with 0 */
1620266a177SLong Li 	port = ucmd.port;
1630266a177SLong Li 	if (port < 1 || port > mc->num_ports) {
1640266a177SLong Li 		ibdev_dbg(&mdev->ib_dev, "Invalid port %u in creating qp\n",
1650266a177SLong Li 			  port);
1660266a177SLong Li 		return -EINVAL;
1670266a177SLong Li 	}
1680266a177SLong Li 	ndev = mc->ports[port - 1];
1690266a177SLong Li 	mpc = netdev_priv(ndev);
1700266a177SLong Li 
1710266a177SLong Li 	ibdev_dbg(&mdev->ib_dev, "rx_hash_function %d port %d\n",
1720266a177SLong Li 		  ucmd.rx_hash_function, port);
1730266a177SLong Li 
1740266a177SLong Li 	mana_ind_table = kcalloc(ind_tbl_size, sizeof(mana_handle_t),
1750266a177SLong Li 				 GFP_KERNEL);
1760266a177SLong Li 	if (!mana_ind_table) {
1770266a177SLong Li 		ret = -ENOMEM;
1780266a177SLong Li 		goto fail;
1790266a177SLong Li 	}
1800266a177SLong Li 
1810266a177SLong Li 	qp->port = port;
1820266a177SLong Li 
1830266a177SLong Li 	for (i = 0; i < ind_tbl_size; i++) {
1840266a177SLong Li 		struct mana_obj_spec wq_spec = {};
1850266a177SLong Li 		struct mana_obj_spec cq_spec = {};
1860266a177SLong Li 
1870266a177SLong Li 		ibwq = ind_tbl->ind_tbl[i];
1880266a177SLong Li 		wq = container_of(ibwq, struct mana_ib_wq, ibwq);
1890266a177SLong Li 
1900266a177SLong Li 		ibcq = ibwq->cq;
1910266a177SLong Li 		cq = container_of(ibcq, struct mana_ib_cq, ibcq);
1920266a177SLong Li 
1930266a177SLong Li 		wq_spec.gdma_region = wq->gdma_region;
1940266a177SLong Li 		wq_spec.queue_size = wq->wq_buf_size;
1950266a177SLong Li 
1960266a177SLong Li 		cq_spec.gdma_region = cq->gdma_region;
1970266a177SLong Li 		cq_spec.queue_size = cq->cqe * COMP_ENTRY_SIZE;
1980266a177SLong Li 		cq_spec.modr_ctx_id = 0;
1990266a177SLong Li 		cq_spec.attached_eq = GDMA_CQ_NO_EQ;
2000266a177SLong Li 
2010266a177SLong Li 		ret = mana_create_wq_obj(mpc, mpc->port_handle, GDMA_RQ,
2020266a177SLong Li 					 &wq_spec, &cq_spec, &wq->rx_object);
2030266a177SLong Li 		if (ret)
2040266a177SLong Li 			goto fail;
2050266a177SLong Li 
2060266a177SLong Li 		/* The GDMA regions are now owned by the WQ object */
2070266a177SLong Li 		wq->gdma_region = GDMA_INVALID_DMA_REGION;
2080266a177SLong Li 		cq->gdma_region = GDMA_INVALID_DMA_REGION;
2090266a177SLong Li 
2100266a177SLong Li 		wq->id = wq_spec.queue_index;
2110266a177SLong Li 		cq->id = cq_spec.queue_index;
2120266a177SLong Li 
2130266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
2140266a177SLong Li 			  "ret %d rx_object 0x%llx wq id %llu cq id %llu\n",
2150266a177SLong Li 			  ret, wq->rx_object, wq->id, cq->id);
2160266a177SLong Li 
2170266a177SLong Li 		resp.entries[i].cqid = cq->id;
2180266a177SLong Li 		resp.entries[i].wqid = wq->id;
2190266a177SLong Li 
2200266a177SLong Li 		mana_ind_table[i] = wq->rx_object;
2210266a177SLong Li 	}
2220266a177SLong Li 	resp.num_entries = i;
2230266a177SLong Li 
2240266a177SLong Li 	ret = mana_ib_cfg_vport_steering(mdev, ndev, wq->rx_object,
2250266a177SLong Li 					 mana_ind_table,
2260266a177SLong Li 					 ind_tbl->log_ind_tbl_size,
2270266a177SLong Li 					 ucmd.rx_hash_key_len,
2280266a177SLong Li 					 ucmd.rx_hash_key);
2290266a177SLong Li 	if (ret)
2300266a177SLong Li 		goto fail;
2310266a177SLong Li 
2320266a177SLong Li 	ret = ib_copy_to_udata(udata, &resp, sizeof(resp));
2330266a177SLong Li 	if (ret) {
2340266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
2350266a177SLong Li 			  "Failed to copy to udata create rss-qp, %d\n",
2360266a177SLong Li 			  ret);
2370266a177SLong Li 		goto fail;
2380266a177SLong Li 	}
2390266a177SLong Li 
2400266a177SLong Li 	kfree(mana_ind_table);
2410266a177SLong Li 
2420266a177SLong Li 	return 0;
2430266a177SLong Li 
2440266a177SLong Li fail:
2450266a177SLong Li 	while (i-- > 0) {
2460266a177SLong Li 		ibwq = ind_tbl->ind_tbl[i];
2470266a177SLong Li 		wq = container_of(ibwq, struct mana_ib_wq, ibwq);
2480266a177SLong Li 		mana_destroy_wq_obj(mpc, GDMA_RQ, wq->rx_object);
2490266a177SLong Li 	}
2500266a177SLong Li 
2510266a177SLong Li 	kfree(mana_ind_table);
2520266a177SLong Li 
2530266a177SLong Li 	return ret;
2540266a177SLong Li }
2550266a177SLong Li 
mana_ib_create_qp_raw(struct ib_qp * ibqp,struct ib_pd * ibpd,struct ib_qp_init_attr * attr,struct ib_udata * udata)2560266a177SLong Li static int mana_ib_create_qp_raw(struct ib_qp *ibqp, struct ib_pd *ibpd,
2570266a177SLong Li 				 struct ib_qp_init_attr *attr,
2580266a177SLong Li 				 struct ib_udata *udata)
2590266a177SLong Li {
2600266a177SLong Li 	struct mana_ib_pd *pd = container_of(ibpd, struct mana_ib_pd, ibpd);
2610266a177SLong Li 	struct mana_ib_qp *qp = container_of(ibqp, struct mana_ib_qp, ibqp);
2620266a177SLong Li 	struct mana_ib_dev *mdev =
2630266a177SLong Li 		container_of(ibpd->device, struct mana_ib_dev, ib_dev);
2640266a177SLong Li 	struct mana_ib_cq *send_cq =
2650266a177SLong Li 		container_of(attr->send_cq, struct mana_ib_cq, ibcq);
2660266a177SLong Li 	struct mana_ib_ucontext *mana_ucontext =
2670266a177SLong Li 		rdma_udata_to_drv_context(udata, struct mana_ib_ucontext,
2680266a177SLong Li 					  ibucontext);
2690266a177SLong Li 	struct mana_ib_create_qp_resp resp = {};
2700266a177SLong Li 	struct gdma_dev *gd = mdev->gdma_dev;
2710266a177SLong Li 	struct mana_ib_create_qp ucmd = {};
2720266a177SLong Li 	struct mana_obj_spec wq_spec = {};
2730266a177SLong Li 	struct mana_obj_spec cq_spec = {};
2740266a177SLong Li 	struct mana_port_context *mpc;
2750266a177SLong Li 	struct mana_context *mc;
2760266a177SLong Li 	struct net_device *ndev;
2770266a177SLong Li 	struct ib_umem *umem;
2780266a177SLong Li 	int err;
2790266a177SLong Li 	u32 port;
2800266a177SLong Li 
2810266a177SLong Li 	mc = gd->driver_data;
2820266a177SLong Li 
2830266a177SLong Li 	if (!mana_ucontext || udata->inlen < sizeof(ucmd))
2840266a177SLong Li 		return -EINVAL;
2850266a177SLong Li 
2860266a177SLong Li 	err = ib_copy_from_udata(&ucmd, udata, min(sizeof(ucmd), udata->inlen));
2870266a177SLong Li 	if (err) {
2880266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
2890266a177SLong Li 			  "Failed to copy from udata create qp-raw, %d\n", err);
2900266a177SLong Li 		return err;
2910266a177SLong Li 	}
2920266a177SLong Li 
2930266a177SLong Li 	/* IB ports start with 1, MANA Ethernet ports start with 0 */
2940266a177SLong Li 	port = ucmd.port;
295563ca0e9SDan Carpenter 	if (port < 1 || port > mc->num_ports)
2960266a177SLong Li 		return -EINVAL;
2970266a177SLong Li 
2980266a177SLong Li 	if (attr->cap.max_send_wr > MAX_SEND_BUFFERS_PER_QUEUE) {
2990266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3000266a177SLong Li 			  "Requested max_send_wr %d exceeding limit\n",
3010266a177SLong Li 			  attr->cap.max_send_wr);
3020266a177SLong Li 		return -EINVAL;
3030266a177SLong Li 	}
3040266a177SLong Li 
3050266a177SLong Li 	if (attr->cap.max_send_sge > MAX_TX_WQE_SGL_ENTRIES) {
3060266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3070266a177SLong Li 			  "Requested max_send_sge %d exceeding limit\n",
3080266a177SLong Li 			  attr->cap.max_send_sge);
3090266a177SLong Li 		return -EINVAL;
3100266a177SLong Li 	}
3110266a177SLong Li 
3120266a177SLong Li 	ndev = mc->ports[port - 1];
3130266a177SLong Li 	mpc = netdev_priv(ndev);
3140266a177SLong Li 	ibdev_dbg(&mdev->ib_dev, "port %u ndev %p mpc %p\n", port, ndev, mpc);
3150266a177SLong Li 
3160266a177SLong Li 	err = mana_ib_cfg_vport(mdev, port - 1, pd, mana_ucontext->doorbell);
3170266a177SLong Li 	if (err)
3180266a177SLong Li 		return -ENODEV;
3190266a177SLong Li 
3200266a177SLong Li 	qp->port = port;
3210266a177SLong Li 
3220266a177SLong Li 	ibdev_dbg(&mdev->ib_dev, "ucmd sq_buf_addr 0x%llx port %u\n",
3230266a177SLong Li 		  ucmd.sq_buf_addr, ucmd.port);
3240266a177SLong Li 
3250266a177SLong Li 	umem = ib_umem_get(ibpd->device, ucmd.sq_buf_addr, ucmd.sq_buf_size,
3260266a177SLong Li 			   IB_ACCESS_LOCAL_WRITE);
3270266a177SLong Li 	if (IS_ERR(umem)) {
3280266a177SLong Li 		err = PTR_ERR(umem);
3290266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3300266a177SLong Li 			  "Failed to get umem for create qp-raw, err %d\n",
3310266a177SLong Li 			  err);
3320266a177SLong Li 		goto err_free_vport;
3330266a177SLong Li 	}
3340266a177SLong Li 	qp->sq_umem = umem;
3350266a177SLong Li 
3360266a177SLong Li 	err = mana_ib_gd_create_dma_region(mdev, qp->sq_umem,
3370266a177SLong Li 					   &qp->sq_gdma_region);
3380266a177SLong Li 	if (err) {
3390266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3400266a177SLong Li 			  "Failed to create dma region for create qp-raw, %d\n",
3410266a177SLong Li 			  err);
3420266a177SLong Li 		goto err_release_umem;
3430266a177SLong Li 	}
3440266a177SLong Li 
3450266a177SLong Li 	ibdev_dbg(&mdev->ib_dev,
3460266a177SLong Li 		  "mana_ib_gd_create_dma_region ret %d gdma_region 0x%llx\n",
3470266a177SLong Li 		  err, qp->sq_gdma_region);
3480266a177SLong Li 
3490266a177SLong Li 	/* Create a WQ on the same port handle used by the Ethernet */
3500266a177SLong Li 	wq_spec.gdma_region = qp->sq_gdma_region;
3510266a177SLong Li 	wq_spec.queue_size = ucmd.sq_buf_size;
3520266a177SLong Li 
3530266a177SLong Li 	cq_spec.gdma_region = send_cq->gdma_region;
3540266a177SLong Li 	cq_spec.queue_size = send_cq->cqe * COMP_ENTRY_SIZE;
3550266a177SLong Li 	cq_spec.modr_ctx_id = 0;
3560266a177SLong Li 	cq_spec.attached_eq = GDMA_CQ_NO_EQ;
3570266a177SLong Li 
3580266a177SLong Li 	err = mana_create_wq_obj(mpc, mpc->port_handle, GDMA_SQ, &wq_spec,
3590266a177SLong Li 				 &cq_spec, &qp->tx_object);
3600266a177SLong Li 	if (err) {
3610266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3620266a177SLong Li 			  "Failed to create wq for create raw-qp, err %d\n",
3630266a177SLong Li 			  err);
3640266a177SLong Li 		goto err_destroy_dma_region;
3650266a177SLong Li 	}
3660266a177SLong Li 
3670266a177SLong Li 	/* The GDMA regions are now owned by the WQ object */
3680266a177SLong Li 	qp->sq_gdma_region = GDMA_INVALID_DMA_REGION;
3690266a177SLong Li 	send_cq->gdma_region = GDMA_INVALID_DMA_REGION;
3700266a177SLong Li 
3710266a177SLong Li 	qp->sq_id = wq_spec.queue_index;
3720266a177SLong Li 	send_cq->id = cq_spec.queue_index;
3730266a177SLong Li 
3740266a177SLong Li 	ibdev_dbg(&mdev->ib_dev,
3750266a177SLong Li 		  "ret %d qp->tx_object 0x%llx sq id %llu cq id %llu\n", err,
3760266a177SLong Li 		  qp->tx_object, qp->sq_id, send_cq->id);
3770266a177SLong Li 
3780266a177SLong Li 	resp.sqid = qp->sq_id;
3790266a177SLong Li 	resp.cqid = send_cq->id;
3800266a177SLong Li 	resp.tx_vp_offset = pd->tx_vp_offset;
3810266a177SLong Li 
3820266a177SLong Li 	err = ib_copy_to_udata(udata, &resp, sizeof(resp));
3830266a177SLong Li 	if (err) {
3840266a177SLong Li 		ibdev_dbg(&mdev->ib_dev,
3850266a177SLong Li 			  "Failed copy udata for create qp-raw, %d\n",
3860266a177SLong Li 			  err);
3870266a177SLong Li 		goto err_destroy_wq_obj;
3880266a177SLong Li 	}
3890266a177SLong Li 
3900266a177SLong Li 	return 0;
3910266a177SLong Li 
3920266a177SLong Li err_destroy_wq_obj:
3930266a177SLong Li 	mana_destroy_wq_obj(mpc, GDMA_SQ, qp->tx_object);
3940266a177SLong Li 
3950266a177SLong Li err_destroy_dma_region:
3960266a177SLong Li 	mana_ib_gd_destroy_dma_region(mdev, qp->sq_gdma_region);
3970266a177SLong Li 
3980266a177SLong Li err_release_umem:
3990266a177SLong Li 	ib_umem_release(umem);
4000266a177SLong Li 
4010266a177SLong Li err_free_vport:
4020266a177SLong Li 	mana_ib_uncfg_vport(mdev, pd, port - 1);
4030266a177SLong Li 
4040266a177SLong Li 	return err;
4050266a177SLong Li }
4060266a177SLong Li 
mana_ib_create_qp(struct ib_qp * ibqp,struct ib_qp_init_attr * attr,struct ib_udata * udata)4070266a177SLong Li int mana_ib_create_qp(struct ib_qp *ibqp, struct ib_qp_init_attr *attr,
4080266a177SLong Li 		      struct ib_udata *udata)
4090266a177SLong Li {
4100266a177SLong Li 	switch (attr->qp_type) {
4110266a177SLong Li 	case IB_QPT_RAW_PACKET:
4120266a177SLong Li 		/* When rwq_ind_tbl is used, it's for creating WQs for RSS */
4130266a177SLong Li 		if (attr->rwq_ind_tbl)
4140266a177SLong Li 			return mana_ib_create_qp_rss(ibqp, ibqp->pd, attr,
4150266a177SLong Li 						     udata);
4160266a177SLong Li 
4170266a177SLong Li 		return mana_ib_create_qp_raw(ibqp, ibqp->pd, attr, udata);
4180266a177SLong Li 	default:
4190266a177SLong Li 		/* Creating QP other than IB_QPT_RAW_PACKET is not supported */
4200266a177SLong Li 		ibdev_dbg(ibqp->device, "Creating QP type %u not supported\n",
4210266a177SLong Li 			  attr->qp_type);
4220266a177SLong Li 	}
4230266a177SLong Li 
4240266a177SLong Li 	return -EINVAL;
4250266a177SLong Li }
4260266a177SLong Li 
mana_ib_modify_qp(struct ib_qp * ibqp,struct ib_qp_attr * attr,int attr_mask,struct ib_udata * udata)4270266a177SLong Li int mana_ib_modify_qp(struct ib_qp *ibqp, struct ib_qp_attr *attr,
4280266a177SLong Li 		      int attr_mask, struct ib_udata *udata)
4290266a177SLong Li {
4300266a177SLong Li 	/* modify_qp is not supported by this version of the driver */
4310266a177SLong Li 	return -EOPNOTSUPP;
4320266a177SLong Li }
4330266a177SLong Li 
mana_ib_destroy_qp_rss(struct mana_ib_qp * qp,struct ib_rwq_ind_table * ind_tbl,struct ib_udata * udata)4340266a177SLong Li static int mana_ib_destroy_qp_rss(struct mana_ib_qp *qp,
4350266a177SLong Li 				  struct ib_rwq_ind_table *ind_tbl,
4360266a177SLong Li 				  struct ib_udata *udata)
4370266a177SLong Li {
4380266a177SLong Li 	struct mana_ib_dev *mdev =
4390266a177SLong Li 		container_of(qp->ibqp.device, struct mana_ib_dev, ib_dev);
4400266a177SLong Li 	struct gdma_dev *gd = mdev->gdma_dev;
4410266a177SLong Li 	struct mana_port_context *mpc;
4420266a177SLong Li 	struct mana_context *mc;
4430266a177SLong Li 	struct net_device *ndev;
4440266a177SLong Li 	struct mana_ib_wq *wq;
4450266a177SLong Li 	struct ib_wq *ibwq;
4460266a177SLong Li 	int i;
4470266a177SLong Li 
4480266a177SLong Li 	mc = gd->driver_data;
4490266a177SLong Li 	ndev = mc->ports[qp->port - 1];
4500266a177SLong Li 	mpc = netdev_priv(ndev);
4510266a177SLong Li 
4520266a177SLong Li 	for (i = 0; i < (1 << ind_tbl->log_ind_tbl_size); i++) {
4530266a177SLong Li 		ibwq = ind_tbl->ind_tbl[i];
4540266a177SLong Li 		wq = container_of(ibwq, struct mana_ib_wq, ibwq);
4550266a177SLong Li 		ibdev_dbg(&mdev->ib_dev, "destroying wq->rx_object %llu\n",
4560266a177SLong Li 			  wq->rx_object);
4570266a177SLong Li 		mana_destroy_wq_obj(mpc, GDMA_RQ, wq->rx_object);
4580266a177SLong Li 	}
4590266a177SLong Li 
4600266a177SLong Li 	return 0;
4610266a177SLong Li }
4620266a177SLong Li 
mana_ib_destroy_qp_raw(struct mana_ib_qp * qp,struct ib_udata * udata)4630266a177SLong Li static int mana_ib_destroy_qp_raw(struct mana_ib_qp *qp, struct ib_udata *udata)
4640266a177SLong Li {
4650266a177SLong Li 	struct mana_ib_dev *mdev =
4660266a177SLong Li 		container_of(qp->ibqp.device, struct mana_ib_dev, ib_dev);
4670266a177SLong Li 	struct gdma_dev *gd = mdev->gdma_dev;
4680266a177SLong Li 	struct ib_pd *ibpd = qp->ibqp.pd;
4690266a177SLong Li 	struct mana_port_context *mpc;
4700266a177SLong Li 	struct mana_context *mc;
4710266a177SLong Li 	struct net_device *ndev;
4720266a177SLong Li 	struct mana_ib_pd *pd;
4730266a177SLong Li 
4740266a177SLong Li 	mc = gd->driver_data;
4750266a177SLong Li 	ndev = mc->ports[qp->port - 1];
4760266a177SLong Li 	mpc = netdev_priv(ndev);
4770266a177SLong Li 	pd = container_of(ibpd, struct mana_ib_pd, ibpd);
4780266a177SLong Li 
4790266a177SLong Li 	mana_destroy_wq_obj(mpc, GDMA_SQ, qp->tx_object);
4800266a177SLong Li 
4810266a177SLong Li 	if (qp->sq_umem) {
4820266a177SLong Li 		mana_ib_gd_destroy_dma_region(mdev, qp->sq_gdma_region);
4830266a177SLong Li 		ib_umem_release(qp->sq_umem);
4840266a177SLong Li 	}
4850266a177SLong Li 
4860266a177SLong Li 	mana_ib_uncfg_vport(mdev, pd, qp->port - 1);
4870266a177SLong Li 
4880266a177SLong Li 	return 0;
4890266a177SLong Li }
4900266a177SLong Li 
mana_ib_destroy_qp(struct ib_qp * ibqp,struct ib_udata * udata)4910266a177SLong Li int mana_ib_destroy_qp(struct ib_qp *ibqp, struct ib_udata *udata)
4920266a177SLong Li {
4930266a177SLong Li 	struct mana_ib_qp *qp = container_of(ibqp, struct mana_ib_qp, ibqp);
4940266a177SLong Li 
4950266a177SLong Li 	switch (ibqp->qp_type) {
4960266a177SLong Li 	case IB_QPT_RAW_PACKET:
4970266a177SLong Li 		if (ibqp->rwq_ind_tbl)
4980266a177SLong Li 			return mana_ib_destroy_qp_rss(qp, ibqp->rwq_ind_tbl,
4990266a177SLong Li 						      udata);
5000266a177SLong Li 
5010266a177SLong Li 		return mana_ib_destroy_qp_raw(qp, udata);
5020266a177SLong Li 
5030266a177SLong Li 	default:
5040266a177SLong Li 		ibdev_dbg(ibqp->device, "Unexpected QP type %u\n",
5050266a177SLong Li 			  ibqp->qp_type);
5060266a177SLong Li 	}
5070266a177SLong Li 
5080266a177SLong Li 	return -ENOENT;
5090266a177SLong Li }
510