xref: /dpdk/drivers/vdpa/sfc/sfc_vdpa_hw.c (revision 4ea22410ea9c0235db40313bb810582b25e845b3)
16dad9a73SVijay Kumar Srivastava /* SPDX-License-Identifier: BSD-3-Clause
26dad9a73SVijay Kumar Srivastava  * Copyright(c) 2020-2021 Xilinx, Inc.
36dad9a73SVijay Kumar Srivastava  */
46dad9a73SVijay Kumar Srivastava 
56dad9a73SVijay Kumar Srivastava #include <unistd.h>
66dad9a73SVijay Kumar Srivastava 
76dad9a73SVijay Kumar Srivastava #include <rte_common.h>
86dad9a73SVijay Kumar Srivastava #include <rte_errno.h>
96dad9a73SVijay Kumar Srivastava #include <rte_vfio.h>
10b1196136SVijay Kumar Srivastava #include <rte_vhost.h>
116dad9a73SVijay Kumar Srivastava 
126dad9a73SVijay Kumar Srivastava #include "efx.h"
136dad9a73SVijay Kumar Srivastava #include "sfc_vdpa.h"
146dad9a73SVijay Kumar Srivastava #include "sfc_vdpa_ops.h"
156dad9a73SVijay Kumar Srivastava 
166dad9a73SVijay Kumar Srivastava #ifndef PAGE_SIZE
176dad9a73SVijay Kumar Srivastava #define PAGE_SIZE   (sysconf(_SC_PAGESIZE))
186dad9a73SVijay Kumar Srivastava #endif
196dad9a73SVijay Kumar Srivastava 
206dad9a73SVijay Kumar Srivastava int
sfc_vdpa_dma_alloc(struct sfc_vdpa_adapter * sva,const char * name,size_t len,efsys_mem_t * esmp)216dad9a73SVijay Kumar Srivastava sfc_vdpa_dma_alloc(struct sfc_vdpa_adapter *sva, const char *name,
226dad9a73SVijay Kumar Srivastava 		   size_t len, efsys_mem_t *esmp)
236dad9a73SVijay Kumar Srivastava {
246dad9a73SVijay Kumar Srivastava 	uint64_t mcdi_iova;
256dad9a73SVijay Kumar Srivastava 	size_t mcdi_buff_size;
26b8162dbeSAbhimanyu Saini 	char mz_name[RTE_MEMZONE_NAMESIZE];
276dad9a73SVijay Kumar Srivastava 	const struct rte_memzone *mz = NULL;
286dad9a73SVijay Kumar Srivastava 	int numa_node = sva->pdev->device.numa_node;
296dad9a73SVijay Kumar Srivastava 	int ret;
306dad9a73SVijay Kumar Srivastava 
316dad9a73SVijay Kumar Srivastava 	mcdi_buff_size = RTE_ALIGN_CEIL(len, PAGE_SIZE);
32b8162dbeSAbhimanyu Saini 	ret = snprintf(mz_name, RTE_MEMZONE_NAMESIZE, "%s_%s",
33b8162dbeSAbhimanyu Saini 		       sva->pdev->name, name);
34b8162dbeSAbhimanyu Saini 	if (ret < 0 || ret >= RTE_MEMZONE_NAMESIZE) {
35b8162dbeSAbhimanyu Saini 		sfc_vdpa_err(sva, "%s_%s too long to fit in mz_name",
36b8162dbeSAbhimanyu Saini 			     sva->pdev->name, name);
37b8162dbeSAbhimanyu Saini 		return -EINVAL;
38b8162dbeSAbhimanyu Saini 	}
396dad9a73SVijay Kumar Srivastava 
40b8162dbeSAbhimanyu Saini 	sfc_vdpa_log_init(sva, "name=%s, len=%zu", mz_name, len);
416dad9a73SVijay Kumar Srivastava 
42b8162dbeSAbhimanyu Saini 	mz = rte_memzone_reserve_aligned(mz_name, mcdi_buff_size,
436dad9a73SVijay Kumar Srivastava 					 numa_node,
446dad9a73SVijay Kumar Srivastava 					 RTE_MEMZONE_IOVA_CONTIG,
456dad9a73SVijay Kumar Srivastava 					 PAGE_SIZE);
466dad9a73SVijay Kumar Srivastava 	if (mz == NULL) {
476dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "cannot reserve memory for %s: len=%#x: %s",
48b8162dbeSAbhimanyu Saini 			     mz_name, (unsigned int)len,
49b8162dbeSAbhimanyu Saini 			     rte_strerror(rte_errno));
506dad9a73SVijay Kumar Srivastava 		return -ENOMEM;
516dad9a73SVijay Kumar Srivastava 	}
526dad9a73SVijay Kumar Srivastava 
536dad9a73SVijay Kumar Srivastava 	/* IOVA address for MCDI would be re-calculated if mapping
546dad9a73SVijay Kumar Srivastava 	 * using default IOVA would fail.
556dad9a73SVijay Kumar Srivastava 	 * TODO: Earlier there was no way to get valid IOVA range.
566dad9a73SVijay Kumar Srivastava 	 * Recently a patch has been submitted to get the IOVA range
576dad9a73SVijay Kumar Srivastava 	 * using ioctl. VFIO_IOMMU_GET_INFO. This patch is available
586dad9a73SVijay Kumar Srivastava 	 * in the kernel version >= 5.4. Support to get the default
596dad9a73SVijay Kumar Srivastava 	 * IOVA address for MCDI buffer using available IOVA range
606dad9a73SVijay Kumar Srivastava 	 * would be added later. Meanwhile default IOVA for MCDI buffer
616dad9a73SVijay Kumar Srivastava 	 * is kept at high mem at 2TB. In case of overlap new available
626dad9a73SVijay Kumar Srivastava 	 * addresses would be searched and same would be used.
636dad9a73SVijay Kumar Srivastava 	 */
646dad9a73SVijay Kumar Srivastava 	mcdi_iova = SFC_VDPA_DEFAULT_MCDI_IOVA;
656dad9a73SVijay Kumar Srivastava 
666dad9a73SVijay Kumar Srivastava 	for (;;) {
676dad9a73SVijay Kumar Srivastava 		ret = rte_vfio_container_dma_map(sva->vfio_container_fd,
686dad9a73SVijay Kumar Srivastava 						 (uint64_t)mz->addr, mcdi_iova,
696dad9a73SVijay Kumar Srivastava 						 mcdi_buff_size);
706dad9a73SVijay Kumar Srivastava 		if (ret == 0)
716dad9a73SVijay Kumar Srivastava 			break;
726dad9a73SVijay Kumar Srivastava 
736dad9a73SVijay Kumar Srivastava 		mcdi_iova = mcdi_iova >> 1;
746dad9a73SVijay Kumar Srivastava 		if (mcdi_iova < mcdi_buff_size)	{
756dad9a73SVijay Kumar Srivastava 			sfc_vdpa_err(sva,
766dad9a73SVijay Kumar Srivastava 				     "DMA mapping failed for MCDI : %s",
776dad9a73SVijay Kumar Srivastava 				     rte_strerror(rte_errno));
786dad9a73SVijay Kumar Srivastava 			rte_memzone_free(mz);
796dad9a73SVijay Kumar Srivastava 			return ret;
806dad9a73SVijay Kumar Srivastava 		}
816dad9a73SVijay Kumar Srivastava 	}
826dad9a73SVijay Kumar Srivastava 
836dad9a73SVijay Kumar Srivastava 	esmp->esm_addr = mcdi_iova;
846dad9a73SVijay Kumar Srivastava 	esmp->esm_base = mz->addr;
856dad9a73SVijay Kumar Srivastava 	sva->mcdi_buff_size = mcdi_buff_size;
866dad9a73SVijay Kumar Srivastava 
876dad9a73SVijay Kumar Srivastava 	sfc_vdpa_info(sva,
886dad9a73SVijay Kumar Srivastava 		      "DMA name=%s len=%zu => virt=%p iova=0x%" PRIx64,
896dad9a73SVijay Kumar Srivastava 		      name, len, esmp->esm_base, esmp->esm_addr);
906dad9a73SVijay Kumar Srivastava 
916dad9a73SVijay Kumar Srivastava 	return 0;
926dad9a73SVijay Kumar Srivastava }
936dad9a73SVijay Kumar Srivastava 
946dad9a73SVijay Kumar Srivastava void
sfc_vdpa_dma_free(struct sfc_vdpa_adapter * sva,efsys_mem_t * esmp)956dad9a73SVijay Kumar Srivastava sfc_vdpa_dma_free(struct sfc_vdpa_adapter *sva, efsys_mem_t *esmp)
966dad9a73SVijay Kumar Srivastava {
976dad9a73SVijay Kumar Srivastava 	int ret;
986dad9a73SVijay Kumar Srivastava 
996dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "name=%s", esmp->esm_mz->name);
1006dad9a73SVijay Kumar Srivastava 
1016dad9a73SVijay Kumar Srivastava 	ret = rte_vfio_container_dma_unmap(sva->vfio_container_fd,
1026dad9a73SVijay Kumar Srivastava 					   (uint64_t)esmp->esm_base,
1036dad9a73SVijay Kumar Srivastava 					   esmp->esm_addr, sva->mcdi_buff_size);
1046dad9a73SVijay Kumar Srivastava 	if (ret < 0)
1056dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "DMA unmap failed for MCDI : %s",
1066dad9a73SVijay Kumar Srivastava 			     rte_strerror(rte_errno));
1076dad9a73SVijay Kumar Srivastava 
1086dad9a73SVijay Kumar Srivastava 	sfc_vdpa_info(sva,
1096dad9a73SVijay Kumar Srivastava 		      "DMA free name=%s => virt=%p iova=0x%" PRIx64,
1106dad9a73SVijay Kumar Srivastava 		      esmp->esm_mz->name, esmp->esm_base, esmp->esm_addr);
1116dad9a73SVijay Kumar Srivastava 
1126dad9a73SVijay Kumar Srivastava 	rte_free((void *)(esmp->esm_base));
1136dad9a73SVijay Kumar Srivastava 
1146dad9a73SVijay Kumar Srivastava 	sva->mcdi_buff_size = 0;
1156dad9a73SVijay Kumar Srivastava 	memset(esmp, 0, sizeof(*esmp));
1166dad9a73SVijay Kumar Srivastava }
1176dad9a73SVijay Kumar Srivastava 
118b1196136SVijay Kumar Srivastava int
sfc_vdpa_dma_map(struct sfc_vdpa_ops_data * ops_data,bool do_map)119b1196136SVijay Kumar Srivastava sfc_vdpa_dma_map(struct sfc_vdpa_ops_data *ops_data, bool do_map)
120b1196136SVijay Kumar Srivastava {
121b1196136SVijay Kumar Srivastava 	uint32_t i, j;
122b1196136SVijay Kumar Srivastava 	int rc;
123b1196136SVijay Kumar Srivastava 	struct rte_vhost_memory *vhost_mem = NULL;
124b1196136SVijay Kumar Srivastava 	struct rte_vhost_mem_region *mem_reg = NULL;
125b1196136SVijay Kumar Srivastava 	int vfio_container_fd;
126b1196136SVijay Kumar Srivastava 	void *dev;
127b1196136SVijay Kumar Srivastava 
128b1196136SVijay Kumar Srivastava 	dev = ops_data->dev_handle;
129b1196136SVijay Kumar Srivastava 	vfio_container_fd =
130b1196136SVijay Kumar Srivastava 		sfc_vdpa_adapter_by_dev_handle(dev)->vfio_container_fd;
131b1196136SVijay Kumar Srivastava 
132b1196136SVijay Kumar Srivastava 	rc = rte_vhost_get_mem_table(ops_data->vid, &vhost_mem);
133b1196136SVijay Kumar Srivastava 	if (rc < 0) {
134b1196136SVijay Kumar Srivastava 		sfc_vdpa_err(dev,
135b1196136SVijay Kumar Srivastava 			     "failed to get VM memory layout");
136b1196136SVijay Kumar Srivastava 		goto error;
137b1196136SVijay Kumar Srivastava 	}
138b1196136SVijay Kumar Srivastava 
139b1196136SVijay Kumar Srivastava 	for (i = 0; i < vhost_mem->nregions; i++) {
140b1196136SVijay Kumar Srivastava 		mem_reg = &vhost_mem->regions[i];
141b1196136SVijay Kumar Srivastava 
142b1196136SVijay Kumar Srivastava 		if (do_map) {
143b1196136SVijay Kumar Srivastava 			rc = rte_vfio_container_dma_map(vfio_container_fd,
144b1196136SVijay Kumar Srivastava 						mem_reg->host_user_addr,
145b1196136SVijay Kumar Srivastava 						mem_reg->guest_phys_addr,
146b1196136SVijay Kumar Srivastava 						mem_reg->size);
147b1196136SVijay Kumar Srivastava 			if (rc < 0) {
148b1196136SVijay Kumar Srivastava 				sfc_vdpa_err(dev,
149b1196136SVijay Kumar Srivastava 					     "DMA map failed : %s",
150b1196136SVijay Kumar Srivastava 					     rte_strerror(rte_errno));
151b1196136SVijay Kumar Srivastava 				goto failed_vfio_dma_map;
152b1196136SVijay Kumar Srivastava 			}
153b1196136SVijay Kumar Srivastava 		} else {
154b1196136SVijay Kumar Srivastava 			rc = rte_vfio_container_dma_unmap(vfio_container_fd,
155b1196136SVijay Kumar Srivastava 						mem_reg->host_user_addr,
156b1196136SVijay Kumar Srivastava 						mem_reg->guest_phys_addr,
157b1196136SVijay Kumar Srivastava 						mem_reg->size);
158b1196136SVijay Kumar Srivastava 			if (rc < 0) {
159b1196136SVijay Kumar Srivastava 				sfc_vdpa_err(dev,
160b1196136SVijay Kumar Srivastava 					     "DMA unmap failed : %s",
161b1196136SVijay Kumar Srivastava 					     rte_strerror(rte_errno));
162b1196136SVijay Kumar Srivastava 				goto error;
163b1196136SVijay Kumar Srivastava 			}
164b1196136SVijay Kumar Srivastava 		}
165b1196136SVijay Kumar Srivastava 	}
166b1196136SVijay Kumar Srivastava 
167b1196136SVijay Kumar Srivastava 	free(vhost_mem);
168b1196136SVijay Kumar Srivastava 
169b1196136SVijay Kumar Srivastava 	return 0;
170b1196136SVijay Kumar Srivastava 
171b1196136SVijay Kumar Srivastava failed_vfio_dma_map:
172b1196136SVijay Kumar Srivastava 	for (j = 0; j < i; j++) {
173b1196136SVijay Kumar Srivastava 		mem_reg = &vhost_mem->regions[j];
174b1196136SVijay Kumar Srivastava 		rte_vfio_container_dma_unmap(vfio_container_fd,
175b1196136SVijay Kumar Srivastava 					     mem_reg->host_user_addr,
176b1196136SVijay Kumar Srivastava 					     mem_reg->guest_phys_addr,
177b1196136SVijay Kumar Srivastava 					     mem_reg->size);
178b1196136SVijay Kumar Srivastava 	}
179b1196136SVijay Kumar Srivastava 
180b1196136SVijay Kumar Srivastava error:
181b1196136SVijay Kumar Srivastava 	free(vhost_mem);
182b1196136SVijay Kumar Srivastava 
183b1196136SVijay Kumar Srivastava 	return rc;
184b1196136SVijay Kumar Srivastava }
185b1196136SVijay Kumar Srivastava 
1866dad9a73SVijay Kumar Srivastava static int
sfc_vdpa_mem_bar_init(struct sfc_vdpa_adapter * sva,const efx_bar_region_t * mem_ebrp)1876dad9a73SVijay Kumar Srivastava sfc_vdpa_mem_bar_init(struct sfc_vdpa_adapter *sva,
1886dad9a73SVijay Kumar Srivastava 		      const efx_bar_region_t *mem_ebrp)
1896dad9a73SVijay Kumar Srivastava {
1906dad9a73SVijay Kumar Srivastava 	struct rte_pci_device *pci_dev = sva->pdev;
1916dad9a73SVijay Kumar Srivastava 	efsys_bar_t *ebp = &sva->mem_bar;
1926dad9a73SVijay Kumar Srivastava 	struct rte_mem_resource *res =
1936dad9a73SVijay Kumar Srivastava 		&pci_dev->mem_resource[mem_ebrp->ebr_index];
1946dad9a73SVijay Kumar Srivastava 
1956dad9a73SVijay Kumar Srivastava 	SFC_BAR_LOCK_INIT(ebp, pci_dev->name);
1966dad9a73SVijay Kumar Srivastava 	ebp->esb_rid = mem_ebrp->ebr_index;
1976dad9a73SVijay Kumar Srivastava 	ebp->esb_dev = pci_dev;
1986dad9a73SVijay Kumar Srivastava 	ebp->esb_base = res->addr;
1996dad9a73SVijay Kumar Srivastava 
2006dad9a73SVijay Kumar Srivastava 	return 0;
2016dad9a73SVijay Kumar Srivastava }
2026dad9a73SVijay Kumar Srivastava 
2036dad9a73SVijay Kumar Srivastava static void
sfc_vdpa_mem_bar_fini(struct sfc_vdpa_adapter * sva)2046dad9a73SVijay Kumar Srivastava sfc_vdpa_mem_bar_fini(struct sfc_vdpa_adapter *sva)
2056dad9a73SVijay Kumar Srivastava {
2066dad9a73SVijay Kumar Srivastava 	efsys_bar_t *ebp = &sva->mem_bar;
2076dad9a73SVijay Kumar Srivastava 
2086dad9a73SVijay Kumar Srivastava 	SFC_BAR_LOCK_DESTROY(ebp);
2096dad9a73SVijay Kumar Srivastava 	memset(ebp, 0, sizeof(*ebp));
2106dad9a73SVijay Kumar Srivastava }
2116dad9a73SVijay Kumar Srivastava 
2126dad9a73SVijay Kumar Srivastava static int
sfc_vdpa_nic_probe(struct sfc_vdpa_adapter * sva)2136dad9a73SVijay Kumar Srivastava sfc_vdpa_nic_probe(struct sfc_vdpa_adapter *sva)
2146dad9a73SVijay Kumar Srivastava {
2156dad9a73SVijay Kumar Srivastava 	efx_nic_t *enp = sva->nic;
2166dad9a73SVijay Kumar Srivastava 	int rc;
2176dad9a73SVijay Kumar Srivastava 
2186dad9a73SVijay Kumar Srivastava 	rc = efx_nic_probe(enp, EFX_FW_VARIANT_DONT_CARE);
2196dad9a73SVijay Kumar Srivastava 	if (rc != 0)
2206dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "nic probe failed: %s", rte_strerror(rc));
2216dad9a73SVijay Kumar Srivastava 
2226dad9a73SVijay Kumar Srivastava 	return rc;
2236dad9a73SVijay Kumar Srivastava }
2246dad9a73SVijay Kumar Srivastava 
2256dad9a73SVijay Kumar Srivastava static int
sfc_vdpa_estimate_resource_limits(struct sfc_vdpa_adapter * sva)2266dad9a73SVijay Kumar Srivastava sfc_vdpa_estimate_resource_limits(struct sfc_vdpa_adapter *sva)
2276dad9a73SVijay Kumar Srivastava {
2286dad9a73SVijay Kumar Srivastava 	efx_drv_limits_t limits;
2296dad9a73SVijay Kumar Srivastava 	int rc;
2306dad9a73SVijay Kumar Srivastava 	uint32_t evq_allocated;
2316dad9a73SVijay Kumar Srivastava 	uint32_t rxq_allocated;
2326dad9a73SVijay Kumar Srivastava 	uint32_t txq_allocated;
2336dad9a73SVijay Kumar Srivastava 	uint32_t max_queue_cnt;
2346dad9a73SVijay Kumar Srivastava 
2356dad9a73SVijay Kumar Srivastava 	memset(&limits, 0, sizeof(limits));
2366dad9a73SVijay Kumar Srivastava 
2376dad9a73SVijay Kumar Srivastava 	/* Request at least one Rx and Tx queue */
2386dad9a73SVijay Kumar Srivastava 	limits.edl_min_rxq_count = 1;
2396dad9a73SVijay Kumar Srivastava 	limits.edl_min_txq_count = 1;
2406dad9a73SVijay Kumar Srivastava 	/* Management event queue plus event queue for Tx/Rx queue */
2416dad9a73SVijay Kumar Srivastava 	limits.edl_min_evq_count =
2426dad9a73SVijay Kumar Srivastava 		1 + RTE_MAX(limits.edl_min_rxq_count, limits.edl_min_txq_count);
2436dad9a73SVijay Kumar Srivastava 
2446dad9a73SVijay Kumar Srivastava 	limits.edl_max_rxq_count = SFC_VDPA_MAX_QUEUE_PAIRS;
2456dad9a73SVijay Kumar Srivastava 	limits.edl_max_txq_count = SFC_VDPA_MAX_QUEUE_PAIRS;
2466dad9a73SVijay Kumar Srivastava 	limits.edl_max_evq_count = 1 + SFC_VDPA_MAX_QUEUE_PAIRS;
2476dad9a73SVijay Kumar Srivastava 
2486dad9a73SVijay Kumar Srivastava 	SFC_VDPA_ASSERT(limits.edl_max_evq_count >= limits.edl_min_rxq_count);
2496dad9a73SVijay Kumar Srivastava 	SFC_VDPA_ASSERT(limits.edl_max_rxq_count >= limits.edl_min_rxq_count);
2506dad9a73SVijay Kumar Srivastava 	SFC_VDPA_ASSERT(limits.edl_max_txq_count >= limits.edl_min_rxq_count);
2516dad9a73SVijay Kumar Srivastava 
2526dad9a73SVijay Kumar Srivastava 	/* Configure the minimum required resources needed for the
2536dad9a73SVijay Kumar Srivastava 	 * driver to operate, and the maximum desired resources that the
2546dad9a73SVijay Kumar Srivastava 	 * driver is capable of using.
2556dad9a73SVijay Kumar Srivastava 	 */
2566dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "set drv limit");
2576dad9a73SVijay Kumar Srivastava 	efx_nic_set_drv_limits(sva->nic, &limits);
2586dad9a73SVijay Kumar Srivastava 
2596dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "init nic");
2606dad9a73SVijay Kumar Srivastava 	rc = efx_nic_init(sva->nic);
2616dad9a73SVijay Kumar Srivastava 	if (rc != 0) {
2626dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "nic init failed: %s", rte_strerror(rc));
2636dad9a73SVijay Kumar Srivastava 		goto fail_nic_init;
2646dad9a73SVijay Kumar Srivastava 	}
2656dad9a73SVijay Kumar Srivastava 
2666dad9a73SVijay Kumar Srivastava 	/* Find resource dimensions assigned by firmware to this function */
2676dad9a73SVijay Kumar Srivastava 	rc = efx_nic_get_vi_pool(sva->nic, &evq_allocated, &rxq_allocated,
2686dad9a73SVijay Kumar Srivastava 				 &txq_allocated);
2696dad9a73SVijay Kumar Srivastava 	if (rc != 0) {
2706dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "vi pool get failed: %s", rte_strerror(rc));
2716dad9a73SVijay Kumar Srivastava 		goto fail_get_vi_pool;
2726dad9a73SVijay Kumar Srivastava 	}
2736dad9a73SVijay Kumar Srivastava 
2746dad9a73SVijay Kumar Srivastava 	/* It still may allocate more than maximum, ensure limit */
2756dad9a73SVijay Kumar Srivastava 	evq_allocated = RTE_MIN(evq_allocated, limits.edl_max_evq_count);
2766dad9a73SVijay Kumar Srivastava 	rxq_allocated = RTE_MIN(rxq_allocated, limits.edl_max_rxq_count);
2776dad9a73SVijay Kumar Srivastava 	txq_allocated = RTE_MIN(txq_allocated, limits.edl_max_txq_count);
2786dad9a73SVijay Kumar Srivastava 
2796dad9a73SVijay Kumar Srivastava 
2806dad9a73SVijay Kumar Srivastava 	max_queue_cnt = RTE_MIN(rxq_allocated, txq_allocated);
2816dad9a73SVijay Kumar Srivastava 	/* Subtract management EVQ not used for traffic */
2826dad9a73SVijay Kumar Srivastava 	max_queue_cnt = RTE_MIN(evq_allocated - 1, max_queue_cnt);
2836dad9a73SVijay Kumar Srivastava 
2846dad9a73SVijay Kumar Srivastava 	SFC_VDPA_ASSERT(max_queue_cnt > 0);
2856dad9a73SVijay Kumar Srivastava 
2866dad9a73SVijay Kumar Srivastava 	sva->max_queue_count = max_queue_cnt;
287*f2c0e634SAbhimanyu Saini 	sfc_vdpa_log_init(sva, "NIC init done with %u pair(s) of queues",
288*f2c0e634SAbhimanyu Saini 			  max_queue_cnt);
2896dad9a73SVijay Kumar Srivastava 
2906dad9a73SVijay Kumar Srivastava 	return 0;
2916dad9a73SVijay Kumar Srivastava 
2926dad9a73SVijay Kumar Srivastava fail_get_vi_pool:
2936dad9a73SVijay Kumar Srivastava 	efx_nic_fini(sva->nic);
2946dad9a73SVijay Kumar Srivastava fail_nic_init:
2956dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "failed: %s", rte_strerror(rc));
2966dad9a73SVijay Kumar Srivastava 	return rc;
2976dad9a73SVijay Kumar Srivastava }
2986dad9a73SVijay Kumar Srivastava 
2996dad9a73SVijay Kumar Srivastava int
sfc_vdpa_hw_init(struct sfc_vdpa_adapter * sva)3006dad9a73SVijay Kumar Srivastava sfc_vdpa_hw_init(struct sfc_vdpa_adapter *sva)
3016dad9a73SVijay Kumar Srivastava {
3026dad9a73SVijay Kumar Srivastava 	efx_bar_region_t mem_ebr;
3036dad9a73SVijay Kumar Srivastava 	efx_nic_t *enp;
3046dad9a73SVijay Kumar Srivastava 	int rc;
3056dad9a73SVijay Kumar Srivastava 
3066dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "entry");
3076dad9a73SVijay Kumar Srivastava 
3086dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "get family");
3096dad9a73SVijay Kumar Srivastava 	rc = sfc_efx_family(sva->pdev, &mem_ebr, &sva->family);
3106dad9a73SVijay Kumar Srivastava 	if (rc != 0)
3116dad9a73SVijay Kumar Srivastava 		goto fail_family;
3126dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva,
3136dad9a73SVijay Kumar Srivastava 			  "family is %u, membar is %d,"
3146dad9a73SVijay Kumar Srivastava 			  "function control window offset is %#" PRIx64,
3156dad9a73SVijay Kumar Srivastava 			  sva->family, mem_ebr.ebr_index, mem_ebr.ebr_offset);
3166dad9a73SVijay Kumar Srivastava 
3176dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "init mem bar");
3186dad9a73SVijay Kumar Srivastava 	rc = sfc_vdpa_mem_bar_init(sva, &mem_ebr);
3196dad9a73SVijay Kumar Srivastava 	if (rc != 0)
3206dad9a73SVijay Kumar Srivastava 		goto fail_mem_bar_init;
3216dad9a73SVijay Kumar Srivastava 
3226dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "create nic");
3236dad9a73SVijay Kumar Srivastava 	rte_spinlock_init(&sva->nic_lock);
3246dad9a73SVijay Kumar Srivastava 	rc = efx_nic_create(sva->family, (efsys_identifier_t *)sva,
3256dad9a73SVijay Kumar Srivastava 			    &sva->mem_bar, mem_ebr.ebr_offset,
3266dad9a73SVijay Kumar Srivastava 			    &sva->nic_lock, &enp);
3276dad9a73SVijay Kumar Srivastava 	if (rc != 0) {
3286dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "nic create failed: %s", rte_strerror(rc));
3296dad9a73SVijay Kumar Srivastava 		goto fail_nic_create;
3306dad9a73SVijay Kumar Srivastava 	}
3316dad9a73SVijay Kumar Srivastava 	sva->nic = enp;
3326dad9a73SVijay Kumar Srivastava 
3336dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "init mcdi");
3346dad9a73SVijay Kumar Srivastava 	rc = sfc_vdpa_mcdi_init(sva);
3356dad9a73SVijay Kumar Srivastava 	if (rc != 0) {
3366dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "mcdi init failed: %s", rte_strerror(rc));
3376dad9a73SVijay Kumar Srivastava 		goto fail_mcdi_init;
3386dad9a73SVijay Kumar Srivastava 	}
3396dad9a73SVijay Kumar Srivastava 
3406dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "probe nic");
3416dad9a73SVijay Kumar Srivastava 	rc = sfc_vdpa_nic_probe(sva);
3426dad9a73SVijay Kumar Srivastava 	if (rc != 0)
3436dad9a73SVijay Kumar Srivastava 		goto fail_nic_probe;
3446dad9a73SVijay Kumar Srivastava 
3456dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "reset nic");
3466dad9a73SVijay Kumar Srivastava 	rc = efx_nic_reset(enp);
3476dad9a73SVijay Kumar Srivastava 	if (rc != 0) {
3486dad9a73SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "nic reset failed: %s", rte_strerror(rc));
3496dad9a73SVijay Kumar Srivastava 		goto fail_nic_reset;
3506dad9a73SVijay Kumar Srivastava 	}
3516dad9a73SVijay Kumar Srivastava 
3526dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "estimate resource limits");
3536dad9a73SVijay Kumar Srivastava 	rc = sfc_vdpa_estimate_resource_limits(sva);
3546dad9a73SVijay Kumar Srivastava 	if (rc != 0)
3556dad9a73SVijay Kumar Srivastava 		goto fail_estimate_rsrc_limits;
3566dad9a73SVijay Kumar Srivastava 
357f66a66e6SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "init virtio");
358f66a66e6SVijay Kumar Srivastava 	rc = efx_virtio_init(enp);
359f66a66e6SVijay Kumar Srivastava 	if (rc != 0) {
360f66a66e6SVijay Kumar Srivastava 		sfc_vdpa_err(sva, "virtio init failed: %s", rte_strerror(rc));
361f66a66e6SVijay Kumar Srivastava 		goto fail_virtio_init;
362f66a66e6SVijay Kumar Srivastava 	}
363f66a66e6SVijay Kumar Srivastava 
364cfeed08aSVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "init filter");
365cfeed08aSVijay Kumar Srivastava 	rc = efx_filter_init(enp);
366cfeed08aSVijay Kumar Srivastava 	if (rc != 0) {
367cfeed08aSVijay Kumar Srivastava 		sfc_vdpa_err(sva, "filter init failed: %s", rte_strerror(rc));
368cfeed08aSVijay Kumar Srivastava 		goto fail_filter_init;
369cfeed08aSVijay Kumar Srivastava 	}
370cfeed08aSVijay Kumar Srivastava 
3716dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "done");
3726dad9a73SVijay Kumar Srivastava 
3736dad9a73SVijay Kumar Srivastava 	return 0;
3746dad9a73SVijay Kumar Srivastava 
375cfeed08aSVijay Kumar Srivastava fail_filter_init:
376cfeed08aSVijay Kumar Srivastava 	efx_virtio_fini(enp);
377cfeed08aSVijay Kumar Srivastava 
378f66a66e6SVijay Kumar Srivastava fail_virtio_init:
379f66a66e6SVijay Kumar Srivastava 	efx_nic_fini(enp);
380f66a66e6SVijay Kumar Srivastava 
3816dad9a73SVijay Kumar Srivastava fail_estimate_rsrc_limits:
3826dad9a73SVijay Kumar Srivastava fail_nic_reset:
3836dad9a73SVijay Kumar Srivastava 	efx_nic_unprobe(enp);
3846dad9a73SVijay Kumar Srivastava 
3856dad9a73SVijay Kumar Srivastava fail_nic_probe:
3866dad9a73SVijay Kumar Srivastava 	sfc_vdpa_mcdi_fini(sva);
3876dad9a73SVijay Kumar Srivastava 
3886dad9a73SVijay Kumar Srivastava fail_mcdi_init:
3896dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "destroy nic");
3906dad9a73SVijay Kumar Srivastava 	sva->nic = NULL;
3916dad9a73SVijay Kumar Srivastava 	efx_nic_destroy(enp);
3926dad9a73SVijay Kumar Srivastava 
3936dad9a73SVijay Kumar Srivastava fail_nic_create:
3946dad9a73SVijay Kumar Srivastava 	sfc_vdpa_mem_bar_fini(sva);
3956dad9a73SVijay Kumar Srivastava 
3966dad9a73SVijay Kumar Srivastava fail_mem_bar_init:
3976dad9a73SVijay Kumar Srivastava fail_family:
3986dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "failed: %s", rte_strerror(rc));
3996dad9a73SVijay Kumar Srivastava 	return rc;
4006dad9a73SVijay Kumar Srivastava }
4016dad9a73SVijay Kumar Srivastava 
4026dad9a73SVijay Kumar Srivastava void
sfc_vdpa_hw_fini(struct sfc_vdpa_adapter * sva)4036dad9a73SVijay Kumar Srivastava sfc_vdpa_hw_fini(struct sfc_vdpa_adapter *sva)
4046dad9a73SVijay Kumar Srivastava {
4056dad9a73SVijay Kumar Srivastava 	efx_nic_t *enp = sva->nic;
4066dad9a73SVijay Kumar Srivastava 
4076dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "entry");
4086dad9a73SVijay Kumar Srivastava 
409f66a66e6SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "virtio fini");
410f66a66e6SVijay Kumar Srivastava 	efx_virtio_fini(enp);
411f66a66e6SVijay Kumar Srivastava 
4126dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "unprobe nic");
4136dad9a73SVijay Kumar Srivastava 	efx_nic_unprobe(enp);
4146dad9a73SVijay Kumar Srivastava 
4156dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "mcdi fini");
4166dad9a73SVijay Kumar Srivastava 	sfc_vdpa_mcdi_fini(sva);
4176dad9a73SVijay Kumar Srivastava 
4186dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "nic fini");
4196dad9a73SVijay Kumar Srivastava 	efx_nic_fini(enp);
4206dad9a73SVijay Kumar Srivastava 
4216dad9a73SVijay Kumar Srivastava 	sfc_vdpa_log_init(sva, "destroy nic");
4226dad9a73SVijay Kumar Srivastava 	sva->nic = NULL;
4236dad9a73SVijay Kumar Srivastava 	efx_nic_destroy(enp);
4246dad9a73SVijay Kumar Srivastava 
4256dad9a73SVijay Kumar Srivastava 	sfc_vdpa_mem_bar_fini(sva);
4266dad9a73SVijay Kumar Srivastava }
427