xref: /freebsd/sys/dev/vnic/nicvf_queues.c (revision 6cac5eb74953a736e10b30672ae0cc53f5e448f8)
13c0086b8SZbigniew Bodek /*
23c0086b8SZbigniew Bodek  * Copyright (C) 2015 Cavium Inc.
33c0086b8SZbigniew Bodek  * All rights reserved.
43c0086b8SZbigniew Bodek  *
53c0086b8SZbigniew Bodek  * Redistribution and use in source and binary forms, with or without
63c0086b8SZbigniew Bodek  * modification, are permitted provided that the following conditions
73c0086b8SZbigniew Bodek  * are met:
83c0086b8SZbigniew Bodek  * 1. Redistributions of source code must retain the above copyright
93c0086b8SZbigniew Bodek  *    notice, this list of conditions and the following disclaimer.
103c0086b8SZbigniew Bodek  * 2. Redistributions in binary form must reproduce the above copyright
113c0086b8SZbigniew Bodek  *    notice, this list of conditions and the following disclaimer in the
123c0086b8SZbigniew Bodek  *    documentation and/or other materials provided with the distribution.
133c0086b8SZbigniew Bodek  *
143c0086b8SZbigniew Bodek  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
153c0086b8SZbigniew Bodek  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
163c0086b8SZbigniew Bodek  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
173c0086b8SZbigniew Bodek  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
183c0086b8SZbigniew Bodek  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
193c0086b8SZbigniew Bodek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
203c0086b8SZbigniew Bodek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
213c0086b8SZbigniew Bodek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
223c0086b8SZbigniew Bodek  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
233c0086b8SZbigniew Bodek  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
243c0086b8SZbigniew Bodek  * SUCH DAMAGE.
253c0086b8SZbigniew Bodek  *
263c0086b8SZbigniew Bodek  * $FreeBSD$
273c0086b8SZbigniew Bodek  *
283c0086b8SZbigniew Bodek  */
292306b72aSZbigniew Bodek #include <sys/cdefs.h>
302306b72aSZbigniew Bodek __FBSDID("$FreeBSD$");
313c0086b8SZbigniew Bodek 
32856dce91SZbigniew Bodek #include "opt_inet.h"
33856dce91SZbigniew Bodek #include "opt_inet6.h"
34856dce91SZbigniew Bodek 
352306b72aSZbigniew Bodek #include <sys/param.h>
362306b72aSZbigniew Bodek #include <sys/systm.h>
372306b72aSZbigniew Bodek #include <sys/bitset.h>
382306b72aSZbigniew Bodek #include <sys/bitstring.h>
392306b72aSZbigniew Bodek #include <sys/buf_ring.h>
402306b72aSZbigniew Bodek #include <sys/bus.h>
412306b72aSZbigniew Bodek #include <sys/endian.h>
422306b72aSZbigniew Bodek #include <sys/kernel.h>
432306b72aSZbigniew Bodek #include <sys/malloc.h>
442306b72aSZbigniew Bodek #include <sys/module.h>
452306b72aSZbigniew Bodek #include <sys/rman.h>
462306b72aSZbigniew Bodek #include <sys/pciio.h>
472306b72aSZbigniew Bodek #include <sys/pcpu.h>
482306b72aSZbigniew Bodek #include <sys/proc.h>
492306b72aSZbigniew Bodek #include <sys/sockio.h>
502306b72aSZbigniew Bodek #include <sys/socket.h>
512306b72aSZbigniew Bodek #include <sys/stdatomic.h>
522306b72aSZbigniew Bodek #include <sys/cpuset.h>
532306b72aSZbigniew Bodek #include <sys/lock.h>
542306b72aSZbigniew Bodek #include <sys/mutex.h>
552306b72aSZbigniew Bodek #include <sys/smp.h>
562306b72aSZbigniew Bodek #include <sys/taskqueue.h>
573c0086b8SZbigniew Bodek 
582306b72aSZbigniew Bodek #include <vm/vm.h>
592306b72aSZbigniew Bodek #include <vm/pmap.h>
602306b72aSZbigniew Bodek 
612306b72aSZbigniew Bodek #include <machine/bus.h>
622306b72aSZbigniew Bodek #include <machine/vmparam.h>
632306b72aSZbigniew Bodek 
642306b72aSZbigniew Bodek #include <net/ethernet.h>
652306b72aSZbigniew Bodek #include <net/if.h>
662306b72aSZbigniew Bodek #include <net/if_var.h>
672306b72aSZbigniew Bodek #include <net/if_media.h>
682306b72aSZbigniew Bodek #include <net/ifq.h>
692306b72aSZbigniew Bodek 
70856dce91SZbigniew Bodek #include <netinet/in_systm.h>
71856dce91SZbigniew Bodek #include <netinet/in.h>
72856dce91SZbigniew Bodek #include <netinet/if_ether.h>
73856dce91SZbigniew Bodek #include <netinet/ip.h>
74856dce91SZbigniew Bodek #include <netinet/ip6.h>
75856dce91SZbigniew Bodek #include <netinet/sctp.h>
76856dce91SZbigniew Bodek #include <netinet/tcp.h>
77856dce91SZbigniew Bodek #include <netinet/tcp_lro.h>
78856dce91SZbigniew Bodek #include <netinet/udp.h>
79856dce91SZbigniew Bodek 
802306b72aSZbigniew Bodek #include <dev/pci/pcireg.h>
812306b72aSZbigniew Bodek #include <dev/pci/pcivar.h>
822306b72aSZbigniew Bodek 
832306b72aSZbigniew Bodek #include "thunder_bgx.h"
843c0086b8SZbigniew Bodek #include "nic_reg.h"
853c0086b8SZbigniew Bodek #include "nic.h"
863c0086b8SZbigniew Bodek #include "q_struct.h"
873c0086b8SZbigniew Bodek #include "nicvf_queues.h"
883c0086b8SZbigniew Bodek 
892306b72aSZbigniew Bodek #define	DEBUG
902306b72aSZbigniew Bodek #undef DEBUG
912306b72aSZbigniew Bodek 
922306b72aSZbigniew Bodek #ifdef DEBUG
932306b72aSZbigniew Bodek #define	dprintf(dev, fmt, ...)	device_printf(dev, fmt, ##__VA_ARGS__)
942306b72aSZbigniew Bodek #else
952306b72aSZbigniew Bodek #define	dprintf(dev, fmt, ...)
962306b72aSZbigniew Bodek #endif
972306b72aSZbigniew Bodek 
982306b72aSZbigniew Bodek MALLOC_DECLARE(M_NICVF);
992306b72aSZbigniew Bodek 
1002306b72aSZbigniew Bodek static void nicvf_free_snd_queue(struct nicvf *, struct snd_queue *);
1012306b72aSZbigniew Bodek static struct mbuf * nicvf_get_rcv_mbuf(struct nicvf *, struct cqe_rx_t *);
1022306b72aSZbigniew Bodek static void nicvf_sq_disable(struct nicvf *, int);
1032306b72aSZbigniew Bodek static void nicvf_sq_enable(struct nicvf *, struct snd_queue *, int);
1042306b72aSZbigniew Bodek static void nicvf_put_sq_desc(struct snd_queue *, int);
1052306b72aSZbigniew Bodek static void nicvf_cmp_queue_config(struct nicvf *, struct queue_set *, int,
1062306b72aSZbigniew Bodek     boolean_t);
1072306b72aSZbigniew Bodek static void nicvf_sq_free_used_descs(struct nicvf *, struct snd_queue *, int);
1082306b72aSZbigniew Bodek 
1092306b72aSZbigniew Bodek static void nicvf_rbdr_task(void *, int);
1102306b72aSZbigniew Bodek static void nicvf_rbdr_task_nowait(void *, int);
1112306b72aSZbigniew Bodek 
1123c0086b8SZbigniew Bodek struct rbuf_info {
1132306b72aSZbigniew Bodek 	bus_dma_tag_t	dmat;
1142306b72aSZbigniew Bodek 	bus_dmamap_t	dmap;
1152306b72aSZbigniew Bodek 	struct mbuf *	mbuf;
1163c0086b8SZbigniew Bodek };
1173c0086b8SZbigniew Bodek 
1182306b72aSZbigniew Bodek #define GET_RBUF_INFO(x) ((struct rbuf_info *)((x) - NICVF_RCV_BUF_ALIGN_BYTES))
1193c0086b8SZbigniew Bodek 
1203c0086b8SZbigniew Bodek /* Poll a register for a specific value */
1213c0086b8SZbigniew Bodek static int nicvf_poll_reg(struct nicvf *nic, int qidx,
1222306b72aSZbigniew Bodek 			  uint64_t reg, int bit_pos, int bits, int val)
1233c0086b8SZbigniew Bodek {
1242306b72aSZbigniew Bodek 	uint64_t bit_mask;
1252306b72aSZbigniew Bodek 	uint64_t reg_val;
1263c0086b8SZbigniew Bodek 	int timeout = 10;
1273c0086b8SZbigniew Bodek 
1282306b72aSZbigniew Bodek 	bit_mask = (1UL << bits) - 1;
1293c0086b8SZbigniew Bodek 	bit_mask = (bit_mask << bit_pos);
1303c0086b8SZbigniew Bodek 
1313c0086b8SZbigniew Bodek 	while (timeout) {
1323c0086b8SZbigniew Bodek 		reg_val = nicvf_queue_reg_read(nic, reg, qidx);
1333c0086b8SZbigniew Bodek 		if (((reg_val & bit_mask) >> bit_pos) == val)
1342306b72aSZbigniew Bodek 			return (0);
1352306b72aSZbigniew Bodek 
1362306b72aSZbigniew Bodek 		DELAY(1000);
1373c0086b8SZbigniew Bodek 		timeout--;
1383c0086b8SZbigniew Bodek 	}
1392306b72aSZbigniew Bodek 	device_printf(nic->dev, "Poll on reg 0x%lx failed\n", reg);
1402306b72aSZbigniew Bodek 	return (ETIMEDOUT);
1412306b72aSZbigniew Bodek }
1422306b72aSZbigniew Bodek 
1432306b72aSZbigniew Bodek /* Callback for bus_dmamap_load() */
1442306b72aSZbigniew Bodek static void
1452306b72aSZbigniew Bodek nicvf_dmamap_q_cb(void *arg, bus_dma_segment_t *segs, int nseg, int error)
1462306b72aSZbigniew Bodek {
1472306b72aSZbigniew Bodek 	bus_addr_t *paddr;
1482306b72aSZbigniew Bodek 
1492306b72aSZbigniew Bodek 	KASSERT(nseg == 1, ("wrong number of segments, should be 1"));
1502306b72aSZbigniew Bodek 	paddr = arg;
1512306b72aSZbigniew Bodek 	*paddr = segs->ds_addr;
1523c0086b8SZbigniew Bodek }
1533c0086b8SZbigniew Bodek 
1543c0086b8SZbigniew Bodek /* Allocate memory for a queue's descriptors */
1552306b72aSZbigniew Bodek static int
1562306b72aSZbigniew Bodek nicvf_alloc_q_desc_mem(struct nicvf *nic, struct q_desc_mem *dmem,
1573c0086b8SZbigniew Bodek     int q_len, int desc_size, int align_bytes)
1583c0086b8SZbigniew Bodek {
1592306b72aSZbigniew Bodek 	int err, err_dmat;
1603c0086b8SZbigniew Bodek 
1612306b72aSZbigniew Bodek 	/* Create DMA tag first */
1622306b72aSZbigniew Bodek 	err = bus_dma_tag_create(
1632306b72aSZbigniew Bodek 	    bus_get_dma_tag(nic->dev),		/* parent tag */
1642306b72aSZbigniew Bodek 	    align_bytes,			/* alignment */
1652306b72aSZbigniew Bodek 	    0,					/* boundary */
1662306b72aSZbigniew Bodek 	    BUS_SPACE_MAXADDR,			/* lowaddr */
1672306b72aSZbigniew Bodek 	    BUS_SPACE_MAXADDR,			/* highaddr */
1682306b72aSZbigniew Bodek 	    NULL, NULL,				/* filtfunc, filtfuncarg */
1692306b72aSZbigniew Bodek 	    (q_len * desc_size),		/* maxsize */
1702306b72aSZbigniew Bodek 	    1,					/* nsegments */
1712306b72aSZbigniew Bodek 	    (q_len * desc_size),		/* maxsegsize */
1722306b72aSZbigniew Bodek 	    0,					/* flags */
1732306b72aSZbigniew Bodek 	    NULL, NULL,				/* lockfunc, lockfuncarg */
1742306b72aSZbigniew Bodek 	    &dmem->dmat);			/* dmat */
1752306b72aSZbigniew Bodek 
1762306b72aSZbigniew Bodek 	if (err != 0) {
1772306b72aSZbigniew Bodek 		device_printf(nic->dev,
1782306b72aSZbigniew Bodek 		    "Failed to create busdma tag for descriptors ring\n");
1792306b72aSZbigniew Bodek 		return (err);
1802306b72aSZbigniew Bodek 	}
1812306b72aSZbigniew Bodek 
1822306b72aSZbigniew Bodek 	/* Allocate segment of continuous DMA safe memory */
1832306b72aSZbigniew Bodek 	err = bus_dmamem_alloc(
1842306b72aSZbigniew Bodek 	    dmem->dmat,				/* DMA tag */
1852306b72aSZbigniew Bodek 	    &dmem->base,			/* virtual address */
1862306b72aSZbigniew Bodek 	    (BUS_DMA_NOWAIT | BUS_DMA_ZERO),	/* flags */
1872306b72aSZbigniew Bodek 	    &dmem->dmap);			/* DMA map */
1882306b72aSZbigniew Bodek 	if (err != 0) {
1892306b72aSZbigniew Bodek 		device_printf(nic->dev, "Failed to allocate DMA safe memory for"
1902306b72aSZbigniew Bodek 		    "descriptors ring\n");
1912306b72aSZbigniew Bodek 		goto dmamem_fail;
1922306b72aSZbigniew Bodek 	}
1932306b72aSZbigniew Bodek 
1942306b72aSZbigniew Bodek 	err = bus_dmamap_load(
1952306b72aSZbigniew Bodek 	    dmem->dmat,
1962306b72aSZbigniew Bodek 	    dmem->dmap,
1972306b72aSZbigniew Bodek 	    dmem->base,
1982306b72aSZbigniew Bodek 	    (q_len * desc_size),		/* allocation size */
1992306b72aSZbigniew Bodek 	    nicvf_dmamap_q_cb,			/* map to DMA address cb. */
2002306b72aSZbigniew Bodek 	    &dmem->phys_base,			/* physical address */
2012306b72aSZbigniew Bodek 	    BUS_DMA_NOWAIT);
2022306b72aSZbigniew Bodek 	if (err != 0) {
2032306b72aSZbigniew Bodek 		device_printf(nic->dev,
2042306b72aSZbigniew Bodek 		    "Cannot load DMA map of descriptors ring\n");
2052306b72aSZbigniew Bodek 		goto dmamap_fail;
2062306b72aSZbigniew Bodek 	}
2072306b72aSZbigniew Bodek 
2082306b72aSZbigniew Bodek 	dmem->q_len = q_len;
2092306b72aSZbigniew Bodek 	dmem->size = (desc_size * q_len);
2102306b72aSZbigniew Bodek 
2112306b72aSZbigniew Bodek 	return (0);
2122306b72aSZbigniew Bodek 
2132306b72aSZbigniew Bodek dmamap_fail:
2142306b72aSZbigniew Bodek 	bus_dmamem_free(dmem->dmat, dmem->base, dmem->dmap);
2152306b72aSZbigniew Bodek 	dmem->phys_base = 0;
2162306b72aSZbigniew Bodek dmamem_fail:
2172306b72aSZbigniew Bodek 	err_dmat = bus_dma_tag_destroy(dmem->dmat);
2182306b72aSZbigniew Bodek 	dmem->base = NULL;
2192306b72aSZbigniew Bodek 	KASSERT(err_dmat == 0,
2202306b72aSZbigniew Bodek 	    ("%s: Trying to destroy BUSY DMA tag", __func__));
2212306b72aSZbigniew Bodek 
2222306b72aSZbigniew Bodek 	return (err);
2233c0086b8SZbigniew Bodek }
2243c0086b8SZbigniew Bodek 
2253c0086b8SZbigniew Bodek /* Free queue's descriptor memory */
2262306b72aSZbigniew Bodek static void
2272306b72aSZbigniew Bodek nicvf_free_q_desc_mem(struct nicvf *nic, struct q_desc_mem *dmem)
2283c0086b8SZbigniew Bodek {
2292306b72aSZbigniew Bodek 	int err;
2302306b72aSZbigniew Bodek 
2312306b72aSZbigniew Bodek 	if ((dmem == NULL) || (dmem->base == NULL))
2323c0086b8SZbigniew Bodek 		return;
2333c0086b8SZbigniew Bodek 
2342306b72aSZbigniew Bodek 	/* Unload a map */
2352306b72aSZbigniew Bodek 	bus_dmamap_sync(dmem->dmat, dmem->dmap, BUS_DMASYNC_POSTREAD);
2362306b72aSZbigniew Bodek 	bus_dmamap_unload(dmem->dmat, dmem->dmap);
2372306b72aSZbigniew Bodek 	/* Free DMA memory */
2382306b72aSZbigniew Bodek 	bus_dmamem_free(dmem->dmat, dmem->base, dmem->dmap);
2392306b72aSZbigniew Bodek 	/* Destroy DMA tag */
2402306b72aSZbigniew Bodek 	err = bus_dma_tag_destroy(dmem->dmat);
2412306b72aSZbigniew Bodek 
2422306b72aSZbigniew Bodek 	KASSERT(err == 0,
2432306b72aSZbigniew Bodek 	    ("%s: Trying to destroy BUSY DMA tag", __func__));
2442306b72aSZbigniew Bodek 
2452306b72aSZbigniew Bodek 	dmem->phys_base = 0;
2463c0086b8SZbigniew Bodek 	dmem->base = NULL;
2473c0086b8SZbigniew Bodek }
2483c0086b8SZbigniew Bodek 
2492306b72aSZbigniew Bodek /*
2502306b72aSZbigniew Bodek  * Allocate buffer for packet reception
2513c0086b8SZbigniew Bodek  * HW returns memory address where packet is DMA'ed but not a pointer
2523c0086b8SZbigniew Bodek  * into RBDR ring, so save buffer address at the start of fragment and
2533c0086b8SZbigniew Bodek  * align the start address to a cache aligned address
2543c0086b8SZbigniew Bodek  */
2552306b72aSZbigniew Bodek static __inline int
2562306b72aSZbigniew Bodek nicvf_alloc_rcv_buffer(struct nicvf *nic, struct rbdr *rbdr,
2572306b72aSZbigniew Bodek     bus_dmamap_t dmap, int mflags, uint32_t buf_len, bus_addr_t *rbuf)
2583c0086b8SZbigniew Bodek {
2592306b72aSZbigniew Bodek 	struct mbuf *mbuf;
2603c0086b8SZbigniew Bodek 	struct rbuf_info *rinfo;
2612306b72aSZbigniew Bodek 	bus_dma_segment_t segs[1];
2622306b72aSZbigniew Bodek 	int nsegs;
2632306b72aSZbigniew Bodek 	int err;
2643c0086b8SZbigniew Bodek 
2652306b72aSZbigniew Bodek 	mbuf = m_getjcl(mflags, MT_DATA, M_PKTHDR, MCLBYTES);
2662306b72aSZbigniew Bodek 	if (mbuf == NULL)
2672306b72aSZbigniew Bodek 		return (ENOMEM);
2682306b72aSZbigniew Bodek 
2692306b72aSZbigniew Bodek 	/*
2702306b72aSZbigniew Bodek 	 * The length is equal to the actual length + one 128b line
2712306b72aSZbigniew Bodek 	 * used as a room for rbuf_info structure.
2722306b72aSZbigniew Bodek 	 */
2732306b72aSZbigniew Bodek 	mbuf->m_len = mbuf->m_pkthdr.len = buf_len;
2742306b72aSZbigniew Bodek 
2752306b72aSZbigniew Bodek 	err = bus_dmamap_load_mbuf_sg(rbdr->rbdr_buff_dmat, dmap, mbuf, segs,
2762306b72aSZbigniew Bodek 	    &nsegs, BUS_DMA_NOWAIT);
2772306b72aSZbigniew Bodek 	if (err != 0) {
2782306b72aSZbigniew Bodek 		device_printf(nic->dev,
2792306b72aSZbigniew Bodek 		    "Failed to map mbuf into DMA visible memory, err: %d\n",
2802306b72aSZbigniew Bodek 		    err);
2812306b72aSZbigniew Bodek 		m_freem(mbuf);
2822306b72aSZbigniew Bodek 		bus_dmamap_destroy(rbdr->rbdr_buff_dmat, dmap);
2832306b72aSZbigniew Bodek 		return (err);
2843c0086b8SZbigniew Bodek 	}
2852306b72aSZbigniew Bodek 	if (nsegs != 1)
2862306b72aSZbigniew Bodek 		panic("Unexpected number of DMA segments for RB: %d", nsegs);
2872306b72aSZbigniew Bodek 	/*
2882306b72aSZbigniew Bodek 	 * Now use the room for rbuf_info structure
2892306b72aSZbigniew Bodek 	 * and adjust mbuf data and length.
2902306b72aSZbigniew Bodek 	 */
2912306b72aSZbigniew Bodek 	rinfo = (struct rbuf_info *)mbuf->m_data;
2922306b72aSZbigniew Bodek 	m_adj(mbuf, NICVF_RCV_BUF_ALIGN_BYTES);
2932306b72aSZbigniew Bodek 
2942306b72aSZbigniew Bodek 	rinfo->dmat = rbdr->rbdr_buff_dmat;
2952306b72aSZbigniew Bodek 	rinfo->dmap = dmap;
2962306b72aSZbigniew Bodek 	rinfo->mbuf = mbuf;
2972306b72aSZbigniew Bodek 
2982306b72aSZbigniew Bodek 	*rbuf = segs[0].ds_addr + NICVF_RCV_BUF_ALIGN_BYTES;
2992306b72aSZbigniew Bodek 
3002306b72aSZbigniew Bodek 	return (0);
3013c0086b8SZbigniew Bodek }
3023c0086b8SZbigniew Bodek 
3032306b72aSZbigniew Bodek /* Retrieve mbuf for received packet */
3042306b72aSZbigniew Bodek static struct mbuf *
3052306b72aSZbigniew Bodek nicvf_rb_ptr_to_mbuf(struct nicvf *nic, bus_addr_t rb_ptr)
3063c0086b8SZbigniew Bodek {
3072306b72aSZbigniew Bodek 	struct mbuf *mbuf;
3083c0086b8SZbigniew Bodek 	struct rbuf_info *rinfo;
3093c0086b8SZbigniew Bodek 
3103c0086b8SZbigniew Bodek 	/* Get buffer start address and alignment offset */
3112306b72aSZbigniew Bodek 	rinfo = GET_RBUF_INFO(PHYS_TO_DMAP(rb_ptr));
3123c0086b8SZbigniew Bodek 
3132306b72aSZbigniew Bodek 	/* Now retrieve mbuf to give to stack */
3142306b72aSZbigniew Bodek 	mbuf = rinfo->mbuf;
3152306b72aSZbigniew Bodek 	if (__predict_false(mbuf == NULL)) {
3162306b72aSZbigniew Bodek 		panic("%s: Received packet fragment with NULL mbuf",
3172306b72aSZbigniew Bodek 		    device_get_nameunit(nic->dev));
3183c0086b8SZbigniew Bodek 	}
3192306b72aSZbigniew Bodek 	/*
3202306b72aSZbigniew Bodek 	 * Clear the mbuf in the descriptor to indicate
3212306b72aSZbigniew Bodek 	 * that this slot is processed and free to use.
3222306b72aSZbigniew Bodek 	 */
3232306b72aSZbigniew Bodek 	rinfo->mbuf = NULL;
3243c0086b8SZbigniew Bodek 
3252306b72aSZbigniew Bodek 	bus_dmamap_sync(rinfo->dmat, rinfo->dmap, BUS_DMASYNC_POSTREAD);
3262306b72aSZbigniew Bodek 	bus_dmamap_unload(rinfo->dmat, rinfo->dmap);
3273c0086b8SZbigniew Bodek 
3282306b72aSZbigniew Bodek 	return (mbuf);
3293c0086b8SZbigniew Bodek }
3303c0086b8SZbigniew Bodek 
3313c0086b8SZbigniew Bodek /* Allocate RBDR ring and populate receive buffers */
3322306b72aSZbigniew Bodek static int
3332306b72aSZbigniew Bodek nicvf_init_rbdr(struct nicvf *nic, struct rbdr *rbdr, int ring_len,
3342306b72aSZbigniew Bodek     int buf_size, int qidx)
3353c0086b8SZbigniew Bodek {
3362306b72aSZbigniew Bodek 	bus_dmamap_t dmap;
3372306b72aSZbigniew Bodek 	bus_addr_t rbuf;
3383c0086b8SZbigniew Bodek 	struct rbdr_entry_t *desc;
3392306b72aSZbigniew Bodek 	int idx;
3403c0086b8SZbigniew Bodek 	int err;
3413c0086b8SZbigniew Bodek 
3422306b72aSZbigniew Bodek 	/* Allocate rbdr descriptors ring */
3433c0086b8SZbigniew Bodek 	err = nicvf_alloc_q_desc_mem(nic, &rbdr->dmem, ring_len,
3442306b72aSZbigniew Bodek 	    sizeof(struct rbdr_entry_t), NICVF_RCV_BUF_ALIGN_BYTES);
3452306b72aSZbigniew Bodek 	if (err != 0) {
3462306b72aSZbigniew Bodek 		device_printf(nic->dev,
3472306b72aSZbigniew Bodek 		    "Failed to create RBDR descriptors ring\n");
3482306b72aSZbigniew Bodek 		return (err);
3492306b72aSZbigniew Bodek 	}
3503c0086b8SZbigniew Bodek 
3513c0086b8SZbigniew Bodek 	rbdr->desc = rbdr->dmem.base;
3522306b72aSZbigniew Bodek 	/*
3532306b72aSZbigniew Bodek 	 * Buffer size has to be in multiples of 128 bytes.
3542306b72aSZbigniew Bodek 	 * Make room for metadata of size of one line (128 bytes).
3552306b72aSZbigniew Bodek 	 */
3562306b72aSZbigniew Bodek 	rbdr->dma_size = buf_size - NICVF_RCV_BUF_ALIGN_BYTES;
3572306b72aSZbigniew Bodek 	rbdr->enable = TRUE;
3583c0086b8SZbigniew Bodek 	rbdr->thresh = RBDR_THRESH;
3592306b72aSZbigniew Bodek 	rbdr->nic = nic;
3602306b72aSZbigniew Bodek 	rbdr->idx = qidx;
3613c0086b8SZbigniew Bodek 
3622306b72aSZbigniew Bodek 	/*
3632306b72aSZbigniew Bodek 	 * Create DMA tag for Rx buffers.
3642306b72aSZbigniew Bodek 	 * Each map created using this tag is intended to store Rx payload for
3652306b72aSZbigniew Bodek 	 * one fragment and one header structure containing rbuf_info (thus
3662306b72aSZbigniew Bodek 	 * additional 128 byte line since RB must be a multiple of 128 byte
3672306b72aSZbigniew Bodek 	 * cache line).
3682306b72aSZbigniew Bodek 	 */
3692306b72aSZbigniew Bodek 	if (buf_size > MCLBYTES) {
3702306b72aSZbigniew Bodek 		device_printf(nic->dev,
3712306b72aSZbigniew Bodek 		    "Buffer size to large for mbuf cluster\n");
3722306b72aSZbigniew Bodek 		return (EINVAL);
3732306b72aSZbigniew Bodek 	}
3742306b72aSZbigniew Bodek 	err = bus_dma_tag_create(
3752306b72aSZbigniew Bodek 	    bus_get_dma_tag(nic->dev),		/* parent tag */
3762306b72aSZbigniew Bodek 	    NICVF_RCV_BUF_ALIGN_BYTES,		/* alignment */
3772306b72aSZbigniew Bodek 	    0,					/* boundary */
3782306b72aSZbigniew Bodek 	    DMAP_MAX_PHYSADDR,			/* lowaddr */
3792306b72aSZbigniew Bodek 	    DMAP_MIN_PHYSADDR,			/* highaddr */
3802306b72aSZbigniew Bodek 	    NULL, NULL,				/* filtfunc, filtfuncarg */
3812306b72aSZbigniew Bodek 	    roundup2(buf_size, MCLBYTES),	/* maxsize */
3822306b72aSZbigniew Bodek 	    1,					/* nsegments */
3832306b72aSZbigniew Bodek 	    roundup2(buf_size, MCLBYTES),	/* maxsegsize */
3842306b72aSZbigniew Bodek 	    0,					/* flags */
3852306b72aSZbigniew Bodek 	    NULL, NULL,				/* lockfunc, lockfuncarg */
3862306b72aSZbigniew Bodek 	    &rbdr->rbdr_buff_dmat);		/* dmat */
3872306b72aSZbigniew Bodek 
3882306b72aSZbigniew Bodek 	if (err != 0) {
3892306b72aSZbigniew Bodek 		device_printf(nic->dev,
3902306b72aSZbigniew Bodek 		    "Failed to create busdma tag for RBDR buffers\n");
3912306b72aSZbigniew Bodek 		return (err);
3922306b72aSZbigniew Bodek 	}
3932306b72aSZbigniew Bodek 
3942306b72aSZbigniew Bodek 	rbdr->rbdr_buff_dmaps = malloc(sizeof(*rbdr->rbdr_buff_dmaps) *
3952306b72aSZbigniew Bodek 	    ring_len, M_NICVF, (M_WAITOK | M_ZERO));
3962306b72aSZbigniew Bodek 
3973c0086b8SZbigniew Bodek 	for (idx = 0; idx < ring_len; idx++) {
3982306b72aSZbigniew Bodek 		err = bus_dmamap_create(rbdr->rbdr_buff_dmat, 0, &dmap);
3992306b72aSZbigniew Bodek 		if (err != 0) {
4002306b72aSZbigniew Bodek 			device_printf(nic->dev,
4012306b72aSZbigniew Bodek 			    "Failed to create DMA map for RB\n");
4022306b72aSZbigniew Bodek 			return (err);
4032306b72aSZbigniew Bodek 		}
4042306b72aSZbigniew Bodek 		rbdr->rbdr_buff_dmaps[idx] = dmap;
4052306b72aSZbigniew Bodek 
4062306b72aSZbigniew Bodek 		err = nicvf_alloc_rcv_buffer(nic, rbdr, dmap, M_WAITOK,
4072306b72aSZbigniew Bodek 		    DMA_BUFFER_LEN, &rbuf);
4082306b72aSZbigniew Bodek 		if (err != 0)
4092306b72aSZbigniew Bodek 			return (err);
4103c0086b8SZbigniew Bodek 
4113c0086b8SZbigniew Bodek 		desc = GET_RBDR_DESC(rbdr, idx);
4122306b72aSZbigniew Bodek 		desc->buf_addr = (rbuf >> NICVF_RCV_BUF_ALIGN);
4133c0086b8SZbigniew Bodek 	}
4142306b72aSZbigniew Bodek 
4152306b72aSZbigniew Bodek 	/* Allocate taskqueue */
4162306b72aSZbigniew Bodek 	TASK_INIT(&rbdr->rbdr_task, 0, nicvf_rbdr_task, rbdr);
4172306b72aSZbigniew Bodek 	TASK_INIT(&rbdr->rbdr_task_nowait, 0, nicvf_rbdr_task_nowait, rbdr);
4182306b72aSZbigniew Bodek 	rbdr->rbdr_taskq = taskqueue_create_fast("nicvf_rbdr_taskq", M_WAITOK,
4192306b72aSZbigniew Bodek 	    taskqueue_thread_enqueue, &rbdr->rbdr_taskq);
4202306b72aSZbigniew Bodek 	taskqueue_start_threads(&rbdr->rbdr_taskq, 1, PI_NET, "%s: rbdr_taskq",
4212306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev));
4222306b72aSZbigniew Bodek 
4232306b72aSZbigniew Bodek 	return (0);
4243c0086b8SZbigniew Bodek }
4253c0086b8SZbigniew Bodek 
4263c0086b8SZbigniew Bodek /* Free RBDR ring and its receive buffers */
4272306b72aSZbigniew Bodek static void
4282306b72aSZbigniew Bodek nicvf_free_rbdr(struct nicvf *nic, struct rbdr *rbdr)
4293c0086b8SZbigniew Bodek {
4302306b72aSZbigniew Bodek 	struct mbuf *mbuf;
4312306b72aSZbigniew Bodek 	struct queue_set *qs;
4323c0086b8SZbigniew Bodek 	struct rbdr_entry_t *desc;
4333c0086b8SZbigniew Bodek 	struct rbuf_info *rinfo;
4342306b72aSZbigniew Bodek 	bus_addr_t buf_addr;
4352306b72aSZbigniew Bodek 	int head, tail, idx;
4362306b72aSZbigniew Bodek 	int err;
4373c0086b8SZbigniew Bodek 
4382306b72aSZbigniew Bodek 	qs = nic->qs;
4392306b72aSZbigniew Bodek 
4402306b72aSZbigniew Bodek 	if ((qs == NULL) || (rbdr == NULL))
4413c0086b8SZbigniew Bodek 		return;
4423c0086b8SZbigniew Bodek 
4432306b72aSZbigniew Bodek 	rbdr->enable = FALSE;
4442306b72aSZbigniew Bodek 	if (rbdr->rbdr_taskq != NULL) {
4452306b72aSZbigniew Bodek 		/* Remove tasks */
4462306b72aSZbigniew Bodek 		while (taskqueue_cancel(rbdr->rbdr_taskq,
4472306b72aSZbigniew Bodek 		    &rbdr->rbdr_task_nowait, NULL) != 0) {
4482306b72aSZbigniew Bodek 			/* Finish the nowait task first */
4492306b72aSZbigniew Bodek 			taskqueue_drain(rbdr->rbdr_taskq,
4502306b72aSZbigniew Bodek 			    &rbdr->rbdr_task_nowait);
4512306b72aSZbigniew Bodek 		}
4522306b72aSZbigniew Bodek 		taskqueue_free(rbdr->rbdr_taskq);
4532306b72aSZbigniew Bodek 		rbdr->rbdr_taskq = NULL;
4543c0086b8SZbigniew Bodek 
4552306b72aSZbigniew Bodek 		while (taskqueue_cancel(taskqueue_thread,
4562306b72aSZbigniew Bodek 		    &rbdr->rbdr_task, NULL) != 0) {
4572306b72aSZbigniew Bodek 			/* Now finish the sleepable task */
4582306b72aSZbigniew Bodek 			taskqueue_drain(taskqueue_thread, &rbdr->rbdr_task);
4592306b72aSZbigniew Bodek 		}
4602306b72aSZbigniew Bodek 	}
4612306b72aSZbigniew Bodek 
4622306b72aSZbigniew Bodek 	/*
4632306b72aSZbigniew Bodek 	 * Free all of the memory under the RB descriptors.
4642306b72aSZbigniew Bodek 	 * There are assumptions here:
4652306b72aSZbigniew Bodek 	 * 1. Corresponding RBDR is disabled
4662306b72aSZbigniew Bodek 	 *    - it is safe to operate using head and tail indexes
4672306b72aSZbigniew Bodek 	 * 2. All bffers that were received are properly freed by
4682306b72aSZbigniew Bodek 	 *    the receive handler
4692306b72aSZbigniew Bodek 	 *    - there is no need to unload DMA map and free MBUF for other
4702306b72aSZbigniew Bodek 	 *      descriptors than unused ones
4712306b72aSZbigniew Bodek 	 */
4722306b72aSZbigniew Bodek 	if (rbdr->rbdr_buff_dmat != NULL) {
4733c0086b8SZbigniew Bodek 		head = rbdr->head;
4743c0086b8SZbigniew Bodek 		tail = rbdr->tail;
4753c0086b8SZbigniew Bodek 		while (head != tail) {
4763c0086b8SZbigniew Bodek 			desc = GET_RBDR_DESC(rbdr, head);
4773c0086b8SZbigniew Bodek 			buf_addr = desc->buf_addr << NICVF_RCV_BUF_ALIGN;
4782306b72aSZbigniew Bodek 			rinfo = GET_RBUF_INFO(PHYS_TO_DMAP(buf_addr));
4792306b72aSZbigniew Bodek 			bus_dmamap_unload(rbdr->rbdr_buff_dmat, rinfo->dmap);
4802306b72aSZbigniew Bodek 			mbuf = rinfo->mbuf;
4812306b72aSZbigniew Bodek 			/* This will destroy everything including rinfo! */
4822306b72aSZbigniew Bodek 			m_freem(mbuf);
4833c0086b8SZbigniew Bodek 			head++;
4843c0086b8SZbigniew Bodek 			head &= (rbdr->dmem.q_len - 1);
4853c0086b8SZbigniew Bodek 		}
4862306b72aSZbigniew Bodek 		/* Free tail descriptor */
4873c0086b8SZbigniew Bodek 		desc = GET_RBDR_DESC(rbdr, tail);
4883c0086b8SZbigniew Bodek 		buf_addr = desc->buf_addr << NICVF_RCV_BUF_ALIGN;
4892306b72aSZbigniew Bodek 		rinfo = GET_RBUF_INFO(PHYS_TO_DMAP(buf_addr));
4902306b72aSZbigniew Bodek 		bus_dmamap_unload(rbdr->rbdr_buff_dmat, rinfo->dmap);
4912306b72aSZbigniew Bodek 		mbuf = rinfo->mbuf;
4922306b72aSZbigniew Bodek 		/* This will destroy everything including rinfo! */
4932306b72aSZbigniew Bodek 		m_freem(mbuf);
4942306b72aSZbigniew Bodek 
4952306b72aSZbigniew Bodek 		/* Destroy DMA maps */
4962306b72aSZbigniew Bodek 		for (idx = 0; idx < qs->rbdr_len; idx++) {
4972306b72aSZbigniew Bodek 			if (rbdr->rbdr_buff_dmaps[idx] == NULL)
4982306b72aSZbigniew Bodek 				continue;
4992306b72aSZbigniew Bodek 			err = bus_dmamap_destroy(rbdr->rbdr_buff_dmat,
5002306b72aSZbigniew Bodek 			    rbdr->rbdr_buff_dmaps[idx]);
5012306b72aSZbigniew Bodek 			KASSERT(err == 0,
5022306b72aSZbigniew Bodek 			    ("%s: Could not destroy DMA map for RB, desc: %d",
5032306b72aSZbigniew Bodek 			    __func__, idx));
5042306b72aSZbigniew Bodek 			rbdr->rbdr_buff_dmaps[idx] = NULL;
5052306b72aSZbigniew Bodek 		}
5062306b72aSZbigniew Bodek 
5072306b72aSZbigniew Bodek 		/* Now destroy the tag */
5082306b72aSZbigniew Bodek 		err = bus_dma_tag_destroy(rbdr->rbdr_buff_dmat);
5092306b72aSZbigniew Bodek 		KASSERT(err == 0,
5102306b72aSZbigniew Bodek 		    ("%s: Trying to destroy BUSY DMA tag", __func__));
5112306b72aSZbigniew Bodek 
5122306b72aSZbigniew Bodek 		rbdr->head = 0;
5132306b72aSZbigniew Bodek 		rbdr->tail = 0;
5142306b72aSZbigniew Bodek 	}
5153c0086b8SZbigniew Bodek 
5163c0086b8SZbigniew Bodek 	/* Free RBDR ring */
5173c0086b8SZbigniew Bodek 	nicvf_free_q_desc_mem(nic, &rbdr->dmem);
5183c0086b8SZbigniew Bodek }
5193c0086b8SZbigniew Bodek 
5202306b72aSZbigniew Bodek /*
5212306b72aSZbigniew Bodek  * Refill receive buffer descriptors with new buffers.
5223c0086b8SZbigniew Bodek  */
5232306b72aSZbigniew Bodek static int
5242306b72aSZbigniew Bodek nicvf_refill_rbdr(struct rbdr *rbdr, int mflags)
5253c0086b8SZbigniew Bodek {
5262306b72aSZbigniew Bodek 	struct nicvf *nic;
5272306b72aSZbigniew Bodek 	struct queue_set *qs;
5282306b72aSZbigniew Bodek 	int rbdr_idx;
5293c0086b8SZbigniew Bodek 	int tail, qcount;
5303c0086b8SZbigniew Bodek 	int refill_rb_cnt;
5313c0086b8SZbigniew Bodek 	struct rbdr_entry_t *desc;
5322306b72aSZbigniew Bodek 	bus_dmamap_t dmap;
5332306b72aSZbigniew Bodek 	bus_addr_t rbuf;
5342306b72aSZbigniew Bodek 	boolean_t rb_alloc_fail;
5352306b72aSZbigniew Bodek 	int new_rb;
5363c0086b8SZbigniew Bodek 
5372306b72aSZbigniew Bodek 	rb_alloc_fail = TRUE;
5382306b72aSZbigniew Bodek 	new_rb = 0;
5392306b72aSZbigniew Bodek 	nic = rbdr->nic;
5402306b72aSZbigniew Bodek 	qs = nic->qs;
5412306b72aSZbigniew Bodek 	rbdr_idx = rbdr->idx;
5422306b72aSZbigniew Bodek 
5433c0086b8SZbigniew Bodek 	/* Check if it's enabled */
5443c0086b8SZbigniew Bodek 	if (!rbdr->enable)
5452306b72aSZbigniew Bodek 		return (0);
5463c0086b8SZbigniew Bodek 
5473c0086b8SZbigniew Bodek 	/* Get no of desc's to be refilled */
5483c0086b8SZbigniew Bodek 	qcount = nicvf_queue_reg_read(nic, NIC_QSET_RBDR_0_1_STATUS0, rbdr_idx);
5493c0086b8SZbigniew Bodek 	qcount &= 0x7FFFF;
5503c0086b8SZbigniew Bodek 	/* Doorbell can be ringed with a max of ring size minus 1 */
5512306b72aSZbigniew Bodek 	if (qcount >= (qs->rbdr_len - 1)) {
5522306b72aSZbigniew Bodek 		rb_alloc_fail = FALSE;
5532306b72aSZbigniew Bodek 		goto out;
5542306b72aSZbigniew Bodek 	} else
5553c0086b8SZbigniew Bodek 		refill_rb_cnt = qs->rbdr_len - qcount - 1;
5563c0086b8SZbigniew Bodek 
5573c0086b8SZbigniew Bodek 	/* Start filling descs from tail */
5583c0086b8SZbigniew Bodek 	tail = nicvf_queue_reg_read(nic, NIC_QSET_RBDR_0_1_TAIL, rbdr_idx) >> 3;
5593c0086b8SZbigniew Bodek 	while (refill_rb_cnt) {
5603c0086b8SZbigniew Bodek 		tail++;
5613c0086b8SZbigniew Bodek 		tail &= (rbdr->dmem.q_len - 1);
5623c0086b8SZbigniew Bodek 
5632306b72aSZbigniew Bodek 		dmap = rbdr->rbdr_buff_dmaps[tail];
5642306b72aSZbigniew Bodek 		if (nicvf_alloc_rcv_buffer(nic, rbdr, dmap, mflags,
5652306b72aSZbigniew Bodek 		    DMA_BUFFER_LEN, &rbuf)) {
5662306b72aSZbigniew Bodek 			/* Something went wrong. Resign */
5673c0086b8SZbigniew Bodek 			break;
5682306b72aSZbigniew Bodek 		}
5693c0086b8SZbigniew Bodek 		desc = GET_RBDR_DESC(rbdr, tail);
5702306b72aSZbigniew Bodek 		desc->buf_addr = (rbuf >> NICVF_RCV_BUF_ALIGN);
5713c0086b8SZbigniew Bodek 		refill_rb_cnt--;
5723c0086b8SZbigniew Bodek 		new_rb++;
5733c0086b8SZbigniew Bodek 	}
5743c0086b8SZbigniew Bodek 
5753c0086b8SZbigniew Bodek 	/* make sure all memory stores are done before ringing doorbell */
5762306b72aSZbigniew Bodek 	wmb();
5773c0086b8SZbigniew Bodek 
5783c0086b8SZbigniew Bodek 	/* Check if buffer allocation failed */
5792306b72aSZbigniew Bodek 	if (refill_rb_cnt == 0)
5802306b72aSZbigniew Bodek 		rb_alloc_fail = FALSE;
5813c0086b8SZbigniew Bodek 
5823c0086b8SZbigniew Bodek 	/* Notify HW */
5833c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_DOOR,
5843c0086b8SZbigniew Bodek 			      rbdr_idx, new_rb);
5852306b72aSZbigniew Bodek out:
5862306b72aSZbigniew Bodek 	if (!rb_alloc_fail) {
5872306b72aSZbigniew Bodek 		/*
5882306b72aSZbigniew Bodek 		 * Re-enable RBDR interrupts only
5892306b72aSZbigniew Bodek 		 * if buffer allocation is success.
5902306b72aSZbigniew Bodek 		 */
5913c0086b8SZbigniew Bodek 		nicvf_enable_intr(nic, NICVF_INTR_RBDR, rbdr_idx);
5923c0086b8SZbigniew Bodek 
5932306b72aSZbigniew Bodek 		return (0);
5943c0086b8SZbigniew Bodek 	}
5953c0086b8SZbigniew Bodek 
5962306b72aSZbigniew Bodek 	return (ENOMEM);
5972306b72aSZbigniew Bodek }
5982306b72aSZbigniew Bodek 
5992306b72aSZbigniew Bodek /* Refill RBs even if sleep is needed to reclaim memory */
6002306b72aSZbigniew Bodek static void
6012306b72aSZbigniew Bodek nicvf_rbdr_task(void *arg, int pending)
6023c0086b8SZbigniew Bodek {
6032306b72aSZbigniew Bodek 	struct rbdr *rbdr;
6042306b72aSZbigniew Bodek 	int err;
6053c0086b8SZbigniew Bodek 
6062306b72aSZbigniew Bodek 	rbdr = (struct rbdr *)arg;
6072306b72aSZbigniew Bodek 
6082306b72aSZbigniew Bodek 	err = nicvf_refill_rbdr(rbdr, M_WAITOK);
6092306b72aSZbigniew Bodek 	if (__predict_false(err != 0)) {
6102306b72aSZbigniew Bodek 		panic("%s: Failed to refill RBs even when sleep enabled",
6112306b72aSZbigniew Bodek 		    __func__);
6122306b72aSZbigniew Bodek 	}
6133c0086b8SZbigniew Bodek }
6143c0086b8SZbigniew Bodek 
6152306b72aSZbigniew Bodek /* Refill RBs as soon as possible without waiting */
6162306b72aSZbigniew Bodek static void
6172306b72aSZbigniew Bodek nicvf_rbdr_task_nowait(void *arg, int pending)
6183c0086b8SZbigniew Bodek {
6192306b72aSZbigniew Bodek 	struct rbdr *rbdr;
6202306b72aSZbigniew Bodek 	int err;
6213c0086b8SZbigniew Bodek 
6222306b72aSZbigniew Bodek 	rbdr = (struct rbdr *)arg;
6232306b72aSZbigniew Bodek 
6242306b72aSZbigniew Bodek 	err = nicvf_refill_rbdr(rbdr, M_NOWAIT);
6252306b72aSZbigniew Bodek 	if (err != 0) {
6262306b72aSZbigniew Bodek 		/*
6272306b72aSZbigniew Bodek 		 * Schedule another, sleepable kernel thread
6282306b72aSZbigniew Bodek 		 * that will for sure refill the buffers.
6292306b72aSZbigniew Bodek 		 */
6302306b72aSZbigniew Bodek 		taskqueue_enqueue(taskqueue_thread, &rbdr->rbdr_task);
6313c0086b8SZbigniew Bodek 	}
6323c0086b8SZbigniew Bodek }
6333c0086b8SZbigniew Bodek 
6342306b72aSZbigniew Bodek static int
6352306b72aSZbigniew Bodek nicvf_rcv_pkt_handler(struct nicvf *nic, struct cmp_queue *cq,
6362306b72aSZbigniew Bodek     struct cqe_rx_t *cqe_rx, int cqe_type)
6372306b72aSZbigniew Bodek {
6382306b72aSZbigniew Bodek 	struct mbuf *mbuf;
639053f3d0eSZbigniew Bodek 	struct rcv_queue *rq;
6402306b72aSZbigniew Bodek 	int rq_idx;
6412306b72aSZbigniew Bodek 	int err = 0;
6422306b72aSZbigniew Bodek 
6432306b72aSZbigniew Bodek 	rq_idx = cqe_rx->rq_idx;
644053f3d0eSZbigniew Bodek 	rq = &nic->qs->rq[rq_idx];
6452306b72aSZbigniew Bodek 
6462306b72aSZbigniew Bodek 	/* Check for errors */
6472306b72aSZbigniew Bodek 	err = nicvf_check_cqe_rx_errs(nic, cq, cqe_rx);
6482306b72aSZbigniew Bodek 	if (err && !cqe_rx->rb_cnt)
6492306b72aSZbigniew Bodek 		return (0);
6502306b72aSZbigniew Bodek 
6512306b72aSZbigniew Bodek 	mbuf = nicvf_get_rcv_mbuf(nic, cqe_rx);
6522306b72aSZbigniew Bodek 	if (mbuf == NULL) {
6532306b72aSZbigniew Bodek 		dprintf(nic->dev, "Packet not received\n");
6542306b72aSZbigniew Bodek 		return (0);
6552306b72aSZbigniew Bodek 	}
6562306b72aSZbigniew Bodek 
6572306b72aSZbigniew Bodek 	/* If error packet */
6582306b72aSZbigniew Bodek 	if (err != 0) {
6592306b72aSZbigniew Bodek 		m_freem(mbuf);
6602306b72aSZbigniew Bodek 		return (0);
6612306b72aSZbigniew Bodek 	}
6622306b72aSZbigniew Bodek 
663053f3d0eSZbigniew Bodek 	if (rq->lro_enabled &&
664053f3d0eSZbigniew Bodek 	    ((cqe_rx->l3_type == L3TYPE_IPV4) && (cqe_rx->l4_type == L4TYPE_TCP)) &&
665053f3d0eSZbigniew Bodek 	    (mbuf->m_pkthdr.csum_flags & (CSUM_DATA_VALID | CSUM_PSEUDO_HDR)) ==
666053f3d0eSZbigniew Bodek             (CSUM_DATA_VALID | CSUM_PSEUDO_HDR)) {
667053f3d0eSZbigniew Bodek 		/*
668053f3d0eSZbigniew Bodek 		 * At this point it is known that there are no errors in the
669053f3d0eSZbigniew Bodek 		 * packet. Attempt to LRO enqueue. Send to stack if no resources
670053f3d0eSZbigniew Bodek 		 * or enqueue error.
671053f3d0eSZbigniew Bodek 		 */
672053f3d0eSZbigniew Bodek 		if ((rq->lro.lro_cnt != 0) &&
673053f3d0eSZbigniew Bodek 		    (tcp_lro_rx(&rq->lro, mbuf, 0) == 0))
674053f3d0eSZbigniew Bodek 			return (0);
675053f3d0eSZbigniew Bodek 	}
6762306b72aSZbigniew Bodek 	/*
6772306b72aSZbigniew Bodek 	 * Push this packet to the stack later to avoid
6782306b72aSZbigniew Bodek 	 * unlocking completion task in the middle of work.
6792306b72aSZbigniew Bodek 	 */
6802306b72aSZbigniew Bodek 	err = buf_ring_enqueue(cq->rx_br, mbuf);
6812306b72aSZbigniew Bodek 	if (err != 0) {
6822306b72aSZbigniew Bodek 		/*
6832306b72aSZbigniew Bodek 		 * Failed to enqueue this mbuf.
6842306b72aSZbigniew Bodek 		 * We don't drop it, just schedule another task.
6852306b72aSZbigniew Bodek 		 */
6862306b72aSZbigniew Bodek 		return (err);
6872306b72aSZbigniew Bodek 	}
6882306b72aSZbigniew Bodek 
6892306b72aSZbigniew Bodek 	return (0);
6902306b72aSZbigniew Bodek }
6912306b72aSZbigniew Bodek 
6922306b72aSZbigniew Bodek static int
6932306b72aSZbigniew Bodek nicvf_snd_pkt_handler(struct nicvf *nic, struct cmp_queue *cq,
6942306b72aSZbigniew Bodek     struct cqe_send_t *cqe_tx, int cqe_type)
6952306b72aSZbigniew Bodek {
6962306b72aSZbigniew Bodek 	bus_dmamap_t dmap;
6972306b72aSZbigniew Bodek 	struct mbuf *mbuf;
6982306b72aSZbigniew Bodek 	struct snd_queue *sq;
6992306b72aSZbigniew Bodek 	struct sq_hdr_subdesc *hdr;
7002306b72aSZbigniew Bodek 
7012306b72aSZbigniew Bodek 	mbuf = NULL;
7022306b72aSZbigniew Bodek 	sq = &nic->qs->sq[cqe_tx->sq_idx];
7032306b72aSZbigniew Bodek 	/* Avoid blocking here since we hold a non-sleepable NICVF_CMP_LOCK */
7042306b72aSZbigniew Bodek 	if (NICVF_TX_TRYLOCK(sq) == 0)
7052306b72aSZbigniew Bodek 		return (EAGAIN);
7062306b72aSZbigniew Bodek 
7072306b72aSZbigniew Bodek 	hdr = (struct sq_hdr_subdesc *)GET_SQ_DESC(sq, cqe_tx->sqe_ptr);
7082306b72aSZbigniew Bodek 	if (hdr->subdesc_type != SQ_DESC_TYPE_HEADER) {
7092306b72aSZbigniew Bodek 		NICVF_TX_UNLOCK(sq);
7102306b72aSZbigniew Bodek 		return (0);
7112306b72aSZbigniew Bodek 	}
7122306b72aSZbigniew Bodek 
7132306b72aSZbigniew Bodek 	dprintf(nic->dev,
7142306b72aSZbigniew Bodek 	    "%s Qset #%d SQ #%d SQ ptr #%d subdesc count %d\n",
7152306b72aSZbigniew Bodek 	    __func__, cqe_tx->sq_qs, cqe_tx->sq_idx,
7162306b72aSZbigniew Bodek 	    cqe_tx->sqe_ptr, hdr->subdesc_cnt);
7172306b72aSZbigniew Bodek 
7182306b72aSZbigniew Bodek 	dmap = (bus_dmamap_t)sq->snd_buff[cqe_tx->sqe_ptr].dmap;
7192306b72aSZbigniew Bodek 	bus_dmamap_unload(sq->snd_buff_dmat, dmap);
7202306b72aSZbigniew Bodek 
7212306b72aSZbigniew Bodek 	mbuf = (struct mbuf *)sq->snd_buff[cqe_tx->sqe_ptr].mbuf;
7222306b72aSZbigniew Bodek 	if (mbuf != NULL) {
7232306b72aSZbigniew Bodek 		m_freem(mbuf);
7242306b72aSZbigniew Bodek 		sq->snd_buff[cqe_tx->sqe_ptr].mbuf = NULL;
725*6cac5eb7SZbigniew Bodek 		nicvf_put_sq_desc(sq, hdr->subdesc_cnt + 1);
7262306b72aSZbigniew Bodek 	}
7272306b72aSZbigniew Bodek 
7282306b72aSZbigniew Bodek 	nicvf_check_cqe_tx_errs(nic, cq, cqe_tx);
7292306b72aSZbigniew Bodek 
7302306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
7312306b72aSZbigniew Bodek 	return (0);
7322306b72aSZbigniew Bodek }
7332306b72aSZbigniew Bodek 
7342306b72aSZbigniew Bodek static int
7352306b72aSZbigniew Bodek nicvf_cq_intr_handler(struct nicvf *nic, uint8_t cq_idx)
7362306b72aSZbigniew Bodek {
7372306b72aSZbigniew Bodek 	struct mbuf *mbuf;
7382306b72aSZbigniew Bodek 	struct ifnet *ifp;
7392306b72aSZbigniew Bodek 	int processed_cqe, work_done = 0, tx_done = 0;
7402306b72aSZbigniew Bodek 	int cqe_count, cqe_head;
7412306b72aSZbigniew Bodek 	struct queue_set *qs = nic->qs;
7422306b72aSZbigniew Bodek 	struct cmp_queue *cq = &qs->cq[cq_idx];
743053f3d0eSZbigniew Bodek 	struct rcv_queue *rq;
7442306b72aSZbigniew Bodek 	struct cqe_rx_t *cq_desc;
745053f3d0eSZbigniew Bodek 	struct lro_ctrl	*lro;
746053f3d0eSZbigniew Bodek 	struct lro_entry *queued;
747053f3d0eSZbigniew Bodek 	int rq_idx;
7482306b72aSZbigniew Bodek 	int cmp_err;
7492306b72aSZbigniew Bodek 
7502306b72aSZbigniew Bodek 	NICVF_CMP_LOCK(cq);
7512306b72aSZbigniew Bodek 	cmp_err = 0;
7522306b72aSZbigniew Bodek 	processed_cqe = 0;
7532306b72aSZbigniew Bodek 	/* Get no of valid CQ entries to process */
7542306b72aSZbigniew Bodek 	cqe_count = nicvf_queue_reg_read(nic, NIC_QSET_CQ_0_7_STATUS, cq_idx);
7552306b72aSZbigniew Bodek 	cqe_count &= CQ_CQE_COUNT;
7562306b72aSZbigniew Bodek 	if (cqe_count == 0)
7572306b72aSZbigniew Bodek 		goto out;
7582306b72aSZbigniew Bodek 
7592306b72aSZbigniew Bodek 	/* Get head of the valid CQ entries */
7602306b72aSZbigniew Bodek 	cqe_head = nicvf_queue_reg_read(nic, NIC_QSET_CQ_0_7_HEAD, cq_idx) >> 9;
7612306b72aSZbigniew Bodek 	cqe_head &= 0xFFFF;
7622306b72aSZbigniew Bodek 
7632306b72aSZbigniew Bodek 	dprintf(nic->dev, "%s CQ%d cqe_count %d cqe_head %d\n",
7642306b72aSZbigniew Bodek 	    __func__, cq_idx, cqe_count, cqe_head);
7652306b72aSZbigniew Bodek 	while (processed_cqe < cqe_count) {
7662306b72aSZbigniew Bodek 		/* Get the CQ descriptor */
7672306b72aSZbigniew Bodek 		cq_desc = (struct cqe_rx_t *)GET_CQ_DESC(cq, cqe_head);
7682306b72aSZbigniew Bodek 		cqe_head++;
7692306b72aSZbigniew Bodek 		cqe_head &= (cq->dmem.q_len - 1);
77044dd2693SZbigniew Bodek 		/* Prefetch next CQ descriptor */
77144dd2693SZbigniew Bodek 		__builtin_prefetch((struct cqe_rx_t *)GET_CQ_DESC(cq, cqe_head));
7722306b72aSZbigniew Bodek 
7732306b72aSZbigniew Bodek 		dprintf(nic->dev, "CQ%d cq_desc->cqe_type %d\n", cq_idx,
7742306b72aSZbigniew Bodek 		    cq_desc->cqe_type);
7752306b72aSZbigniew Bodek 		switch (cq_desc->cqe_type) {
7762306b72aSZbigniew Bodek 		case CQE_TYPE_RX:
7772306b72aSZbigniew Bodek 			cmp_err = nicvf_rcv_pkt_handler(nic, cq, cq_desc,
7782306b72aSZbigniew Bodek 			    CQE_TYPE_RX);
7792306b72aSZbigniew Bodek 			if (__predict_false(cmp_err != 0)) {
7802306b72aSZbigniew Bodek 				/*
7812306b72aSZbigniew Bodek 				 * Ups. Cannot finish now.
7822306b72aSZbigniew Bodek 				 * Let's try again later.
7832306b72aSZbigniew Bodek 				 */
7842306b72aSZbigniew Bodek 				goto done;
7852306b72aSZbigniew Bodek 			}
7862306b72aSZbigniew Bodek 			work_done++;
7872306b72aSZbigniew Bodek 			break;
7882306b72aSZbigniew Bodek 		case CQE_TYPE_SEND:
7892306b72aSZbigniew Bodek 			cmp_err = nicvf_snd_pkt_handler(nic, cq,
7902306b72aSZbigniew Bodek 			    (void *)cq_desc, CQE_TYPE_SEND);
7912306b72aSZbigniew Bodek 			if (__predict_false(cmp_err != 0)) {
7922306b72aSZbigniew Bodek 				/*
7932306b72aSZbigniew Bodek 				 * Ups. Cannot finish now.
7942306b72aSZbigniew Bodek 				 * Let's try again later.
7952306b72aSZbigniew Bodek 				 */
7962306b72aSZbigniew Bodek 				goto done;
7972306b72aSZbigniew Bodek 			}
7982306b72aSZbigniew Bodek 
7992306b72aSZbigniew Bodek 			tx_done++;
8002306b72aSZbigniew Bodek 			break;
8012306b72aSZbigniew Bodek 		case CQE_TYPE_INVALID:
8022306b72aSZbigniew Bodek 		case CQE_TYPE_RX_SPLIT:
8032306b72aSZbigniew Bodek 		case CQE_TYPE_RX_TCP:
8042306b72aSZbigniew Bodek 		case CQE_TYPE_SEND_PTP:
8052306b72aSZbigniew Bodek 			/* Ignore for now */
8062306b72aSZbigniew Bodek 			break;
8072306b72aSZbigniew Bodek 		}
8082306b72aSZbigniew Bodek 		processed_cqe++;
8092306b72aSZbigniew Bodek 	}
8102306b72aSZbigniew Bodek done:
8112306b72aSZbigniew Bodek 	dprintf(nic->dev,
8122306b72aSZbigniew Bodek 	    "%s CQ%d processed_cqe %d work_done %d\n",
8132306b72aSZbigniew Bodek 	    __func__, cq_idx, processed_cqe, work_done);
8142306b72aSZbigniew Bodek 
8152306b72aSZbigniew Bodek 	/* Ring doorbell to inform H/W to reuse processed CQEs */
8162306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_DOOR, cq_idx, processed_cqe);
8172306b72aSZbigniew Bodek 
8182306b72aSZbigniew Bodek 	if ((tx_done > 0) &&
8192306b72aSZbigniew Bodek 	    ((if_getdrvflags(nic->ifp) & IFF_DRV_RUNNING) != 0)) {
8202306b72aSZbigniew Bodek 		/* Reenable TXQ if its stopped earlier due to SQ full */
8212306b72aSZbigniew Bodek 		if_setdrvflagbits(nic->ifp, IFF_DRV_RUNNING, IFF_DRV_OACTIVE);
8222306b72aSZbigniew Bodek 	}
8232306b72aSZbigniew Bodek out:
824053f3d0eSZbigniew Bodek 	/*
825053f3d0eSZbigniew Bodek 	 * Flush any outstanding LRO work
826053f3d0eSZbigniew Bodek 	 */
827053f3d0eSZbigniew Bodek 	rq_idx = cq_idx;
828053f3d0eSZbigniew Bodek 	rq = &nic->qs->rq[rq_idx];
829053f3d0eSZbigniew Bodek 	lro = &rq->lro;
830053f3d0eSZbigniew Bodek 	while ((queued = SLIST_FIRST(&lro->lro_active)) != NULL) {
831053f3d0eSZbigniew Bodek 		SLIST_REMOVE_HEAD(&lro->lro_active, next);
832053f3d0eSZbigniew Bodek 		tcp_lro_flush(lro, queued);
833053f3d0eSZbigniew Bodek 	}
834053f3d0eSZbigniew Bodek 
8352306b72aSZbigniew Bodek 	NICVF_CMP_UNLOCK(cq);
8362306b72aSZbigniew Bodek 
8372306b72aSZbigniew Bodek 	ifp = nic->ifp;
8382306b72aSZbigniew Bodek 	/* Push received MBUFs to the stack */
8392306b72aSZbigniew Bodek 	while (!buf_ring_empty(cq->rx_br)) {
8402306b72aSZbigniew Bodek 		mbuf = buf_ring_dequeue_mc(cq->rx_br);
8412306b72aSZbigniew Bodek 		if (__predict_true(mbuf != NULL))
8422306b72aSZbigniew Bodek 			(*ifp->if_input)(ifp, mbuf);
8432306b72aSZbigniew Bodek 	}
8442306b72aSZbigniew Bodek 
8452306b72aSZbigniew Bodek 	return (cmp_err);
8462306b72aSZbigniew Bodek }
8472306b72aSZbigniew Bodek 
8482306b72aSZbigniew Bodek /*
8492306b72aSZbigniew Bodek  * Qset error interrupt handler
8502306b72aSZbigniew Bodek  *
8512306b72aSZbigniew Bodek  * As of now only CQ errors are handled
8522306b72aSZbigniew Bodek  */
8532306b72aSZbigniew Bodek static void
8542306b72aSZbigniew Bodek nicvf_qs_err_task(void *arg, int pending)
8552306b72aSZbigniew Bodek {
8562306b72aSZbigniew Bodek 	struct nicvf *nic;
8572306b72aSZbigniew Bodek 	struct queue_set *qs;
8582306b72aSZbigniew Bodek 	int qidx;
8592306b72aSZbigniew Bodek 	uint64_t status;
8602306b72aSZbigniew Bodek 	boolean_t enable = TRUE;
8612306b72aSZbigniew Bodek 
8622306b72aSZbigniew Bodek 	nic = (struct nicvf *)arg;
8632306b72aSZbigniew Bodek 	qs = nic->qs;
8642306b72aSZbigniew Bodek 
8652306b72aSZbigniew Bodek 	/* Deactivate network interface */
8662306b72aSZbigniew Bodek 	if_setdrvflagbits(nic->ifp, IFF_DRV_OACTIVE, IFF_DRV_RUNNING);
8672306b72aSZbigniew Bodek 
8682306b72aSZbigniew Bodek 	/* Check if it is CQ err */
8692306b72aSZbigniew Bodek 	for (qidx = 0; qidx < qs->cq_cnt; qidx++) {
8702306b72aSZbigniew Bodek 		status = nicvf_queue_reg_read(nic, NIC_QSET_CQ_0_7_STATUS,
8712306b72aSZbigniew Bodek 		    qidx);
8722306b72aSZbigniew Bodek 		if ((status & CQ_ERR_MASK) == 0)
8732306b72aSZbigniew Bodek 			continue;
8742306b72aSZbigniew Bodek 		/* Process already queued CQEs and reconfig CQ */
8752306b72aSZbigniew Bodek 		nicvf_disable_intr(nic, NICVF_INTR_CQ, qidx);
8762306b72aSZbigniew Bodek 		nicvf_sq_disable(nic, qidx);
8772306b72aSZbigniew Bodek 		(void)nicvf_cq_intr_handler(nic, qidx);
8782306b72aSZbigniew Bodek 		nicvf_cmp_queue_config(nic, qs, qidx, enable);
8792306b72aSZbigniew Bodek 		nicvf_sq_free_used_descs(nic, &qs->sq[qidx], qidx);
8802306b72aSZbigniew Bodek 		nicvf_sq_enable(nic, &qs->sq[qidx], qidx);
8812306b72aSZbigniew Bodek 		nicvf_enable_intr(nic, NICVF_INTR_CQ, qidx);
8822306b72aSZbigniew Bodek 	}
8832306b72aSZbigniew Bodek 
8842306b72aSZbigniew Bodek 	if_setdrvflagbits(nic->ifp, IFF_DRV_RUNNING, IFF_DRV_OACTIVE);
8852306b72aSZbigniew Bodek 	/* Re-enable Qset error interrupt */
8862306b72aSZbigniew Bodek 	nicvf_enable_intr(nic, NICVF_INTR_QS_ERR, 0);
8872306b72aSZbigniew Bodek }
8882306b72aSZbigniew Bodek 
8892306b72aSZbigniew Bodek static void
8902306b72aSZbigniew Bodek nicvf_cmp_task(void *arg, int pending)
8912306b72aSZbigniew Bodek {
8922306b72aSZbigniew Bodek 	struct cmp_queue *cq;
8932306b72aSZbigniew Bodek 	struct nicvf *nic;
8942306b72aSZbigniew Bodek 	int cmp_err;
8952306b72aSZbigniew Bodek 
8962306b72aSZbigniew Bodek 	cq = (struct cmp_queue *)arg;
8972306b72aSZbigniew Bodek 	nic = cq->nic;
8982306b72aSZbigniew Bodek 
8992306b72aSZbigniew Bodek 	/* Handle CQ descriptors */
9002306b72aSZbigniew Bodek 	cmp_err = nicvf_cq_intr_handler(nic, cq->idx);
9012306b72aSZbigniew Bodek 	if (__predict_false(cmp_err != 0)) {
9022306b72aSZbigniew Bodek 		/*
9032306b72aSZbigniew Bodek 		 * Schedule another thread here since we did not
9042306b72aSZbigniew Bodek 		 * process the entire CQ due to Tx or Rx CQ parse error.
9052306b72aSZbigniew Bodek 		 */
9062306b72aSZbigniew Bodek 		taskqueue_enqueue(cq->cmp_taskq, &cq->cmp_task);
9072306b72aSZbigniew Bodek 
9082306b72aSZbigniew Bodek 	}
9092306b72aSZbigniew Bodek 
9100196c2e8SZbigniew Bodek 	nicvf_clear_intr(nic, NICVF_INTR_CQ, cq->idx);
9112306b72aSZbigniew Bodek 	/* Reenable interrupt (previously disabled in nicvf_intr_handler() */
9122306b72aSZbigniew Bodek 	nicvf_enable_intr(nic, NICVF_INTR_CQ, cq->idx);
9132306b72aSZbigniew Bodek 
9142306b72aSZbigniew Bodek }
9152306b72aSZbigniew Bodek 
9163c0086b8SZbigniew Bodek /* Initialize completion queue */
9172306b72aSZbigniew Bodek static int
9182306b72aSZbigniew Bodek nicvf_init_cmp_queue(struct nicvf *nic, struct cmp_queue *cq, int q_len,
9192306b72aSZbigniew Bodek     int qidx)
9203c0086b8SZbigniew Bodek {
9213c0086b8SZbigniew Bodek 	int err;
9223c0086b8SZbigniew Bodek 
9232306b72aSZbigniew Bodek 	/* Initizalize lock */
9242306b72aSZbigniew Bodek 	snprintf(cq->mtx_name, sizeof(cq->mtx_name), "%s: CQ(%d) lock",
9252306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev), qidx);
9262306b72aSZbigniew Bodek 	mtx_init(&cq->mtx, cq->mtx_name, NULL, MTX_DEF);
9272306b72aSZbigniew Bodek 
9283c0086b8SZbigniew Bodek 	err = nicvf_alloc_q_desc_mem(nic, &cq->dmem, q_len, CMP_QUEUE_DESC_SIZE,
9293c0086b8SZbigniew Bodek 				     NICVF_CQ_BASE_ALIGN_BYTES);
9302306b72aSZbigniew Bodek 
9312306b72aSZbigniew Bodek 	if (err != 0) {
9322306b72aSZbigniew Bodek 		device_printf(nic->dev,
9332306b72aSZbigniew Bodek 		    "Could not allocate DMA memory for CQ\n");
9342306b72aSZbigniew Bodek 		return (err);
9352306b72aSZbigniew Bodek 	}
9363c0086b8SZbigniew Bodek 
9373c0086b8SZbigniew Bodek 	cq->desc = cq->dmem.base;
9383bdcfeadSZbigniew Bodek 	cq->thresh = pass1_silicon(nic->dev) ? 0 : CMP_QUEUE_CQE_THRESH;
9392306b72aSZbigniew Bodek 	cq->nic = nic;
9402306b72aSZbigniew Bodek 	cq->idx = qidx;
9413c0086b8SZbigniew Bodek 	nic->cq_coalesce_usecs = (CMP_QUEUE_TIMER_THRESH * 0.05) - 1;
9423c0086b8SZbigniew Bodek 
9432306b72aSZbigniew Bodek 	cq->rx_br = buf_ring_alloc(CMP_QUEUE_LEN * 8, M_DEVBUF, M_WAITOK,
9442306b72aSZbigniew Bodek 	    &cq->mtx);
9452306b72aSZbigniew Bodek 
9462306b72aSZbigniew Bodek 	/* Allocate taskqueue */
9472306b72aSZbigniew Bodek 	TASK_INIT(&cq->cmp_task, 0, nicvf_cmp_task, cq);
9482306b72aSZbigniew Bodek 	cq->cmp_taskq = taskqueue_create_fast("nicvf_cmp_taskq", M_WAITOK,
9492306b72aSZbigniew Bodek 	    taskqueue_thread_enqueue, &cq->cmp_taskq);
9502306b72aSZbigniew Bodek 	taskqueue_start_threads(&cq->cmp_taskq, 1, PI_NET, "%s: cmp_taskq(%d)",
9512306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev), qidx);
9522306b72aSZbigniew Bodek 
9532306b72aSZbigniew Bodek 	return (0);
9543c0086b8SZbigniew Bodek }
9553c0086b8SZbigniew Bodek 
9562306b72aSZbigniew Bodek static void
9572306b72aSZbigniew Bodek nicvf_free_cmp_queue(struct nicvf *nic, struct cmp_queue *cq)
9583c0086b8SZbigniew Bodek {
9593c0086b8SZbigniew Bodek 
9602306b72aSZbigniew Bodek 	if (cq == NULL)
9612306b72aSZbigniew Bodek 		return;
9622306b72aSZbigniew Bodek 	/*
9632306b72aSZbigniew Bodek 	 * The completion queue itself should be disabled by now
9642306b72aSZbigniew Bodek 	 * (ref. nicvf_snd_queue_config()).
9652306b72aSZbigniew Bodek 	 * Ensure that it is safe to disable it or panic.
9662306b72aSZbigniew Bodek 	 */
9672306b72aSZbigniew Bodek 	if (cq->enable)
9682306b72aSZbigniew Bodek 		panic("%s: Trying to free working CQ(%d)", __func__, cq->idx);
9692306b72aSZbigniew Bodek 
9702306b72aSZbigniew Bodek 	if (cq->cmp_taskq != NULL) {
9712306b72aSZbigniew Bodek 		/* Remove task */
9722306b72aSZbigniew Bodek 		while (taskqueue_cancel(cq->cmp_taskq, &cq->cmp_task, NULL) != 0)
9732306b72aSZbigniew Bodek 			taskqueue_drain(cq->cmp_taskq, &cq->cmp_task);
9742306b72aSZbigniew Bodek 
9752306b72aSZbigniew Bodek 		taskqueue_free(cq->cmp_taskq);
9762306b72aSZbigniew Bodek 		cq->cmp_taskq = NULL;
9772306b72aSZbigniew Bodek 	}
9782306b72aSZbigniew Bodek 	/*
9792306b72aSZbigniew Bodek 	 * Completion interrupt will possibly enable interrupts again
9802306b72aSZbigniew Bodek 	 * so disable interrupting now after we finished processing
9812306b72aSZbigniew Bodek 	 * completion task. It is safe to do so since the corresponding CQ
9822306b72aSZbigniew Bodek 	 * was already disabled.
9832306b72aSZbigniew Bodek 	 */
9842306b72aSZbigniew Bodek 	nicvf_disable_intr(nic, NICVF_INTR_CQ, cq->idx);
9852306b72aSZbigniew Bodek 	nicvf_clear_intr(nic, NICVF_INTR_CQ, cq->idx);
9862306b72aSZbigniew Bodek 
9872306b72aSZbigniew Bodek 	NICVF_CMP_LOCK(cq);
9883c0086b8SZbigniew Bodek 	nicvf_free_q_desc_mem(nic, &cq->dmem);
9892306b72aSZbigniew Bodek 	drbr_free(cq->rx_br, M_DEVBUF);
9902306b72aSZbigniew Bodek 	NICVF_CMP_UNLOCK(cq);
9912306b72aSZbigniew Bodek 	mtx_destroy(&cq->mtx);
9922306b72aSZbigniew Bodek 	memset(cq->mtx_name, 0, sizeof(cq->mtx_name));
9932306b72aSZbigniew Bodek }
9942306b72aSZbigniew Bodek 
9952306b72aSZbigniew Bodek static void
9962306b72aSZbigniew Bodek nicvf_snd_task(void *arg, int pending)
9972306b72aSZbigniew Bodek {
9982306b72aSZbigniew Bodek 	struct snd_queue *sq = (struct snd_queue *)arg;
9992306b72aSZbigniew Bodek 	struct mbuf *mbuf;
10002306b72aSZbigniew Bodek 
10012306b72aSZbigniew Bodek 	NICVF_TX_LOCK(sq);
10022306b72aSZbigniew Bodek 	while (1) {
10032306b72aSZbigniew Bodek 		mbuf = drbr_dequeue(NULL, sq->br);
10042306b72aSZbigniew Bodek 		if (mbuf == NULL)
10052306b72aSZbigniew Bodek 			break;
10062306b72aSZbigniew Bodek 
10072306b72aSZbigniew Bodek 		if (nicvf_tx_mbuf_locked(sq, mbuf) != 0) {
10082306b72aSZbigniew Bodek 			/* XXX ARM64TODO: Increase Tx drop counter */
10092306b72aSZbigniew Bodek 			m_freem(mbuf);
10102306b72aSZbigniew Bodek 			break;
10112306b72aSZbigniew Bodek 		}
10122306b72aSZbigniew Bodek 	}
10132306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
10143c0086b8SZbigniew Bodek }
10153c0086b8SZbigniew Bodek 
10163c0086b8SZbigniew Bodek /* Initialize transmit queue */
10172306b72aSZbigniew Bodek static int
10182306b72aSZbigniew Bodek nicvf_init_snd_queue(struct nicvf *nic, struct snd_queue *sq, int q_len,
10192306b72aSZbigniew Bodek     int qidx)
10203c0086b8SZbigniew Bodek {
10212306b72aSZbigniew Bodek 	size_t i;
10223c0086b8SZbigniew Bodek 	int err;
10233c0086b8SZbigniew Bodek 
10242306b72aSZbigniew Bodek 	/* Initizalize TX lock for this queue */
10252306b72aSZbigniew Bodek 	snprintf(sq->mtx_name, sizeof(sq->mtx_name), "%s: SQ(%d) lock",
10262306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev), qidx);
10272306b72aSZbigniew Bodek 	mtx_init(&sq->mtx, sq->mtx_name, NULL, MTX_DEF);
10282306b72aSZbigniew Bodek 
10292306b72aSZbigniew Bodek 	NICVF_TX_LOCK(sq);
10302306b72aSZbigniew Bodek 	/* Allocate buffer ring */
10312306b72aSZbigniew Bodek 	sq->br = buf_ring_alloc(q_len / MIN_SQ_DESC_PER_PKT_XMIT, M_DEVBUF,
10322306b72aSZbigniew Bodek 	    M_NOWAIT, &sq->mtx);
10332306b72aSZbigniew Bodek 	if (sq->br == NULL) {
10342306b72aSZbigniew Bodek 		device_printf(nic->dev,
10352306b72aSZbigniew Bodek 		    "ERROR: Could not set up buf ring for SQ(%d)\n", qidx);
10362306b72aSZbigniew Bodek 		err = ENOMEM;
10372306b72aSZbigniew Bodek 		goto error;
10382306b72aSZbigniew Bodek 	}
10392306b72aSZbigniew Bodek 
10402306b72aSZbigniew Bodek 	/* Allocate DMA memory for Tx descriptors */
10413c0086b8SZbigniew Bodek 	err = nicvf_alloc_q_desc_mem(nic, &sq->dmem, q_len, SND_QUEUE_DESC_SIZE,
10423c0086b8SZbigniew Bodek 				     NICVF_SQ_BASE_ALIGN_BYTES);
10432306b72aSZbigniew Bodek 	if (err != 0) {
10442306b72aSZbigniew Bodek 		device_printf(nic->dev,
10452306b72aSZbigniew Bodek 		    "Could not allocate DMA memory for SQ\n");
10462306b72aSZbigniew Bodek 		goto error;
10472306b72aSZbigniew Bodek 	}
10483c0086b8SZbigniew Bodek 
10493c0086b8SZbigniew Bodek 	sq->desc = sq->dmem.base;
10502306b72aSZbigniew Bodek 	sq->head = sq->tail = 0;
10512306b72aSZbigniew Bodek 	atomic_store_rel_int(&sq->free_cnt, q_len - 1);
10523c0086b8SZbigniew Bodek 	sq->thresh = SND_QUEUE_THRESH;
10532306b72aSZbigniew Bodek 	sq->idx = qidx;
10542306b72aSZbigniew Bodek 	sq->nic = nic;
10553c0086b8SZbigniew Bodek 
10562306b72aSZbigniew Bodek 	/*
10572306b72aSZbigniew Bodek 	 * Allocate DMA maps for Tx buffers
10582306b72aSZbigniew Bodek 	 */
10593c0086b8SZbigniew Bodek 
10602306b72aSZbigniew Bodek 	/* Create DMA tag first */
10612306b72aSZbigniew Bodek 	err = bus_dma_tag_create(
10622306b72aSZbigniew Bodek 	    bus_get_dma_tag(nic->dev),		/* parent tag */
10632306b72aSZbigniew Bodek 	    1,					/* alignment */
10642306b72aSZbigniew Bodek 	    0,					/* boundary */
10652306b72aSZbigniew Bodek 	    BUS_SPACE_MAXADDR,			/* lowaddr */
10662306b72aSZbigniew Bodek 	    BUS_SPACE_MAXADDR,			/* highaddr */
10672306b72aSZbigniew Bodek 	    NULL, NULL,				/* filtfunc, filtfuncarg */
1068af8fe8f1SZbigniew Bodek 	    NICVF_TSO_MAXSIZE,			/* maxsize */
1069af8fe8f1SZbigniew Bodek 	    NICVF_TSO_NSEGS,			/* nsegments */
10702306b72aSZbigniew Bodek 	    MCLBYTES,				/* maxsegsize */
10712306b72aSZbigniew Bodek 	    0,					/* flags */
10722306b72aSZbigniew Bodek 	    NULL, NULL,				/* lockfunc, lockfuncarg */
10732306b72aSZbigniew Bodek 	    &sq->snd_buff_dmat);		/* dmat */
10742306b72aSZbigniew Bodek 
10752306b72aSZbigniew Bodek 	if (err != 0) {
10762306b72aSZbigniew Bodek 		device_printf(nic->dev,
10772306b72aSZbigniew Bodek 		    "Failed to create busdma tag for Tx buffers\n");
10782306b72aSZbigniew Bodek 		goto error;
10793c0086b8SZbigniew Bodek 	}
10803c0086b8SZbigniew Bodek 
10812306b72aSZbigniew Bodek 	/* Allocate send buffers array */
10822306b72aSZbigniew Bodek 	sq->snd_buff = malloc(sizeof(*sq->snd_buff) * q_len, M_NICVF,
10832306b72aSZbigniew Bodek 	    (M_NOWAIT | M_ZERO));
10842306b72aSZbigniew Bodek 	if (sq->snd_buff == NULL) {
10852306b72aSZbigniew Bodek 		device_printf(nic->dev,
10862306b72aSZbigniew Bodek 		    "Could not allocate memory for Tx buffers array\n");
10872306b72aSZbigniew Bodek 		err = ENOMEM;
10882306b72aSZbigniew Bodek 		goto error;
10892306b72aSZbigniew Bodek 	}
10902306b72aSZbigniew Bodek 
10912306b72aSZbigniew Bodek 	/* Now populate maps */
10922306b72aSZbigniew Bodek 	for (i = 0; i < q_len; i++) {
10932306b72aSZbigniew Bodek 		err = bus_dmamap_create(sq->snd_buff_dmat, 0,
10942306b72aSZbigniew Bodek 		    &sq->snd_buff[i].dmap);
10952306b72aSZbigniew Bodek 		if (err != 0) {
10962306b72aSZbigniew Bodek 			device_printf(nic->dev,
10972306b72aSZbigniew Bodek 			    "Failed to create DMA maps for Tx buffers\n");
10982306b72aSZbigniew Bodek 			goto error;
10992306b72aSZbigniew Bodek 		}
11002306b72aSZbigniew Bodek 	}
11012306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
11022306b72aSZbigniew Bodek 
11032306b72aSZbigniew Bodek 	/* Allocate taskqueue */
11042306b72aSZbigniew Bodek 	TASK_INIT(&sq->snd_task, 0, nicvf_snd_task, sq);
11052306b72aSZbigniew Bodek 	sq->snd_taskq = taskqueue_create_fast("nicvf_snd_taskq", M_WAITOK,
11062306b72aSZbigniew Bodek 	    taskqueue_thread_enqueue, &sq->snd_taskq);
11072306b72aSZbigniew Bodek 	taskqueue_start_threads(&sq->snd_taskq, 1, PI_NET, "%s: snd_taskq(%d)",
11082306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev), qidx);
11092306b72aSZbigniew Bodek 
11102306b72aSZbigniew Bodek 	return (0);
11112306b72aSZbigniew Bodek error:
11122306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
11132306b72aSZbigniew Bodek 	return (err);
11142306b72aSZbigniew Bodek }
11152306b72aSZbigniew Bodek 
11162306b72aSZbigniew Bodek static void
11172306b72aSZbigniew Bodek nicvf_free_snd_queue(struct nicvf *nic, struct snd_queue *sq)
11183c0086b8SZbigniew Bodek {
11192306b72aSZbigniew Bodek 	struct queue_set *qs = nic->qs;
11202306b72aSZbigniew Bodek 	size_t i;
11212306b72aSZbigniew Bodek 	int err;
11222306b72aSZbigniew Bodek 
11232306b72aSZbigniew Bodek 	if (sq == NULL)
11243c0086b8SZbigniew Bodek 		return;
11253c0086b8SZbigniew Bodek 
11262306b72aSZbigniew Bodek 	if (sq->snd_taskq != NULL) {
11272306b72aSZbigniew Bodek 		/* Remove task */
11282306b72aSZbigniew Bodek 		while (taskqueue_cancel(sq->snd_taskq, &sq->snd_task, NULL) != 0)
11292306b72aSZbigniew Bodek 			taskqueue_drain(sq->snd_taskq, &sq->snd_task);
11303c0086b8SZbigniew Bodek 
11312306b72aSZbigniew Bodek 		taskqueue_free(sq->snd_taskq);
11322306b72aSZbigniew Bodek 		sq->snd_taskq = NULL;
11332306b72aSZbigniew Bodek 	}
11342306b72aSZbigniew Bodek 
11352306b72aSZbigniew Bodek 	NICVF_TX_LOCK(sq);
11362306b72aSZbigniew Bodek 	if (sq->snd_buff_dmat != NULL) {
11372306b72aSZbigniew Bodek 		if (sq->snd_buff != NULL) {
11382306b72aSZbigniew Bodek 			for (i = 0; i < qs->sq_len; i++) {
11392306b72aSZbigniew Bodek 				m_freem(sq->snd_buff[i].mbuf);
11402306b72aSZbigniew Bodek 				sq->snd_buff[i].mbuf = NULL;
11412306b72aSZbigniew Bodek 
11422306b72aSZbigniew Bodek 				bus_dmamap_unload(sq->snd_buff_dmat,
11432306b72aSZbigniew Bodek 				    sq->snd_buff[i].dmap);
11442306b72aSZbigniew Bodek 				err = bus_dmamap_destroy(sq->snd_buff_dmat,
11452306b72aSZbigniew Bodek 				    sq->snd_buff[i].dmap);
11462306b72aSZbigniew Bodek 				/*
11472306b72aSZbigniew Bodek 				 * If bus_dmamap_destroy fails it can cause
11482306b72aSZbigniew Bodek 				 * random panic later if the tag is also
11492306b72aSZbigniew Bodek 				 * destroyed in the process.
11502306b72aSZbigniew Bodek 				 */
11512306b72aSZbigniew Bodek 				KASSERT(err == 0,
11522306b72aSZbigniew Bodek 				    ("%s: Could not destroy DMA map for SQ",
11532306b72aSZbigniew Bodek 				    __func__));
11542306b72aSZbigniew Bodek 			}
11552306b72aSZbigniew Bodek 		}
11562306b72aSZbigniew Bodek 
11572306b72aSZbigniew Bodek 		free(sq->snd_buff, M_NICVF);
11582306b72aSZbigniew Bodek 
11592306b72aSZbigniew Bodek 		err = bus_dma_tag_destroy(sq->snd_buff_dmat);
11602306b72aSZbigniew Bodek 		KASSERT(err == 0,
11612306b72aSZbigniew Bodek 		    ("%s: Trying to destroy BUSY DMA tag", __func__));
11622306b72aSZbigniew Bodek 	}
11632306b72aSZbigniew Bodek 
11642306b72aSZbigniew Bodek 	/* Free private driver ring for this send queue */
11652306b72aSZbigniew Bodek 	if (sq->br != NULL)
11662306b72aSZbigniew Bodek 		drbr_free(sq->br, M_DEVBUF);
11672306b72aSZbigniew Bodek 
11682306b72aSZbigniew Bodek 	if (sq->dmem.base != NULL)
11693c0086b8SZbigniew Bodek 		nicvf_free_q_desc_mem(nic, &sq->dmem);
11702306b72aSZbigniew Bodek 
11712306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
11722306b72aSZbigniew Bodek 	/* Destroy Tx lock */
11732306b72aSZbigniew Bodek 	mtx_destroy(&sq->mtx);
11742306b72aSZbigniew Bodek 	memset(sq->mtx_name, 0, sizeof(sq->mtx_name));
11753c0086b8SZbigniew Bodek }
11763c0086b8SZbigniew Bodek 
11772306b72aSZbigniew Bodek static void
11782306b72aSZbigniew Bodek nicvf_reclaim_snd_queue(struct nicvf *nic, struct queue_set *qs, int qidx)
11793c0086b8SZbigniew Bodek {
11802306b72aSZbigniew Bodek 
11813c0086b8SZbigniew Bodek 	/* Disable send queue */
11823c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, 0);
11833c0086b8SZbigniew Bodek 	/* Check if SQ is stopped */
11843c0086b8SZbigniew Bodek 	if (nicvf_poll_reg(nic, qidx, NIC_QSET_SQ_0_7_STATUS, 21, 1, 0x01))
11853c0086b8SZbigniew Bodek 		return;
11863c0086b8SZbigniew Bodek 	/* Reset send queue */
11873c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, NICVF_SQ_RESET);
11883c0086b8SZbigniew Bodek }
11893c0086b8SZbigniew Bodek 
11902306b72aSZbigniew Bodek static void
11912306b72aSZbigniew Bodek nicvf_reclaim_rcv_queue(struct nicvf *nic, struct queue_set *qs, int qidx)
11923c0086b8SZbigniew Bodek {
11933c0086b8SZbigniew Bodek 	union nic_mbx mbx = {};
11943c0086b8SZbigniew Bodek 
11953c0086b8SZbigniew Bodek 	/* Make sure all packets in the pipeline are written back into mem */
11963c0086b8SZbigniew Bodek 	mbx.msg.msg = NIC_MBOX_MSG_RQ_SW_SYNC;
11973c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
11983c0086b8SZbigniew Bodek }
11993c0086b8SZbigniew Bodek 
12002306b72aSZbigniew Bodek static void
12012306b72aSZbigniew Bodek nicvf_reclaim_cmp_queue(struct nicvf *nic, struct queue_set *qs, int qidx)
12023c0086b8SZbigniew Bodek {
12032306b72aSZbigniew Bodek 
12043c0086b8SZbigniew Bodek 	/* Disable timer threshold (doesn't get reset upon CQ reset */
12053c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG2, qidx, 0);
12063c0086b8SZbigniew Bodek 	/* Disable completion queue */
12073c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG, qidx, 0);
12083c0086b8SZbigniew Bodek 	/* Reset completion queue */
12093c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG, qidx, NICVF_CQ_RESET);
12103c0086b8SZbigniew Bodek }
12113c0086b8SZbigniew Bodek 
12122306b72aSZbigniew Bodek static void
12132306b72aSZbigniew Bodek nicvf_reclaim_rbdr(struct nicvf *nic, struct rbdr *rbdr, int qidx)
12143c0086b8SZbigniew Bodek {
12152306b72aSZbigniew Bodek 	uint64_t tmp, fifo_state;
12163c0086b8SZbigniew Bodek 	int timeout = 10;
12173c0086b8SZbigniew Bodek 
12183c0086b8SZbigniew Bodek 	/* Save head and tail pointers for feeing up buffers */
12192306b72aSZbigniew Bodek 	rbdr->head =
12202306b72aSZbigniew Bodek 	    nicvf_queue_reg_read(nic, NIC_QSET_RBDR_0_1_HEAD, qidx) >> 3;
12212306b72aSZbigniew Bodek 	rbdr->tail =
12222306b72aSZbigniew Bodek 	    nicvf_queue_reg_read(nic, NIC_QSET_RBDR_0_1_TAIL, qidx) >> 3;
12233c0086b8SZbigniew Bodek 
12242306b72aSZbigniew Bodek 	/*
12252306b72aSZbigniew Bodek 	 * If RBDR FIFO is in 'FAIL' state then do a reset first
12263c0086b8SZbigniew Bodek 	 * before relaiming.
12273c0086b8SZbigniew Bodek 	 */
12283c0086b8SZbigniew Bodek 	fifo_state = nicvf_queue_reg_read(nic, NIC_QSET_RBDR_0_1_STATUS0, qidx);
12292306b72aSZbigniew Bodek 	if (((fifo_state >> 62) & 0x03) == 0x3) {
12303c0086b8SZbigniew Bodek 		nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_CFG,
12313c0086b8SZbigniew Bodek 		    qidx, NICVF_RBDR_RESET);
12322306b72aSZbigniew Bodek 	}
12333c0086b8SZbigniew Bodek 
12343c0086b8SZbigniew Bodek 	/* Disable RBDR */
12353c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_CFG, qidx, 0);
12363c0086b8SZbigniew Bodek 	if (nicvf_poll_reg(nic, qidx, NIC_QSET_RBDR_0_1_STATUS0, 62, 2, 0x00))
12373c0086b8SZbigniew Bodek 		return;
12383c0086b8SZbigniew Bodek 	while (1) {
12393c0086b8SZbigniew Bodek 		tmp = nicvf_queue_reg_read(nic,
12402306b72aSZbigniew Bodek 		    NIC_QSET_RBDR_0_1_PREFETCH_STATUS, qidx);
12413c0086b8SZbigniew Bodek 		if ((tmp & 0xFFFFFFFF) == ((tmp >> 32) & 0xFFFFFFFF))
12423c0086b8SZbigniew Bodek 			break;
12432306b72aSZbigniew Bodek 
12442306b72aSZbigniew Bodek 		DELAY(1000);
12453c0086b8SZbigniew Bodek 		timeout--;
12463c0086b8SZbigniew Bodek 		if (!timeout) {
12472306b72aSZbigniew Bodek 			device_printf(nic->dev,
12483c0086b8SZbigniew Bodek 			    "Failed polling on prefetch status\n");
12493c0086b8SZbigniew Bodek 			return;
12503c0086b8SZbigniew Bodek 		}
12513c0086b8SZbigniew Bodek 	}
12522306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_CFG, qidx,
12532306b72aSZbigniew Bodek 	    NICVF_RBDR_RESET);
12543c0086b8SZbigniew Bodek 
12553c0086b8SZbigniew Bodek 	if (nicvf_poll_reg(nic, qidx, NIC_QSET_RBDR_0_1_STATUS0, 62, 2, 0x02))
12563c0086b8SZbigniew Bodek 		return;
12573c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_CFG, qidx, 0x00);
12583c0086b8SZbigniew Bodek 	if (nicvf_poll_reg(nic, qidx, NIC_QSET_RBDR_0_1_STATUS0, 62, 2, 0x00))
12593c0086b8SZbigniew Bodek 		return;
12603c0086b8SZbigniew Bodek }
12613c0086b8SZbigniew Bodek 
12623c0086b8SZbigniew Bodek /* Configures receive queue */
12632306b72aSZbigniew Bodek static void
12642306b72aSZbigniew Bodek nicvf_rcv_queue_config(struct nicvf *nic, struct queue_set *qs,
12653c0086b8SZbigniew Bodek     int qidx, bool enable)
12663c0086b8SZbigniew Bodek {
12673c0086b8SZbigniew Bodek 	union nic_mbx mbx = {};
12683c0086b8SZbigniew Bodek 	struct rcv_queue *rq;
12693c0086b8SZbigniew Bodek 	struct rq_cfg rq_cfg;
1270053f3d0eSZbigniew Bodek 	struct ifnet *ifp;
1271053f3d0eSZbigniew Bodek 	struct lro_ctrl	*lro;
1272053f3d0eSZbigniew Bodek 
1273053f3d0eSZbigniew Bodek 	ifp = nic->ifp;
12743c0086b8SZbigniew Bodek 
12753c0086b8SZbigniew Bodek 	rq = &qs->rq[qidx];
12763c0086b8SZbigniew Bodek 	rq->enable = enable;
12773c0086b8SZbigniew Bodek 
1278053f3d0eSZbigniew Bodek 	lro = &rq->lro;
1279053f3d0eSZbigniew Bodek 
12803c0086b8SZbigniew Bodek 	/* Disable receive queue */
12813c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RQ_0_7_CFG, qidx, 0);
12823c0086b8SZbigniew Bodek 
12833c0086b8SZbigniew Bodek 	if (!rq->enable) {
12843c0086b8SZbigniew Bodek 		nicvf_reclaim_rcv_queue(nic, qs, qidx);
1285053f3d0eSZbigniew Bodek 		/* Free LRO memory */
1286053f3d0eSZbigniew Bodek 		tcp_lro_free(lro);
1287053f3d0eSZbigniew Bodek 		rq->lro_enabled = FALSE;
12883c0086b8SZbigniew Bodek 		return;
12893c0086b8SZbigniew Bodek 	}
12903c0086b8SZbigniew Bodek 
1291053f3d0eSZbigniew Bodek 	/* Configure LRO if enabled */
1292053f3d0eSZbigniew Bodek 	rq->lro_enabled = FALSE;
1293053f3d0eSZbigniew Bodek 	if ((if_getcapenable(ifp) & IFCAP_LRO) != 0) {
1294053f3d0eSZbigniew Bodek 		if (tcp_lro_init(lro) != 0) {
1295053f3d0eSZbigniew Bodek 			device_printf(nic->dev,
1296053f3d0eSZbigniew Bodek 			    "Failed to initialize LRO for RXQ%d\n", qidx);
1297053f3d0eSZbigniew Bodek 		} else {
1298053f3d0eSZbigniew Bodek 			rq->lro_enabled = TRUE;
1299053f3d0eSZbigniew Bodek 			lro->ifp = nic->ifp;
1300053f3d0eSZbigniew Bodek 		}
1301053f3d0eSZbigniew Bodek 	}
1302053f3d0eSZbigniew Bodek 
13033c0086b8SZbigniew Bodek 	rq->cq_qs = qs->vnic_id;
13043c0086b8SZbigniew Bodek 	rq->cq_idx = qidx;
13053c0086b8SZbigniew Bodek 	rq->start_rbdr_qs = qs->vnic_id;
13063c0086b8SZbigniew Bodek 	rq->start_qs_rbdr_idx = qs->rbdr_cnt - 1;
13073c0086b8SZbigniew Bodek 	rq->cont_rbdr_qs = qs->vnic_id;
13083c0086b8SZbigniew Bodek 	rq->cont_qs_rbdr_idx = qs->rbdr_cnt - 1;
13093c0086b8SZbigniew Bodek 	/* all writes of RBDR data to be loaded into L2 Cache as well*/
13103c0086b8SZbigniew Bodek 	rq->caching = 1;
13113c0086b8SZbigniew Bodek 
13123c0086b8SZbigniew Bodek 	/* Send a mailbox msg to PF to config RQ */
13133c0086b8SZbigniew Bodek 	mbx.rq.msg = NIC_MBOX_MSG_RQ_CFG;
13143c0086b8SZbigniew Bodek 	mbx.rq.qs_num = qs->vnic_id;
13153c0086b8SZbigniew Bodek 	mbx.rq.rq_num = qidx;
13163c0086b8SZbigniew Bodek 	mbx.rq.cfg = (rq->caching << 26) | (rq->cq_qs << 19) |
13173c0086b8SZbigniew Bodek 	    (rq->cq_idx << 16) | (rq->cont_rbdr_qs << 9) |
13182306b72aSZbigniew Bodek 	    (rq->cont_qs_rbdr_idx << 8) | (rq->start_rbdr_qs << 1) |
13192306b72aSZbigniew Bodek 	    (rq->start_qs_rbdr_idx);
13203c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
13213c0086b8SZbigniew Bodek 
13223c0086b8SZbigniew Bodek 	mbx.rq.msg = NIC_MBOX_MSG_RQ_BP_CFG;
13232306b72aSZbigniew Bodek 	mbx.rq.cfg = (1UL << 63) | (1UL << 62) | (qs->vnic_id << 0);
13243c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
13253c0086b8SZbigniew Bodek 
13262306b72aSZbigniew Bodek 	/*
13272306b72aSZbigniew Bodek 	 * RQ drop config
13283c0086b8SZbigniew Bodek 	 * Enable CQ drop to reserve sufficient CQEs for all tx packets
13293c0086b8SZbigniew Bodek 	 */
13303c0086b8SZbigniew Bodek 	mbx.rq.msg = NIC_MBOX_MSG_RQ_DROP_CFG;
13312306b72aSZbigniew Bodek 	mbx.rq.cfg = (1UL << 62) | (RQ_CQ_DROP << 8);
13323c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
13333c0086b8SZbigniew Bodek 
13343c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RQ_GEN_CFG, 0, 0x00);
13353c0086b8SZbigniew Bodek 
13363c0086b8SZbigniew Bodek 	/* Enable Receive queue */
13373c0086b8SZbigniew Bodek 	rq_cfg.ena = 1;
13383c0086b8SZbigniew Bodek 	rq_cfg.tcp_ena = 0;
13392306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RQ_0_7_CFG, qidx,
13402306b72aSZbigniew Bodek 	    *(uint64_t *)&rq_cfg);
13413c0086b8SZbigniew Bodek }
13423c0086b8SZbigniew Bodek 
13433c0086b8SZbigniew Bodek /* Configures completion queue */
13442306b72aSZbigniew Bodek static void
13452306b72aSZbigniew Bodek nicvf_cmp_queue_config(struct nicvf *nic, struct queue_set *qs,
13462306b72aSZbigniew Bodek     int qidx, boolean_t enable)
13473c0086b8SZbigniew Bodek {
13483c0086b8SZbigniew Bodek 	struct cmp_queue *cq;
13493c0086b8SZbigniew Bodek 	struct cq_cfg cq_cfg;
13503c0086b8SZbigniew Bodek 
13513c0086b8SZbigniew Bodek 	cq = &qs->cq[qidx];
13523c0086b8SZbigniew Bodek 	cq->enable = enable;
13533c0086b8SZbigniew Bodek 
13543c0086b8SZbigniew Bodek 	if (!cq->enable) {
13553c0086b8SZbigniew Bodek 		nicvf_reclaim_cmp_queue(nic, qs, qidx);
13563c0086b8SZbigniew Bodek 		return;
13573c0086b8SZbigniew Bodek 	}
13583c0086b8SZbigniew Bodek 
13593c0086b8SZbigniew Bodek 	/* Reset completion queue */
13603c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG, qidx, NICVF_CQ_RESET);
13613c0086b8SZbigniew Bodek 
13623c0086b8SZbigniew Bodek 	/* Set completion queue base address */
13632306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_BASE, qidx,
13642306b72aSZbigniew Bodek 	    (uint64_t)(cq->dmem.phys_base));
13653c0086b8SZbigniew Bodek 
13663c0086b8SZbigniew Bodek 	/* Enable Completion queue */
13673c0086b8SZbigniew Bodek 	cq_cfg.ena = 1;
13683c0086b8SZbigniew Bodek 	cq_cfg.reset = 0;
13693c0086b8SZbigniew Bodek 	cq_cfg.caching = 0;
13703c0086b8SZbigniew Bodek 	cq_cfg.qsize = CMP_QSIZE;
13713c0086b8SZbigniew Bodek 	cq_cfg.avg_con = 0;
13722306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG, qidx, *(uint64_t *)&cq_cfg);
13733c0086b8SZbigniew Bodek 
13743c0086b8SZbigniew Bodek 	/* Set threshold value for interrupt generation */
13753c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_THRESH, qidx, cq->thresh);
13762306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_CQ_0_7_CFG2, qidx,
13772306b72aSZbigniew Bodek 	    nic->cq_coalesce_usecs);
13783c0086b8SZbigniew Bodek }
13793c0086b8SZbigniew Bodek 
13803c0086b8SZbigniew Bodek /* Configures transmit queue */
13812306b72aSZbigniew Bodek static void
13822306b72aSZbigniew Bodek nicvf_snd_queue_config(struct nicvf *nic, struct queue_set *qs, int qidx,
13832306b72aSZbigniew Bodek     boolean_t enable)
13843c0086b8SZbigniew Bodek {
13853c0086b8SZbigniew Bodek 	union nic_mbx mbx = {};
13863c0086b8SZbigniew Bodek 	struct snd_queue *sq;
13873c0086b8SZbigniew Bodek 	struct sq_cfg sq_cfg;
13883c0086b8SZbigniew Bodek 
13893c0086b8SZbigniew Bodek 	sq = &qs->sq[qidx];
13903c0086b8SZbigniew Bodek 	sq->enable = enable;
13913c0086b8SZbigniew Bodek 
13923c0086b8SZbigniew Bodek 	if (!sq->enable) {
13933c0086b8SZbigniew Bodek 		nicvf_reclaim_snd_queue(nic, qs, qidx);
13943c0086b8SZbigniew Bodek 		return;
13953c0086b8SZbigniew Bodek 	}
13963c0086b8SZbigniew Bodek 
13973c0086b8SZbigniew Bodek 	/* Reset send queue */
13983c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, NICVF_SQ_RESET);
13993c0086b8SZbigniew Bodek 
14003c0086b8SZbigniew Bodek 	sq->cq_qs = qs->vnic_id;
14013c0086b8SZbigniew Bodek 	sq->cq_idx = qidx;
14023c0086b8SZbigniew Bodek 
14033c0086b8SZbigniew Bodek 	/* Send a mailbox msg to PF to config SQ */
14043c0086b8SZbigniew Bodek 	mbx.sq.msg = NIC_MBOX_MSG_SQ_CFG;
14053c0086b8SZbigniew Bodek 	mbx.sq.qs_num = qs->vnic_id;
14063c0086b8SZbigniew Bodek 	mbx.sq.sq_num = qidx;
14073c0086b8SZbigniew Bodek 	mbx.sq.sqs_mode = nic->sqs_mode;
14083c0086b8SZbigniew Bodek 	mbx.sq.cfg = (sq->cq_qs << 3) | sq->cq_idx;
14093c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
14103c0086b8SZbigniew Bodek 
14113c0086b8SZbigniew Bodek 	/* Set queue base address */
14122306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_BASE, qidx,
14132306b72aSZbigniew Bodek 	    (uint64_t)(sq->dmem.phys_base));
14143c0086b8SZbigniew Bodek 
14153c0086b8SZbigniew Bodek 	/* Enable send queue  & set queue size */
14163c0086b8SZbigniew Bodek 	sq_cfg.ena = 1;
14173c0086b8SZbigniew Bodek 	sq_cfg.reset = 0;
14183c0086b8SZbigniew Bodek 	sq_cfg.ldwb = 0;
14193c0086b8SZbigniew Bodek 	sq_cfg.qsize = SND_QSIZE;
14203c0086b8SZbigniew Bodek 	sq_cfg.tstmp_bgx_intf = 0;
14212306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, *(uint64_t *)&sq_cfg);
14223c0086b8SZbigniew Bodek 
14233c0086b8SZbigniew Bodek 	/* Set threshold value for interrupt generation */
14243c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_THRESH, qidx, sq->thresh);
14253c0086b8SZbigniew Bodek }
14263c0086b8SZbigniew Bodek 
14273c0086b8SZbigniew Bodek /* Configures receive buffer descriptor ring */
14282306b72aSZbigniew Bodek static void
14292306b72aSZbigniew Bodek nicvf_rbdr_config(struct nicvf *nic, struct queue_set *qs, int qidx,
14302306b72aSZbigniew Bodek     boolean_t enable)
14313c0086b8SZbigniew Bodek {
14323c0086b8SZbigniew Bodek 	struct rbdr *rbdr;
14333c0086b8SZbigniew Bodek 	struct rbdr_cfg rbdr_cfg;
14343c0086b8SZbigniew Bodek 
14353c0086b8SZbigniew Bodek 	rbdr = &qs->rbdr[qidx];
14363c0086b8SZbigniew Bodek 	nicvf_reclaim_rbdr(nic, rbdr, qidx);
14373c0086b8SZbigniew Bodek 	if (!enable)
14383c0086b8SZbigniew Bodek 		return;
14393c0086b8SZbigniew Bodek 
14403c0086b8SZbigniew Bodek 	/* Set descriptor base address */
14412306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_BASE, qidx,
14422306b72aSZbigniew Bodek 	    (uint64_t)(rbdr->dmem.phys_base));
14433c0086b8SZbigniew Bodek 
14443c0086b8SZbigniew Bodek 	/* Enable RBDR  & set queue size */
14453c0086b8SZbigniew Bodek 	/* Buffer size should be in multiples of 128 bytes */
14463c0086b8SZbigniew Bodek 	rbdr_cfg.ena = 1;
14473c0086b8SZbigniew Bodek 	rbdr_cfg.reset = 0;
14483c0086b8SZbigniew Bodek 	rbdr_cfg.ldwb = 0;
14493c0086b8SZbigniew Bodek 	rbdr_cfg.qsize = RBDR_SIZE;
14503c0086b8SZbigniew Bodek 	rbdr_cfg.avg_con = 0;
14513c0086b8SZbigniew Bodek 	rbdr_cfg.lines = rbdr->dma_size / 128;
14522306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_CFG, qidx,
14532306b72aSZbigniew Bodek 	    *(uint64_t *)&rbdr_cfg);
14543c0086b8SZbigniew Bodek 
14553c0086b8SZbigniew Bodek 	/* Notify HW */
14562306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_DOOR, qidx,
14572306b72aSZbigniew Bodek 	    qs->rbdr_len - 1);
14583c0086b8SZbigniew Bodek 
14593c0086b8SZbigniew Bodek 	/* Set threshold value for interrupt generation */
14602306b72aSZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_RBDR_0_1_THRESH, qidx,
14612306b72aSZbigniew Bodek 	    rbdr->thresh - 1);
14623c0086b8SZbigniew Bodek }
14633c0086b8SZbigniew Bodek 
14643c0086b8SZbigniew Bodek /* Requests PF to assign and enable Qset */
14652306b72aSZbigniew Bodek void
14662306b72aSZbigniew Bodek nicvf_qset_config(struct nicvf *nic, boolean_t enable)
14673c0086b8SZbigniew Bodek {
14683c0086b8SZbigniew Bodek 	union nic_mbx mbx = {};
14692306b72aSZbigniew Bodek 	struct queue_set *qs;
14703c0086b8SZbigniew Bodek 	struct qs_cfg *qs_cfg;
14713c0086b8SZbigniew Bodek 
14722306b72aSZbigniew Bodek 	qs = nic->qs;
14732306b72aSZbigniew Bodek 	if (qs == NULL) {
14742306b72aSZbigniew Bodek 		device_printf(nic->dev,
14753c0086b8SZbigniew Bodek 		    "Qset is still not allocated, don't init queues\n");
14763c0086b8SZbigniew Bodek 		return;
14773c0086b8SZbigniew Bodek 	}
14783c0086b8SZbigniew Bodek 
14793c0086b8SZbigniew Bodek 	qs->enable = enable;
14803c0086b8SZbigniew Bodek 	qs->vnic_id = nic->vf_id;
14813c0086b8SZbigniew Bodek 
14823c0086b8SZbigniew Bodek 	/* Send a mailbox msg to PF to config Qset */
14833c0086b8SZbigniew Bodek 	mbx.qs.msg = NIC_MBOX_MSG_QS_CFG;
14843c0086b8SZbigniew Bodek 	mbx.qs.num = qs->vnic_id;
14853c0086b8SZbigniew Bodek 
14863c0086b8SZbigniew Bodek 	mbx.qs.cfg = 0;
14873c0086b8SZbigniew Bodek 	qs_cfg = (struct qs_cfg *)&mbx.qs.cfg;
14883c0086b8SZbigniew Bodek 	if (qs->enable) {
14893c0086b8SZbigniew Bodek 		qs_cfg->ena = 1;
14903c0086b8SZbigniew Bodek 		qs_cfg->vnic = qs->vnic_id;
14913c0086b8SZbigniew Bodek 	}
14923c0086b8SZbigniew Bodek 	nicvf_send_msg_to_pf(nic, &mbx);
14933c0086b8SZbigniew Bodek }
14943c0086b8SZbigniew Bodek 
14952306b72aSZbigniew Bodek static void
14962306b72aSZbigniew Bodek nicvf_free_resources(struct nicvf *nic)
14973c0086b8SZbigniew Bodek {
14983c0086b8SZbigniew Bodek 	int qidx;
14992306b72aSZbigniew Bodek 	struct queue_set *qs;
15003c0086b8SZbigniew Bodek 
15012306b72aSZbigniew Bodek 	qs = nic->qs;
15022306b72aSZbigniew Bodek 	/*
15032306b72aSZbigniew Bodek 	 * Remove QS error task first since it has to be dead
15042306b72aSZbigniew Bodek 	 * to safely free completion queue tasks.
15052306b72aSZbigniew Bodek 	 */
15062306b72aSZbigniew Bodek 	if (qs->qs_err_taskq != NULL) {
15072306b72aSZbigniew Bodek 		/* Shut down QS error tasks */
15082306b72aSZbigniew Bodek 		while (taskqueue_cancel(qs->qs_err_taskq,
15092306b72aSZbigniew Bodek 		    &qs->qs_err_task,  NULL) != 0) {
15102306b72aSZbigniew Bodek 			taskqueue_drain(qs->qs_err_taskq, &qs->qs_err_task);
15112306b72aSZbigniew Bodek 
15122306b72aSZbigniew Bodek 		}
15132306b72aSZbigniew Bodek 		taskqueue_free(qs->qs_err_taskq);
15142306b72aSZbigniew Bodek 		qs->qs_err_taskq = NULL;
15152306b72aSZbigniew Bodek 	}
15163c0086b8SZbigniew Bodek 	/* Free receive buffer descriptor ring */
15173c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->rbdr_cnt; qidx++)
15183c0086b8SZbigniew Bodek 		nicvf_free_rbdr(nic, &qs->rbdr[qidx]);
15193c0086b8SZbigniew Bodek 
15203c0086b8SZbigniew Bodek 	/* Free completion queue */
15213c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->cq_cnt; qidx++)
15223c0086b8SZbigniew Bodek 		nicvf_free_cmp_queue(nic, &qs->cq[qidx]);
15233c0086b8SZbigniew Bodek 
15243c0086b8SZbigniew Bodek 	/* Free send queue */
15253c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->sq_cnt; qidx++)
15263c0086b8SZbigniew Bodek 		nicvf_free_snd_queue(nic, &qs->sq[qidx]);
15273c0086b8SZbigniew Bodek }
15283c0086b8SZbigniew Bodek 
15292306b72aSZbigniew Bodek static int
15302306b72aSZbigniew Bodek nicvf_alloc_resources(struct nicvf *nic)
15313c0086b8SZbigniew Bodek {
15323c0086b8SZbigniew Bodek 	struct queue_set *qs = nic->qs;
15332306b72aSZbigniew Bodek 	int qidx;
15343c0086b8SZbigniew Bodek 
15353c0086b8SZbigniew Bodek 	/* Alloc receive buffer descriptor ring */
15363c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->rbdr_cnt; qidx++) {
15373c0086b8SZbigniew Bodek 		if (nicvf_init_rbdr(nic, &qs->rbdr[qidx], qs->rbdr_len,
15382306b72aSZbigniew Bodek 				    DMA_BUFFER_LEN, qidx))
15393c0086b8SZbigniew Bodek 			goto alloc_fail;
15403c0086b8SZbigniew Bodek 	}
15413c0086b8SZbigniew Bodek 
15423c0086b8SZbigniew Bodek 	/* Alloc send queue */
15433c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->sq_cnt; qidx++) {
15442306b72aSZbigniew Bodek 		if (nicvf_init_snd_queue(nic, &qs->sq[qidx], qs->sq_len, qidx))
15453c0086b8SZbigniew Bodek 			goto alloc_fail;
15463c0086b8SZbigniew Bodek 	}
15473c0086b8SZbigniew Bodek 
15483c0086b8SZbigniew Bodek 	/* Alloc completion queue */
15493c0086b8SZbigniew Bodek 	for (qidx = 0; qidx < qs->cq_cnt; qidx++) {
15502306b72aSZbigniew Bodek 		if (nicvf_init_cmp_queue(nic, &qs->cq[qidx], qs->cq_len, qidx))
15513c0086b8SZbigniew Bodek 			goto alloc_fail;
15523c0086b8SZbigniew Bodek 	}
15533c0086b8SZbigniew Bodek 
15542306b72aSZbigniew Bodek 	/* Allocate QS error taskqueue */
15552306b72aSZbigniew Bodek 	TASK_INIT(&qs->qs_err_task, 0, nicvf_qs_err_task, nic);
15562306b72aSZbigniew Bodek 	qs->qs_err_taskq = taskqueue_create_fast("nicvf_qs_err_taskq", M_WAITOK,
15572306b72aSZbigniew Bodek 	    taskqueue_thread_enqueue, &qs->qs_err_taskq);
15582306b72aSZbigniew Bodek 	taskqueue_start_threads(&qs->qs_err_taskq, 1, PI_NET, "%s: qs_taskq",
15592306b72aSZbigniew Bodek 	    device_get_nameunit(nic->dev));
15602306b72aSZbigniew Bodek 
15612306b72aSZbigniew Bodek 	return (0);
15623c0086b8SZbigniew Bodek alloc_fail:
15633c0086b8SZbigniew Bodek 	nicvf_free_resources(nic);
15642306b72aSZbigniew Bodek 	return (ENOMEM);
15653c0086b8SZbigniew Bodek }
15663c0086b8SZbigniew Bodek 
15672306b72aSZbigniew Bodek int
15682306b72aSZbigniew Bodek nicvf_set_qset_resources(struct nicvf *nic)
15693c0086b8SZbigniew Bodek {
15703c0086b8SZbigniew Bodek 	struct queue_set *qs;
15713c0086b8SZbigniew Bodek 
15722306b72aSZbigniew Bodek 	qs = malloc(sizeof(*qs), M_NICVF, (M_ZERO | M_WAITOK));
15733c0086b8SZbigniew Bodek 	nic->qs = qs;
15743c0086b8SZbigniew Bodek 
15753c0086b8SZbigniew Bodek 	/* Set count of each queue */
15763c0086b8SZbigniew Bodek 	qs->rbdr_cnt = RBDR_CNT;
15772306b72aSZbigniew Bodek 	/* With no RSS we stay with single RQ */
15783c0086b8SZbigniew Bodek 	qs->rq_cnt = 1;
15792306b72aSZbigniew Bodek 
15803c0086b8SZbigniew Bodek 	qs->sq_cnt = SND_QUEUE_CNT;
15813c0086b8SZbigniew Bodek 	qs->cq_cnt = CMP_QUEUE_CNT;
15823c0086b8SZbigniew Bodek 
15833c0086b8SZbigniew Bodek 	/* Set queue lengths */
15843c0086b8SZbigniew Bodek 	qs->rbdr_len = RCV_BUF_COUNT;
15853c0086b8SZbigniew Bodek 	qs->sq_len = SND_QUEUE_LEN;
15863c0086b8SZbigniew Bodek 	qs->cq_len = CMP_QUEUE_LEN;
15873c0086b8SZbigniew Bodek 
15883c0086b8SZbigniew Bodek 	nic->rx_queues = qs->rq_cnt;
15893c0086b8SZbigniew Bodek 	nic->tx_queues = qs->sq_cnt;
15903c0086b8SZbigniew Bodek 
15912306b72aSZbigniew Bodek 	return (0);
15923c0086b8SZbigniew Bodek }
15933c0086b8SZbigniew Bodek 
15942306b72aSZbigniew Bodek int
15952306b72aSZbigniew Bodek nicvf_config_data_transfer(struct nicvf *nic, boolean_t enable)
15963c0086b8SZbigniew Bodek {
15972306b72aSZbigniew Bodek 	boolean_t disable = FALSE;
15982306b72aSZbigniew Bodek 	struct queue_set *qs;
15993c0086b8SZbigniew Bodek 	int qidx;
16003c0086b8SZbigniew Bodek 
16012306b72aSZbigniew Bodek 	qs = nic->qs;
16022306b72aSZbigniew Bodek 	if (qs == NULL)
16032306b72aSZbigniew Bodek 		return (0);
16043c0086b8SZbigniew Bodek 
16053c0086b8SZbigniew Bodek 	if (enable) {
16062306b72aSZbigniew Bodek 		if (nicvf_alloc_resources(nic) != 0)
16072306b72aSZbigniew Bodek 			return (ENOMEM);
16083c0086b8SZbigniew Bodek 
16093c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->sq_cnt; qidx++)
16103c0086b8SZbigniew Bodek 			nicvf_snd_queue_config(nic, qs, qidx, enable);
16113c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->cq_cnt; qidx++)
16123c0086b8SZbigniew Bodek 			nicvf_cmp_queue_config(nic, qs, qidx, enable);
16133c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->rbdr_cnt; qidx++)
16143c0086b8SZbigniew Bodek 			nicvf_rbdr_config(nic, qs, qidx, enable);
16153c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->rq_cnt; qidx++)
16163c0086b8SZbigniew Bodek 			nicvf_rcv_queue_config(nic, qs, qidx, enable);
16173c0086b8SZbigniew Bodek 	} else {
16183c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->rq_cnt; qidx++)
16193c0086b8SZbigniew Bodek 			nicvf_rcv_queue_config(nic, qs, qidx, disable);
16203c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->rbdr_cnt; qidx++)
16213c0086b8SZbigniew Bodek 			nicvf_rbdr_config(nic, qs, qidx, disable);
16223c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->sq_cnt; qidx++)
16233c0086b8SZbigniew Bodek 			nicvf_snd_queue_config(nic, qs, qidx, disable);
16243c0086b8SZbigniew Bodek 		for (qidx = 0; qidx < qs->cq_cnt; qidx++)
16253c0086b8SZbigniew Bodek 			nicvf_cmp_queue_config(nic, qs, qidx, disable);
16263c0086b8SZbigniew Bodek 
16273c0086b8SZbigniew Bodek 		nicvf_free_resources(nic);
16283c0086b8SZbigniew Bodek 	}
16293c0086b8SZbigniew Bodek 
16302306b72aSZbigniew Bodek 	return (0);
16313c0086b8SZbigniew Bodek }
16323c0086b8SZbigniew Bodek 
16332306b72aSZbigniew Bodek /*
16342306b72aSZbigniew Bodek  * Get a free desc from SQ
16353c0086b8SZbigniew Bodek  * returns descriptor ponter & descriptor number
16363c0086b8SZbigniew Bodek  */
16372306b72aSZbigniew Bodek static __inline int
16382306b72aSZbigniew Bodek nicvf_get_sq_desc(struct snd_queue *sq, int desc_cnt)
16393c0086b8SZbigniew Bodek {
16403c0086b8SZbigniew Bodek 	int qentry;
16413c0086b8SZbigniew Bodek 
16423c0086b8SZbigniew Bodek 	qentry = sq->tail;
16432306b72aSZbigniew Bodek 	atomic_subtract_int(&sq->free_cnt, desc_cnt);
16443c0086b8SZbigniew Bodek 	sq->tail += desc_cnt;
16453c0086b8SZbigniew Bodek 	sq->tail &= (sq->dmem.q_len - 1);
16463c0086b8SZbigniew Bodek 
16472306b72aSZbigniew Bodek 	return (qentry);
16483c0086b8SZbigniew Bodek }
16493c0086b8SZbigniew Bodek 
16503c0086b8SZbigniew Bodek /* Free descriptor back to SQ for future use */
16512306b72aSZbigniew Bodek static void
16522306b72aSZbigniew Bodek nicvf_put_sq_desc(struct snd_queue *sq, int desc_cnt)
16533c0086b8SZbigniew Bodek {
16542306b72aSZbigniew Bodek 
16552306b72aSZbigniew Bodek 	atomic_add_int(&sq->free_cnt, desc_cnt);
16563c0086b8SZbigniew Bodek 	sq->head += desc_cnt;
16573c0086b8SZbigniew Bodek 	sq->head &= (sq->dmem.q_len - 1);
16583c0086b8SZbigniew Bodek }
16593c0086b8SZbigniew Bodek 
16602306b72aSZbigniew Bodek static __inline int
16612306b72aSZbigniew Bodek nicvf_get_nxt_sqentry(struct snd_queue *sq, int qentry)
16623c0086b8SZbigniew Bodek {
16633c0086b8SZbigniew Bodek 	qentry++;
16643c0086b8SZbigniew Bodek 	qentry &= (sq->dmem.q_len - 1);
16652306b72aSZbigniew Bodek 	return (qentry);
16663c0086b8SZbigniew Bodek }
16673c0086b8SZbigniew Bodek 
16682306b72aSZbigniew Bodek static void
16692306b72aSZbigniew Bodek nicvf_sq_enable(struct nicvf *nic, struct snd_queue *sq, int qidx)
16703c0086b8SZbigniew Bodek {
16712306b72aSZbigniew Bodek 	uint64_t sq_cfg;
16723c0086b8SZbigniew Bodek 
16733c0086b8SZbigniew Bodek 	sq_cfg = nicvf_queue_reg_read(nic, NIC_QSET_SQ_0_7_CFG, qidx);
16743c0086b8SZbigniew Bodek 	sq_cfg |= NICVF_SQ_EN;
16753c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, sq_cfg);
16763c0086b8SZbigniew Bodek 	/* Ring doorbell so that H/W restarts processing SQEs */
16773c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_DOOR, qidx, 0);
16783c0086b8SZbigniew Bodek }
16793c0086b8SZbigniew Bodek 
16802306b72aSZbigniew Bodek static void
16812306b72aSZbigniew Bodek nicvf_sq_disable(struct nicvf *nic, int qidx)
16823c0086b8SZbigniew Bodek {
16832306b72aSZbigniew Bodek 	uint64_t sq_cfg;
16843c0086b8SZbigniew Bodek 
16853c0086b8SZbigniew Bodek 	sq_cfg = nicvf_queue_reg_read(nic, NIC_QSET_SQ_0_7_CFG, qidx);
16863c0086b8SZbigniew Bodek 	sq_cfg &= ~NICVF_SQ_EN;
16873c0086b8SZbigniew Bodek 	nicvf_queue_reg_write(nic, NIC_QSET_SQ_0_7_CFG, qidx, sq_cfg);
16883c0086b8SZbigniew Bodek }
16893c0086b8SZbigniew Bodek 
16902306b72aSZbigniew Bodek static void
16912306b72aSZbigniew Bodek nicvf_sq_free_used_descs(struct nicvf *nic, struct snd_queue *sq, int qidx)
16923c0086b8SZbigniew Bodek {
16932306b72aSZbigniew Bodek 	uint64_t head, tail;
16942306b72aSZbigniew Bodek 	struct snd_buff *snd_buff;
16953c0086b8SZbigniew Bodek 	struct sq_hdr_subdesc *hdr;
16963c0086b8SZbigniew Bodek 
16972306b72aSZbigniew Bodek 	NICVF_TX_LOCK(sq);
16983c0086b8SZbigniew Bodek 	head = nicvf_queue_reg_read(nic, NIC_QSET_SQ_0_7_HEAD, qidx) >> 4;
16993c0086b8SZbigniew Bodek 	tail = nicvf_queue_reg_read(nic, NIC_QSET_SQ_0_7_TAIL, qidx) >> 4;
17003c0086b8SZbigniew Bodek 	while (sq->head != head) {
17013c0086b8SZbigniew Bodek 		hdr = (struct sq_hdr_subdesc *)GET_SQ_DESC(sq, sq->head);
17023c0086b8SZbigniew Bodek 		if (hdr->subdesc_type != SQ_DESC_TYPE_HEADER) {
17033c0086b8SZbigniew Bodek 			nicvf_put_sq_desc(sq, 1);
17043c0086b8SZbigniew Bodek 			continue;
17053c0086b8SZbigniew Bodek 		}
17062306b72aSZbigniew Bodek 		snd_buff = &sq->snd_buff[sq->head];
17072306b72aSZbigniew Bodek 		if (snd_buff->mbuf != NULL) {
17082306b72aSZbigniew Bodek 			bus_dmamap_unload(sq->snd_buff_dmat, snd_buff->dmap);
17092306b72aSZbigniew Bodek 			m_freem(snd_buff->mbuf);
17102306b72aSZbigniew Bodek 			sq->snd_buff[sq->head].mbuf = NULL;
17112306b72aSZbigniew Bodek 		}
17123c0086b8SZbigniew Bodek 		nicvf_put_sq_desc(sq, hdr->subdesc_cnt + 1);
17133c0086b8SZbigniew Bodek 	}
17142306b72aSZbigniew Bodek 	NICVF_TX_UNLOCK(sq);
17153c0086b8SZbigniew Bodek }
17163c0086b8SZbigniew Bodek 
17172306b72aSZbigniew Bodek /*
17182306b72aSZbigniew Bodek  * Add SQ HEADER subdescriptor.
17193c0086b8SZbigniew Bodek  * First subdescriptor for every send descriptor.
17203c0086b8SZbigniew Bodek  */
1721856dce91SZbigniew Bodek static __inline int
17223c0086b8SZbigniew Bodek nicvf_sq_add_hdr_subdesc(struct snd_queue *sq, int qentry,
17232306b72aSZbigniew Bodek 			 int subdesc_cnt, struct mbuf *mbuf, int len)
17243c0086b8SZbigniew Bodek {
1725af8fe8f1SZbigniew Bodek 	struct nicvf *nic;
17263c0086b8SZbigniew Bodek 	struct sq_hdr_subdesc *hdr;
1727856dce91SZbigniew Bodek 	struct ether_vlan_header *eh;
1728856dce91SZbigniew Bodek #ifdef INET
1729856dce91SZbigniew Bodek 	struct ip *ip;
1730af8fe8f1SZbigniew Bodek 	struct tcphdr *th;
1731856dce91SZbigniew Bodek #endif
1732856dce91SZbigniew Bodek 	uint16_t etype;
1733856dce91SZbigniew Bodek 	int ehdrlen, iphlen, poff;
17343c0086b8SZbigniew Bodek 
1735af8fe8f1SZbigniew Bodek 	nic = sq->nic;
1736af8fe8f1SZbigniew Bodek 
17373c0086b8SZbigniew Bodek 	hdr = (struct sq_hdr_subdesc *)GET_SQ_DESC(sq, qentry);
17382306b72aSZbigniew Bodek 	sq->snd_buff[qentry].mbuf = mbuf;
17393c0086b8SZbigniew Bodek 
17403c0086b8SZbigniew Bodek 	memset(hdr, 0, SND_QUEUE_DESC_SIZE);
17413c0086b8SZbigniew Bodek 	hdr->subdesc_type = SQ_DESC_TYPE_HEADER;
17423c0086b8SZbigniew Bodek 	/* Enable notification via CQE after processing SQE */
17433c0086b8SZbigniew Bodek 	hdr->post_cqe = 1;
17443c0086b8SZbigniew Bodek 	/* No of subdescriptors following this */
17453c0086b8SZbigniew Bodek 	hdr->subdesc_cnt = subdesc_cnt;
17463c0086b8SZbigniew Bodek 	hdr->tot_len = len;
17473c0086b8SZbigniew Bodek 
1748856dce91SZbigniew Bodek 	eh = mtod(mbuf, struct ether_vlan_header *);
1749856dce91SZbigniew Bodek 	if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) {
1750856dce91SZbigniew Bodek 		ehdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN;
1751856dce91SZbigniew Bodek 		etype = ntohs(eh->evl_proto);
1752856dce91SZbigniew Bodek 	} else {
1753856dce91SZbigniew Bodek 		ehdrlen = ETHER_HDR_LEN;
1754856dce91SZbigniew Bodek 		etype = ntohs(eh->evl_encap_proto);
1755856dce91SZbigniew Bodek 	}
1756856dce91SZbigniew Bodek 
1757856dce91SZbigniew Bodek 	switch (etype) {
1758856dce91SZbigniew Bodek #ifdef INET6
1759856dce91SZbigniew Bodek 	case ETHERTYPE_IPV6:
1760856dce91SZbigniew Bodek 		/* ARM64TODO: Add support for IPv6 */
1761856dce91SZbigniew Bodek 		hdr->csum_l3 = 0;
1762856dce91SZbigniew Bodek 		sq->snd_buff[qentry].mbuf = NULL;
1763856dce91SZbigniew Bodek 		return (ENXIO);
1764856dce91SZbigniew Bodek #endif
1765856dce91SZbigniew Bodek #ifdef INET
1766856dce91SZbigniew Bodek 	case ETHERTYPE_IP:
1767af8fe8f1SZbigniew Bodek 		if (mbuf->m_len < ehdrlen + sizeof(struct ip)) {
1768af8fe8f1SZbigniew Bodek 			mbuf = m_pullup(mbuf, ehdrlen + sizeof(struct ip));
1769af8fe8f1SZbigniew Bodek 			sq->snd_buff[qentry].mbuf = mbuf;
1770af8fe8f1SZbigniew Bodek 			if (mbuf == NULL)
1771af8fe8f1SZbigniew Bodek 				return (ENOBUFS);
1772af8fe8f1SZbigniew Bodek 		}
1773af8fe8f1SZbigniew Bodek 
1774856dce91SZbigniew Bodek 		ip = (struct ip *)(mbuf->m_data + ehdrlen);
1775856dce91SZbigniew Bodek 		ip->ip_sum = 0;
1776856dce91SZbigniew Bodek 		iphlen = ip->ip_hl << 2;
1777856dce91SZbigniew Bodek 		poff = ehdrlen + iphlen;
1778856dce91SZbigniew Bodek 
1779af8fe8f1SZbigniew Bodek 		if (mbuf->m_pkthdr.csum_flags != 0) {
1780af8fe8f1SZbigniew Bodek 			hdr->csum_l3 = 1; /* Enable IP csum calculation */
1781856dce91SZbigniew Bodek 			switch (ip->ip_p) {
1782856dce91SZbigniew Bodek 			case IPPROTO_TCP:
1783856dce91SZbigniew Bodek 				if ((mbuf->m_pkthdr.csum_flags & CSUM_TCP) == 0)
1784856dce91SZbigniew Bodek 					break;
1785856dce91SZbigniew Bodek 
1786856dce91SZbigniew Bodek 				if (mbuf->m_len < (poff + sizeof(struct tcphdr))) {
1787856dce91SZbigniew Bodek 					mbuf = m_pullup(mbuf, poff + sizeof(struct tcphdr));
1788856dce91SZbigniew Bodek 					sq->snd_buff[qentry].mbuf = mbuf;
1789856dce91SZbigniew Bodek 					if (mbuf == NULL)
1790856dce91SZbigniew Bodek 						return (ENOBUFS);
1791856dce91SZbigniew Bodek 				}
1792856dce91SZbigniew Bodek 				hdr->csum_l4 = SEND_L4_CSUM_TCP;
1793856dce91SZbigniew Bodek 				break;
1794856dce91SZbigniew Bodek 			case IPPROTO_UDP:
1795856dce91SZbigniew Bodek 				if ((mbuf->m_pkthdr.csum_flags & CSUM_UDP) == 0)
1796856dce91SZbigniew Bodek 					break;
1797856dce91SZbigniew Bodek 
1798856dce91SZbigniew Bodek 				if (mbuf->m_len < (poff + sizeof(struct udphdr))) {
1799856dce91SZbigniew Bodek 					mbuf = m_pullup(mbuf, poff + sizeof(struct udphdr));
1800856dce91SZbigniew Bodek 					sq->snd_buff[qentry].mbuf = mbuf;
1801856dce91SZbigniew Bodek 					if (mbuf == NULL)
1802856dce91SZbigniew Bodek 						return (ENOBUFS);
1803856dce91SZbigniew Bodek 				}
1804856dce91SZbigniew Bodek 				hdr->csum_l4 = SEND_L4_CSUM_UDP;
1805856dce91SZbigniew Bodek 				break;
1806856dce91SZbigniew Bodek 			case IPPROTO_SCTP:
1807856dce91SZbigniew Bodek 				if ((mbuf->m_pkthdr.csum_flags & CSUM_SCTP) == 0)
1808856dce91SZbigniew Bodek 					break;
1809856dce91SZbigniew Bodek 
1810856dce91SZbigniew Bodek 				if (mbuf->m_len < (poff + sizeof(struct sctphdr))) {
1811856dce91SZbigniew Bodek 					mbuf = m_pullup(mbuf, poff + sizeof(struct sctphdr));
1812856dce91SZbigniew Bodek 					sq->snd_buff[qentry].mbuf = mbuf;
1813856dce91SZbigniew Bodek 					if (mbuf == NULL)
1814856dce91SZbigniew Bodek 						return (ENOBUFS);
1815856dce91SZbigniew Bodek 				}
1816856dce91SZbigniew Bodek 				hdr->csum_l4 = SEND_L4_CSUM_SCTP;
1817856dce91SZbigniew Bodek 				break;
1818856dce91SZbigniew Bodek 			default:
1819856dce91SZbigniew Bodek 				break;
1820856dce91SZbigniew Bodek 			}
1821af8fe8f1SZbigniew Bodek 			hdr->l3_offset = ehdrlen;
1822af8fe8f1SZbigniew Bodek 			hdr->l4_offset = ehdrlen + iphlen;
1823af8fe8f1SZbigniew Bodek 		}
1824af8fe8f1SZbigniew Bodek 
1825af8fe8f1SZbigniew Bodek 		if ((mbuf->m_pkthdr.tso_segsz != 0) && nic->hw_tso) {
1826af8fe8f1SZbigniew Bodek 			/*
1827af8fe8f1SZbigniew Bodek 			 * Extract ip again as m_data could have been modified.
1828af8fe8f1SZbigniew Bodek 			 */
1829af8fe8f1SZbigniew Bodek 			ip = (struct ip *)(mbuf->m_data + ehdrlen);
1830af8fe8f1SZbigniew Bodek 			th = (struct tcphdr *)((caddr_t)ip + iphlen);
1831af8fe8f1SZbigniew Bodek 
1832af8fe8f1SZbigniew Bodek 			hdr->tso = 1;
1833af8fe8f1SZbigniew Bodek 			hdr->tso_start = ehdrlen + iphlen + (th->th_off * 4);
1834af8fe8f1SZbigniew Bodek 			hdr->tso_max_paysize = mbuf->m_pkthdr.tso_segsz;
1835af8fe8f1SZbigniew Bodek 			hdr->inner_l3_offset = ehdrlen - 2;
1836af8fe8f1SZbigniew Bodek 			nic->drv_stats.tx_tso++;
1837af8fe8f1SZbigniew Bodek 		}
1838856dce91SZbigniew Bodek 		break;
1839856dce91SZbigniew Bodek #endif
1840856dce91SZbigniew Bodek 	default:
1841856dce91SZbigniew Bodek 		hdr->csum_l3 = 0;
1842856dce91SZbigniew Bodek 	}
1843856dce91SZbigniew Bodek 
1844856dce91SZbigniew Bodek 	return (0);
18453c0086b8SZbigniew Bodek }
18463c0086b8SZbigniew Bodek 
18472306b72aSZbigniew Bodek /*
18482306b72aSZbigniew Bodek  * SQ GATHER subdescriptor
18493c0086b8SZbigniew Bodek  * Must follow HDR descriptor
18503c0086b8SZbigniew Bodek  */
18513c0086b8SZbigniew Bodek static inline void nicvf_sq_add_gather_subdesc(struct snd_queue *sq, int qentry,
18522306b72aSZbigniew Bodek 					       int size, uint64_t data)
18533c0086b8SZbigniew Bodek {
18543c0086b8SZbigniew Bodek 	struct sq_gather_subdesc *gather;
18553c0086b8SZbigniew Bodek 
18563c0086b8SZbigniew Bodek 	qentry &= (sq->dmem.q_len - 1);
18573c0086b8SZbigniew Bodek 	gather = (struct sq_gather_subdesc *)GET_SQ_DESC(sq, qentry);
18583c0086b8SZbigniew Bodek 
18593c0086b8SZbigniew Bodek 	memset(gather, 0, SND_QUEUE_DESC_SIZE);
18603c0086b8SZbigniew Bodek 	gather->subdesc_type = SQ_DESC_TYPE_GATHER;
18613c0086b8SZbigniew Bodek 	gather->ld_type = NIC_SEND_LD_TYPE_E_LDD;
18623c0086b8SZbigniew Bodek 	gather->size = size;
18633c0086b8SZbigniew Bodek 	gather->addr = data;
18643c0086b8SZbigniew Bodek }
18653c0086b8SZbigniew Bodek 
18662306b72aSZbigniew Bodek /* Put an mbuf to a SQ for packet transfer. */
1867332c8697SZbigniew Bodek int
18682306b72aSZbigniew Bodek nicvf_tx_mbuf_locked(struct snd_queue *sq, struct mbuf *mbuf)
18693c0086b8SZbigniew Bodek {
18702306b72aSZbigniew Bodek 	bus_dma_segment_t segs[256];
1871af8fe8f1SZbigniew Bodek 	struct nicvf *nic;
18722306b72aSZbigniew Bodek 	struct snd_buff *snd_buff;
18732306b72aSZbigniew Bodek 	size_t seg;
18742306b72aSZbigniew Bodek 	int nsegs, qentry;
1875af8fe8f1SZbigniew Bodek 	int subdesc_cnt;
18762306b72aSZbigniew Bodek 	int err;
18773c0086b8SZbigniew Bodek 
18782306b72aSZbigniew Bodek 	NICVF_TX_LOCK_ASSERT(sq);
18792306b72aSZbigniew Bodek 
18802306b72aSZbigniew Bodek 	if (sq->free_cnt == 0)
18812306b72aSZbigniew Bodek 		return (ENOBUFS);
18822306b72aSZbigniew Bodek 
18832306b72aSZbigniew Bodek 	snd_buff = &sq->snd_buff[sq->tail];
18842306b72aSZbigniew Bodek 
18852306b72aSZbigniew Bodek 	err = bus_dmamap_load_mbuf_sg(sq->snd_buff_dmat, snd_buff->dmap,
18862306b72aSZbigniew Bodek 	    mbuf, segs, &nsegs, BUS_DMA_NOWAIT);
18872306b72aSZbigniew Bodek 	if (err != 0) {
18882306b72aSZbigniew Bodek 		/* ARM64TODO: Add mbuf defragmenting if we lack maps */
18892306b72aSZbigniew Bodek 		return (err);
18903c0086b8SZbigniew Bodek 	}
18912306b72aSZbigniew Bodek 
18922306b72aSZbigniew Bodek 	/* Set how many subdescriptors is required */
1893af8fe8f1SZbigniew Bodek 	nic = sq->nic;
1894af8fe8f1SZbigniew Bodek 	if (mbuf->m_pkthdr.tso_segsz != 0 && nic->hw_tso)
1895af8fe8f1SZbigniew Bodek 		subdesc_cnt = MIN_SQ_DESC_PER_PKT_XMIT;
1896af8fe8f1SZbigniew Bodek 	else
1897af8fe8f1SZbigniew Bodek 		subdesc_cnt = MIN_SQ_DESC_PER_PKT_XMIT + nsegs - 1;
18982306b72aSZbigniew Bodek 
18992306b72aSZbigniew Bodek 	if (subdesc_cnt > sq->free_cnt) {
19002306b72aSZbigniew Bodek 		/* ARM64TODO: Add mbuf defragmentation if we lack descriptors */
19012306b72aSZbigniew Bodek 		bus_dmamap_unload(sq->snd_buff_dmat, snd_buff->dmap);
19022306b72aSZbigniew Bodek 		return (ENOBUFS);
19033c0086b8SZbigniew Bodek 	}
19043c0086b8SZbigniew Bodek 
19053c0086b8SZbigniew Bodek 	qentry = nicvf_get_sq_desc(sq, subdesc_cnt);
19063c0086b8SZbigniew Bodek 
19073c0086b8SZbigniew Bodek 	/* Add SQ header subdesc */
1908856dce91SZbigniew Bodek 	err = nicvf_sq_add_hdr_subdesc(sq, qentry, subdesc_cnt - 1, mbuf,
19092306b72aSZbigniew Bodek 	    mbuf->m_pkthdr.len);
1910856dce91SZbigniew Bodek 	if (err != 0) {
1911856dce91SZbigniew Bodek 		bus_dmamap_unload(sq->snd_buff_dmat, snd_buff->dmap);
1912856dce91SZbigniew Bodek 		return (err);
1913856dce91SZbigniew Bodek 	}
19143c0086b8SZbigniew Bodek 
19153c0086b8SZbigniew Bodek 	/* Add SQ gather subdescs */
19162306b72aSZbigniew Bodek 	for (seg = 0; seg < nsegs; seg++) {
19173c0086b8SZbigniew Bodek 		qentry = nicvf_get_nxt_sqentry(sq, qentry);
19182306b72aSZbigniew Bodek 		nicvf_sq_add_gather_subdesc(sq, qentry, segs[seg].ds_len,
19192306b72aSZbigniew Bodek 		    segs[seg].ds_addr);
19203c0086b8SZbigniew Bodek 	}
19213c0086b8SZbigniew Bodek 
19223c0086b8SZbigniew Bodek 	/* make sure all memory stores are done before ringing doorbell */
19232306b72aSZbigniew Bodek 	bus_dmamap_sync(sq->dmem.dmat, sq->dmem.dmap, BUS_DMASYNC_PREWRITE);
19243c0086b8SZbigniew Bodek 
19252306b72aSZbigniew Bodek 	dprintf(sq->nic->dev, "%s: sq->idx: %d, subdesc_cnt: %d\n",
19262306b72aSZbigniew Bodek 	    __func__, sq->idx, subdesc_cnt);
19273c0086b8SZbigniew Bodek 	/* Inform HW to xmit new packet */
19282306b72aSZbigniew Bodek 	nicvf_queue_reg_write(sq->nic, NIC_QSET_SQ_0_7_DOOR,
19292306b72aSZbigniew Bodek 	    sq->idx, subdesc_cnt);
19302306b72aSZbigniew Bodek 	return (0);
19313c0086b8SZbigniew Bodek }
19323c0086b8SZbigniew Bodek 
19332306b72aSZbigniew Bodek static __inline u_int
19342306b72aSZbigniew Bodek frag_num(u_int i)
19353c0086b8SZbigniew Bodek {
19362306b72aSZbigniew Bodek #if BYTE_ORDER == BIG_ENDIAN
19372306b72aSZbigniew Bodek 	return ((i & ~3) + 3 - (i & 3));
19383c0086b8SZbigniew Bodek #else
19392306b72aSZbigniew Bodek 	return (i);
19403c0086b8SZbigniew Bodek #endif
19413c0086b8SZbigniew Bodek }
19423c0086b8SZbigniew Bodek 
19432306b72aSZbigniew Bodek /* Returns MBUF for a received packet */
19442306b72aSZbigniew Bodek struct mbuf *
19452306b72aSZbigniew Bodek nicvf_get_rcv_mbuf(struct nicvf *nic, struct cqe_rx_t *cqe_rx)
19463c0086b8SZbigniew Bodek {
19473c0086b8SZbigniew Bodek 	int frag;
19483c0086b8SZbigniew Bodek 	int payload_len = 0;
19492306b72aSZbigniew Bodek 	struct mbuf *mbuf;
19502306b72aSZbigniew Bodek 	struct mbuf *mbuf_frag;
19512306b72aSZbigniew Bodek 	uint16_t *rb_lens = NULL;
19522306b72aSZbigniew Bodek 	uint64_t *rb_ptrs = NULL;
19533c0086b8SZbigniew Bodek 
19542306b72aSZbigniew Bodek 	mbuf = NULL;
19552306b72aSZbigniew Bodek 	rb_lens = (uint16_t *)((uint8_t *)cqe_rx + (3 * sizeof(uint64_t)));
19562306b72aSZbigniew Bodek 	rb_ptrs = (uint64_t *)((uint8_t *)cqe_rx + (6 * sizeof(uint64_t)));
19573c0086b8SZbigniew Bodek 
19582306b72aSZbigniew Bodek 	dprintf(nic->dev, "%s rb_cnt %d rb0_ptr %lx rb0_sz %d\n",
19593c0086b8SZbigniew Bodek 	    __func__, cqe_rx->rb_cnt, cqe_rx->rb0_ptr, cqe_rx->rb0_sz);
19603c0086b8SZbigniew Bodek 
19613c0086b8SZbigniew Bodek 	for (frag = 0; frag < cqe_rx->rb_cnt; frag++) {
19623c0086b8SZbigniew Bodek 		payload_len = rb_lens[frag_num(frag)];
19632306b72aSZbigniew Bodek 		if (frag == 0) {
19643c0086b8SZbigniew Bodek 			/* First fragment */
19652306b72aSZbigniew Bodek 			mbuf = nicvf_rb_ptr_to_mbuf(nic,
19662306b72aSZbigniew Bodek 			    (*rb_ptrs - cqe_rx->align_pad));
19672306b72aSZbigniew Bodek 			mbuf->m_len = payload_len;
19682306b72aSZbigniew Bodek 			mbuf->m_data += cqe_rx->align_pad;
19692306b72aSZbigniew Bodek 			if_setrcvif(mbuf, nic->ifp);
19703c0086b8SZbigniew Bodek 		} else {
19713c0086b8SZbigniew Bodek 			/* Add fragments */
19722306b72aSZbigniew Bodek 			mbuf_frag = nicvf_rb_ptr_to_mbuf(nic, *rb_ptrs);
19732306b72aSZbigniew Bodek 			m_append(mbuf, payload_len, mbuf_frag->m_data);
19742306b72aSZbigniew Bodek 			m_freem(mbuf_frag);
19753c0086b8SZbigniew Bodek 		}
19763c0086b8SZbigniew Bodek 		/* Next buffer pointer */
19773c0086b8SZbigniew Bodek 		rb_ptrs++;
19783c0086b8SZbigniew Bodek 	}
19792306b72aSZbigniew Bodek 
19802306b72aSZbigniew Bodek 	if (__predict_true(mbuf != NULL)) {
19812306b72aSZbigniew Bodek 		m_fixhdr(mbuf);
19822306b72aSZbigniew Bodek 		mbuf->m_pkthdr.flowid = cqe_rx->rq_idx;
19832306b72aSZbigniew Bodek 		M_HASHTYPE_SET(mbuf, M_HASHTYPE_OPAQUE);
1984856dce91SZbigniew Bodek 		if (__predict_true((if_getcapenable(nic->ifp) & IFCAP_RXCSUM) != 0)) {
1985856dce91SZbigniew Bodek 			/*
1986856dce91SZbigniew Bodek 			 * HW by default verifies IP & TCP/UDP/SCTP checksums
1987856dce91SZbigniew Bodek 			 */
1988856dce91SZbigniew Bodek 
1989856dce91SZbigniew Bodek 			/* XXX: Do we need to include IP with options too? */
1990856dce91SZbigniew Bodek 			if (__predict_true(cqe_rx->l3_type == L3TYPE_IPV4 ||
1991856dce91SZbigniew Bodek 			    cqe_rx->l3_type == L3TYPE_IPV6)) {
1992856dce91SZbigniew Bodek 				mbuf->m_pkthdr.csum_flags =
1993856dce91SZbigniew Bodek 				    (CSUM_IP_CHECKED | CSUM_IP_VALID);
1994856dce91SZbigniew Bodek 			}
1995856dce91SZbigniew Bodek 			if (cqe_rx->l4_type == L4TYPE_TCP ||
1996856dce91SZbigniew Bodek 			    cqe_rx->l4_type == L4TYPE_UDP ||
1997856dce91SZbigniew Bodek 			    cqe_rx->l4_type == L4TYPE_SCTP) {
1998856dce91SZbigniew Bodek 				mbuf->m_pkthdr.csum_flags |=
1999856dce91SZbigniew Bodek 				    (CSUM_DATA_VALID | CSUM_PSEUDO_HDR);
2000856dce91SZbigniew Bodek 				mbuf->m_pkthdr.csum_data = htons(0xffff);
2001856dce91SZbigniew Bodek 			}
2002856dce91SZbigniew Bodek 		}
20032306b72aSZbigniew Bodek 	}
20042306b72aSZbigniew Bodek 
20052306b72aSZbigniew Bodek 	return (mbuf);
20063c0086b8SZbigniew Bodek }
20073c0086b8SZbigniew Bodek 
20083c0086b8SZbigniew Bodek /* Enable interrupt */
20092306b72aSZbigniew Bodek void
20102306b72aSZbigniew Bodek nicvf_enable_intr(struct nicvf *nic, int int_type, int q_idx)
20113c0086b8SZbigniew Bodek {
20122306b72aSZbigniew Bodek 	uint64_t reg_val;
20133c0086b8SZbigniew Bodek 
20143c0086b8SZbigniew Bodek 	reg_val = nicvf_reg_read(nic, NIC_VF_ENA_W1S);
20153c0086b8SZbigniew Bodek 
20163c0086b8SZbigniew Bodek 	switch (int_type) {
20173c0086b8SZbigniew Bodek 	case NICVF_INTR_CQ:
20182306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_CQ_SHIFT);
20193c0086b8SZbigniew Bodek 		break;
20203c0086b8SZbigniew Bodek 	case NICVF_INTR_SQ:
20212306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_SQ_SHIFT);
20223c0086b8SZbigniew Bodek 		break;
20233c0086b8SZbigniew Bodek 	case NICVF_INTR_RBDR:
20242306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_RBDR_SHIFT);
20253c0086b8SZbigniew Bodek 		break;
20263c0086b8SZbigniew Bodek 	case NICVF_INTR_PKT_DROP:
20272306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_PKT_DROP_SHIFT);
20283c0086b8SZbigniew Bodek 		break;
20293c0086b8SZbigniew Bodek 	case NICVF_INTR_TCP_TIMER:
20302306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_TCP_TIMER_SHIFT);
20313c0086b8SZbigniew Bodek 		break;
20323c0086b8SZbigniew Bodek 	case NICVF_INTR_MBOX:
20332306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_MBOX_SHIFT);
20343c0086b8SZbigniew Bodek 		break;
20353c0086b8SZbigniew Bodek 	case NICVF_INTR_QS_ERR:
20362306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_QS_ERR_SHIFT);
20373c0086b8SZbigniew Bodek 		break;
20383c0086b8SZbigniew Bodek 	default:
20392306b72aSZbigniew Bodek 		device_printf(nic->dev,
20403c0086b8SZbigniew Bodek 			   "Failed to enable interrupt: unknown type\n");
20413c0086b8SZbigniew Bodek 		break;
20423c0086b8SZbigniew Bodek 	}
20433c0086b8SZbigniew Bodek 
20443c0086b8SZbigniew Bodek 	nicvf_reg_write(nic, NIC_VF_ENA_W1S, reg_val);
20453c0086b8SZbigniew Bodek }
20463c0086b8SZbigniew Bodek 
20473c0086b8SZbigniew Bodek /* Disable interrupt */
20482306b72aSZbigniew Bodek void
20492306b72aSZbigniew Bodek nicvf_disable_intr(struct nicvf *nic, int int_type, int q_idx)
20503c0086b8SZbigniew Bodek {
20512306b72aSZbigniew Bodek 	uint64_t reg_val = 0;
20523c0086b8SZbigniew Bodek 
20533c0086b8SZbigniew Bodek 	switch (int_type) {
20543c0086b8SZbigniew Bodek 	case NICVF_INTR_CQ:
20552306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_CQ_SHIFT);
20563c0086b8SZbigniew Bodek 		break;
20573c0086b8SZbigniew Bodek 	case NICVF_INTR_SQ:
20582306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_SQ_SHIFT);
20593c0086b8SZbigniew Bodek 		break;
20603c0086b8SZbigniew Bodek 	case NICVF_INTR_RBDR:
20612306b72aSZbigniew Bodek 		reg_val |= ((1UL << q_idx) << NICVF_INTR_RBDR_SHIFT);
20623c0086b8SZbigniew Bodek 		break;
20633c0086b8SZbigniew Bodek 	case NICVF_INTR_PKT_DROP:
20642306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_PKT_DROP_SHIFT);
20653c0086b8SZbigniew Bodek 		break;
20663c0086b8SZbigniew Bodek 	case NICVF_INTR_TCP_TIMER:
20672306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_TCP_TIMER_SHIFT);
20683c0086b8SZbigniew Bodek 		break;
20693c0086b8SZbigniew Bodek 	case NICVF_INTR_MBOX:
20702306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_MBOX_SHIFT);
20713c0086b8SZbigniew Bodek 		break;
20723c0086b8SZbigniew Bodek 	case NICVF_INTR_QS_ERR:
20732306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_QS_ERR_SHIFT);
20743c0086b8SZbigniew Bodek 		break;
20753c0086b8SZbigniew Bodek 	default:
20762306b72aSZbigniew Bodek 		device_printf(nic->dev,
20773c0086b8SZbigniew Bodek 			   "Failed to disable interrupt: unknown type\n");
20783c0086b8SZbigniew Bodek 		break;
20793c0086b8SZbigniew Bodek 	}
20803c0086b8SZbigniew Bodek 
20813c0086b8SZbigniew Bodek 	nicvf_reg_write(nic, NIC_VF_ENA_W1C, reg_val);
20823c0086b8SZbigniew Bodek }
20833c0086b8SZbigniew Bodek 
20843c0086b8SZbigniew Bodek /* Clear interrupt */
20852306b72aSZbigniew Bodek void
20862306b72aSZbigniew Bodek nicvf_clear_intr(struct nicvf *nic, int int_type, int q_idx)
20873c0086b8SZbigniew Bodek {
20882306b72aSZbigniew Bodek 	uint64_t reg_val = 0;
20893c0086b8SZbigniew Bodek 
20903c0086b8SZbigniew Bodek 	switch (int_type) {
20913c0086b8SZbigniew Bodek 	case NICVF_INTR_CQ:
20922306b72aSZbigniew Bodek 		reg_val = ((1UL << q_idx) << NICVF_INTR_CQ_SHIFT);
20933c0086b8SZbigniew Bodek 		break;
20943c0086b8SZbigniew Bodek 	case NICVF_INTR_SQ:
20952306b72aSZbigniew Bodek 		reg_val = ((1UL << q_idx) << NICVF_INTR_SQ_SHIFT);
20963c0086b8SZbigniew Bodek 		break;
20973c0086b8SZbigniew Bodek 	case NICVF_INTR_RBDR:
20982306b72aSZbigniew Bodek 		reg_val = ((1UL << q_idx) << NICVF_INTR_RBDR_SHIFT);
20993c0086b8SZbigniew Bodek 		break;
21003c0086b8SZbigniew Bodek 	case NICVF_INTR_PKT_DROP:
21012306b72aSZbigniew Bodek 		reg_val = (1UL << NICVF_INTR_PKT_DROP_SHIFT);
21023c0086b8SZbigniew Bodek 		break;
21033c0086b8SZbigniew Bodek 	case NICVF_INTR_TCP_TIMER:
21042306b72aSZbigniew Bodek 		reg_val = (1UL << NICVF_INTR_TCP_TIMER_SHIFT);
21053c0086b8SZbigniew Bodek 		break;
21063c0086b8SZbigniew Bodek 	case NICVF_INTR_MBOX:
21072306b72aSZbigniew Bodek 		reg_val = (1UL << NICVF_INTR_MBOX_SHIFT);
21083c0086b8SZbigniew Bodek 		break;
21093c0086b8SZbigniew Bodek 	case NICVF_INTR_QS_ERR:
21102306b72aSZbigniew Bodek 		reg_val |= (1UL << NICVF_INTR_QS_ERR_SHIFT);
21113c0086b8SZbigniew Bodek 		break;
21123c0086b8SZbigniew Bodek 	default:
21132306b72aSZbigniew Bodek 		device_printf(nic->dev,
21143c0086b8SZbigniew Bodek 			   "Failed to clear interrupt: unknown type\n");
21153c0086b8SZbigniew Bodek 		break;
21163c0086b8SZbigniew Bodek 	}
21173c0086b8SZbigniew Bodek 
21183c0086b8SZbigniew Bodek 	nicvf_reg_write(nic, NIC_VF_INT, reg_val);
21193c0086b8SZbigniew Bodek }
21203c0086b8SZbigniew Bodek 
21213c0086b8SZbigniew Bodek /* Check if interrupt is enabled */
21222306b72aSZbigniew Bodek int
21232306b72aSZbigniew Bodek nicvf_is_intr_enabled(struct nicvf *nic, int int_type, int q_idx)
21243c0086b8SZbigniew Bodek {
21252306b72aSZbigniew Bodek 	uint64_t reg_val;
21262306b72aSZbigniew Bodek 	uint64_t mask = 0xff;
21273c0086b8SZbigniew Bodek 
21283c0086b8SZbigniew Bodek 	reg_val = nicvf_reg_read(nic, NIC_VF_ENA_W1S);
21293c0086b8SZbigniew Bodek 
21303c0086b8SZbigniew Bodek 	switch (int_type) {
21313c0086b8SZbigniew Bodek 	case NICVF_INTR_CQ:
21322306b72aSZbigniew Bodek 		mask = ((1UL << q_idx) << NICVF_INTR_CQ_SHIFT);
21333c0086b8SZbigniew Bodek 		break;
21343c0086b8SZbigniew Bodek 	case NICVF_INTR_SQ:
21352306b72aSZbigniew Bodek 		mask = ((1UL << q_idx) << NICVF_INTR_SQ_SHIFT);
21363c0086b8SZbigniew Bodek 		break;
21373c0086b8SZbigniew Bodek 	case NICVF_INTR_RBDR:
21382306b72aSZbigniew Bodek 		mask = ((1UL << q_idx) << NICVF_INTR_RBDR_SHIFT);
21393c0086b8SZbigniew Bodek 		break;
21403c0086b8SZbigniew Bodek 	case NICVF_INTR_PKT_DROP:
21413c0086b8SZbigniew Bodek 		mask = NICVF_INTR_PKT_DROP_MASK;
21423c0086b8SZbigniew Bodek 		break;
21433c0086b8SZbigniew Bodek 	case NICVF_INTR_TCP_TIMER:
21443c0086b8SZbigniew Bodek 		mask = NICVF_INTR_TCP_TIMER_MASK;
21453c0086b8SZbigniew Bodek 		break;
21463c0086b8SZbigniew Bodek 	case NICVF_INTR_MBOX:
21473c0086b8SZbigniew Bodek 		mask = NICVF_INTR_MBOX_MASK;
21483c0086b8SZbigniew Bodek 		break;
21493c0086b8SZbigniew Bodek 	case NICVF_INTR_QS_ERR:
21503c0086b8SZbigniew Bodek 		mask = NICVF_INTR_QS_ERR_MASK;
21513c0086b8SZbigniew Bodek 		break;
21523c0086b8SZbigniew Bodek 	default:
21532306b72aSZbigniew Bodek 		device_printf(nic->dev,
21543c0086b8SZbigniew Bodek 			   "Failed to check interrupt enable: unknown type\n");
21553c0086b8SZbigniew Bodek 		break;
21563c0086b8SZbigniew Bodek 	}
21573c0086b8SZbigniew Bodek 
21583c0086b8SZbigniew Bodek 	return (reg_val & mask);
21593c0086b8SZbigniew Bodek }
21603c0086b8SZbigniew Bodek 
21612306b72aSZbigniew Bodek void
21622306b72aSZbigniew Bodek nicvf_update_rq_stats(struct nicvf *nic, int rq_idx)
21633c0086b8SZbigniew Bodek {
21643c0086b8SZbigniew Bodek 	struct rcv_queue *rq;
21653c0086b8SZbigniew Bodek 
21663c0086b8SZbigniew Bodek #define GET_RQ_STATS(reg) \
21673c0086b8SZbigniew Bodek 	nicvf_reg_read(nic, NIC_QSET_RQ_0_7_STAT_0_1 |\
21683c0086b8SZbigniew Bodek 			    (rq_idx << NIC_Q_NUM_SHIFT) | (reg << 3))
21693c0086b8SZbigniew Bodek 
21703c0086b8SZbigniew Bodek 	rq = &nic->qs->rq[rq_idx];
21713c0086b8SZbigniew Bodek 	rq->stats.bytes = GET_RQ_STATS(RQ_SQ_STATS_OCTS);
21723c0086b8SZbigniew Bodek 	rq->stats.pkts = GET_RQ_STATS(RQ_SQ_STATS_PKTS);
21733c0086b8SZbigniew Bodek }
21743c0086b8SZbigniew Bodek 
21752306b72aSZbigniew Bodek void
21762306b72aSZbigniew Bodek nicvf_update_sq_stats(struct nicvf *nic, int sq_idx)
21773c0086b8SZbigniew Bodek {
21783c0086b8SZbigniew Bodek 	struct snd_queue *sq;
21793c0086b8SZbigniew Bodek 
21803c0086b8SZbigniew Bodek #define GET_SQ_STATS(reg) \
21813c0086b8SZbigniew Bodek 	nicvf_reg_read(nic, NIC_QSET_SQ_0_7_STAT_0_1 |\
21823c0086b8SZbigniew Bodek 			    (sq_idx << NIC_Q_NUM_SHIFT) | (reg << 3))
21833c0086b8SZbigniew Bodek 
21843c0086b8SZbigniew Bodek 	sq = &nic->qs->sq[sq_idx];
21853c0086b8SZbigniew Bodek 	sq->stats.bytes = GET_SQ_STATS(RQ_SQ_STATS_OCTS);
21863c0086b8SZbigniew Bodek 	sq->stats.pkts = GET_SQ_STATS(RQ_SQ_STATS_PKTS);
21873c0086b8SZbigniew Bodek }
21883c0086b8SZbigniew Bodek 
21893c0086b8SZbigniew Bodek /* Check for errors in the receive cmp.queue entry */
21902306b72aSZbigniew Bodek int
21912306b72aSZbigniew Bodek nicvf_check_cqe_rx_errs(struct nicvf *nic, struct cmp_queue *cq,
21922306b72aSZbigniew Bodek     struct cqe_rx_t *cqe_rx)
21933c0086b8SZbigniew Bodek {
21943c0086b8SZbigniew Bodek 	struct nicvf_hw_stats *stats = &nic->hw_stats;
21953c0086b8SZbigniew Bodek 	struct nicvf_drv_stats *drv_stats = &nic->drv_stats;
21963c0086b8SZbigniew Bodek 
21973c0086b8SZbigniew Bodek 	if (!cqe_rx->err_level && !cqe_rx->err_opcode) {
21983c0086b8SZbigniew Bodek 		drv_stats->rx_frames_ok++;
21992306b72aSZbigniew Bodek 		return (0);
22003c0086b8SZbigniew Bodek 	}
22013c0086b8SZbigniew Bodek 
22023c0086b8SZbigniew Bodek 	switch (cqe_rx->err_opcode) {
22033c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_RE_PARTIAL:
22043c0086b8SZbigniew Bodek 		stats->rx_bgx_truncated_pkts++;
22053c0086b8SZbigniew Bodek 		break;
22063c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_RE_JABBER:
22073c0086b8SZbigniew Bodek 		stats->rx_jabber_errs++;
22083c0086b8SZbigniew Bodek 		break;
22093c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_RE_FCS:
22103c0086b8SZbigniew Bodek 		stats->rx_fcs_errs++;
22113c0086b8SZbigniew Bodek 		break;
22123c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_RE_RX_CTL:
22133c0086b8SZbigniew Bodek 		stats->rx_bgx_errs++;
22143c0086b8SZbigniew Bodek 		break;
22153c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_PREL2_ERR:
22163c0086b8SZbigniew Bodek 		stats->rx_prel2_errs++;
22173c0086b8SZbigniew Bodek 		break;
22183c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L2_MAL:
22193c0086b8SZbigniew Bodek 		stats->rx_l2_hdr_malformed++;
22203c0086b8SZbigniew Bodek 		break;
22213c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L2_OVERSIZE:
22223c0086b8SZbigniew Bodek 		stats->rx_oversize++;
22233c0086b8SZbigniew Bodek 		break;
22243c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L2_UNDERSIZE:
22253c0086b8SZbigniew Bodek 		stats->rx_undersize++;
22263c0086b8SZbigniew Bodek 		break;
22273c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L2_LENMISM:
22283c0086b8SZbigniew Bodek 		stats->rx_l2_len_mismatch++;
22293c0086b8SZbigniew Bodek 		break;
22303c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L2_PCLP:
22313c0086b8SZbigniew Bodek 		stats->rx_l2_pclp++;
22323c0086b8SZbigniew Bodek 		break;
22333c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_IP_NOT:
22343c0086b8SZbigniew Bodek 		stats->rx_ip_ver_errs++;
22353c0086b8SZbigniew Bodek 		break;
22363c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_IP_CSUM_ERR:
22373c0086b8SZbigniew Bodek 		stats->rx_ip_csum_errs++;
22383c0086b8SZbigniew Bodek 		break;
22393c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_IP_MAL:
22403c0086b8SZbigniew Bodek 		stats->rx_ip_hdr_malformed++;
22413c0086b8SZbigniew Bodek 		break;
22423c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_IP_MALD:
22433c0086b8SZbigniew Bodek 		stats->rx_ip_payload_malformed++;
22443c0086b8SZbigniew Bodek 		break;
22453c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_IP_HOP:
22463c0086b8SZbigniew Bodek 		stats->rx_ip_ttl_errs++;
22473c0086b8SZbigniew Bodek 		break;
22483c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L3_PCLP:
22493c0086b8SZbigniew Bodek 		stats->rx_l3_pclp++;
22503c0086b8SZbigniew Bodek 		break;
22513c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L4_MAL:
22523c0086b8SZbigniew Bodek 		stats->rx_l4_malformed++;
22533c0086b8SZbigniew Bodek 		break;
22543c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L4_CHK:
22553c0086b8SZbigniew Bodek 		stats->rx_l4_csum_errs++;
22563c0086b8SZbigniew Bodek 		break;
22573c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_UDP_LEN:
22583c0086b8SZbigniew Bodek 		stats->rx_udp_len_errs++;
22593c0086b8SZbigniew Bodek 		break;
22603c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L4_PORT:
22613c0086b8SZbigniew Bodek 		stats->rx_l4_port_errs++;
22623c0086b8SZbigniew Bodek 		break;
22633c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_TCP_FLAG:
22643c0086b8SZbigniew Bodek 		stats->rx_tcp_flag_errs++;
22653c0086b8SZbigniew Bodek 		break;
22663c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_TCP_OFFSET:
22673c0086b8SZbigniew Bodek 		stats->rx_tcp_offset_errs++;
22683c0086b8SZbigniew Bodek 		break;
22693c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_L4_PCLP:
22703c0086b8SZbigniew Bodek 		stats->rx_l4_pclp++;
22713c0086b8SZbigniew Bodek 		break;
22723c0086b8SZbigniew Bodek 	case CQ_RX_ERROP_RBDR_TRUNC:
22733c0086b8SZbigniew Bodek 		stats->rx_truncated_pkts++;
22743c0086b8SZbigniew Bodek 		break;
22753c0086b8SZbigniew Bodek 	}
22763c0086b8SZbigniew Bodek 
22772306b72aSZbigniew Bodek 	return (1);
22783c0086b8SZbigniew Bodek }
22793c0086b8SZbigniew Bodek 
22803c0086b8SZbigniew Bodek /* Check for errors in the send cmp.queue entry */
22812306b72aSZbigniew Bodek int
22822306b72aSZbigniew Bodek nicvf_check_cqe_tx_errs(struct nicvf *nic, struct cmp_queue *cq,
22832306b72aSZbigniew Bodek     struct cqe_send_t *cqe_tx)
22843c0086b8SZbigniew Bodek {
22853c0086b8SZbigniew Bodek 	struct cmp_queue_stats *stats = &cq->stats;
22863c0086b8SZbigniew Bodek 
22873c0086b8SZbigniew Bodek 	switch (cqe_tx->send_status) {
22883c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_GOOD:
22893c0086b8SZbigniew Bodek 		stats->tx.good++;
22902306b72aSZbigniew Bodek 		return (0);
22913c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_DESC_FAULT:
22923c0086b8SZbigniew Bodek 		stats->tx.desc_fault++;
22933c0086b8SZbigniew Bodek 		break;
22943c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_HDR_CONS_ERR:
22953c0086b8SZbigniew Bodek 		stats->tx.hdr_cons_err++;
22963c0086b8SZbigniew Bodek 		break;
22973c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_SUBDC_ERR:
22983c0086b8SZbigniew Bodek 		stats->tx.subdesc_err++;
22993c0086b8SZbigniew Bodek 		break;
23003c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_IMM_SIZE_OFLOW:
23013c0086b8SZbigniew Bodek 		stats->tx.imm_size_oflow++;
23023c0086b8SZbigniew Bodek 		break;
23033c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_DATA_SEQUENCE_ERR:
23043c0086b8SZbigniew Bodek 		stats->tx.data_seq_err++;
23053c0086b8SZbigniew Bodek 		break;
23063c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_MEM_SEQUENCE_ERR:
23073c0086b8SZbigniew Bodek 		stats->tx.mem_seq_err++;
23083c0086b8SZbigniew Bodek 		break;
23093c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_LOCK_VIOL:
23103c0086b8SZbigniew Bodek 		stats->tx.lock_viol++;
23113c0086b8SZbigniew Bodek 		break;
23123c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_DATA_FAULT:
23133c0086b8SZbigniew Bodek 		stats->tx.data_fault++;
23143c0086b8SZbigniew Bodek 		break;
23153c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_TSTMP_CONFLICT:
23163c0086b8SZbigniew Bodek 		stats->tx.tstmp_conflict++;
23173c0086b8SZbigniew Bodek 		break;
23183c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_TSTMP_TIMEOUT:
23193c0086b8SZbigniew Bodek 		stats->tx.tstmp_timeout++;
23203c0086b8SZbigniew Bodek 		break;
23213c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_MEM_FAULT:
23223c0086b8SZbigniew Bodek 		stats->tx.mem_fault++;
23233c0086b8SZbigniew Bodek 		break;
23243c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_CK_OVERLAP:
23253c0086b8SZbigniew Bodek 		stats->tx.csum_overlap++;
23263c0086b8SZbigniew Bodek 		break;
23273c0086b8SZbigniew Bodek 	case CQ_TX_ERROP_CK_OFLOW:
23283c0086b8SZbigniew Bodek 		stats->tx.csum_overflow++;
23293c0086b8SZbigniew Bodek 		break;
23303c0086b8SZbigniew Bodek 	}
23313c0086b8SZbigniew Bodek 
23322306b72aSZbigniew Bodek 	return (1);
23333c0086b8SZbigniew Bodek }
2334