04e8ec7419
There were defines which originally allowed sharing of some code with the enic kernel driver. The code has long since diverged and now the abstraction just makes the code harder to read. Mostly mechanical replacement of defines and reformatting. Signed-off-by: John Daley <johndale@cisco.com> Reviewed-by: Hyong Youb Kim <hyonkim@cisco.com>
149 lines
3.6 KiB
C
149 lines
3.6 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright 2008-2017 Cisco Systems, Inc. All rights reserved.
|
|
* Copyright 2007 Nuova Systems, Inc. All rights reserved.
|
|
*/
|
|
|
|
#include <rte_memzone.h>
|
|
#include "vnic_dev.h"
|
|
#include "vnic_rq.h"
|
|
|
|
void vnic_rq_free(struct vnic_rq *rq)
|
|
{
|
|
struct vnic_dev *vdev;
|
|
|
|
vdev = rq->vdev;
|
|
|
|
vnic_dev_free_desc_ring(vdev, &rq->ring);
|
|
|
|
rq->ctrl = NULL;
|
|
}
|
|
|
|
int vnic_rq_alloc(struct vnic_dev *vdev, struct vnic_rq *rq, unsigned int index,
|
|
unsigned int desc_count, unsigned int desc_size)
|
|
{
|
|
int rc;
|
|
char res_name[RTE_MEMZONE_NAMESIZE];
|
|
static int instance;
|
|
|
|
rq->index = index;
|
|
rq->vdev = vdev;
|
|
|
|
rq->ctrl = vnic_dev_get_res(vdev, RES_TYPE_RQ, index);
|
|
if (!rq->ctrl) {
|
|
pr_err("Failed to hook RQ[%u] resource\n", index);
|
|
return -EINVAL;
|
|
}
|
|
|
|
vnic_rq_disable(rq);
|
|
|
|
snprintf(res_name, sizeof(res_name), "%d-rq-%u", instance++, index);
|
|
rc = vnic_dev_alloc_desc_ring(vdev, &rq->ring, desc_count, desc_size,
|
|
rq->socket_id, res_name);
|
|
return rc;
|
|
}
|
|
|
|
void vnic_rq_init_start(struct vnic_rq *rq, unsigned int cq_index,
|
|
unsigned int fetch_index, unsigned int posted_index,
|
|
unsigned int error_interrupt_enable,
|
|
unsigned int error_interrupt_offset)
|
|
{
|
|
uint64_t paddr;
|
|
unsigned int count = rq->ring.desc_count;
|
|
|
|
paddr = (uint64_t)rq->ring.base_addr | VNIC_PADDR_TARGET;
|
|
writeq(paddr, &rq->ctrl->ring_base);
|
|
iowrite32(count, &rq->ctrl->ring_size);
|
|
iowrite32(cq_index, &rq->ctrl->cq_index);
|
|
iowrite32(error_interrupt_enable, &rq->ctrl->error_interrupt_enable);
|
|
iowrite32(error_interrupt_offset, &rq->ctrl->error_interrupt_offset);
|
|
iowrite32(0, &rq->ctrl->error_status);
|
|
iowrite32(fetch_index, &rq->ctrl->fetch_index);
|
|
iowrite32(posted_index, &rq->ctrl->posted_index);
|
|
if (rq->data_queue_enable)
|
|
iowrite32(((1 << 10) | rq->data_queue_idx),
|
|
&rq->ctrl->data_ring);
|
|
else
|
|
iowrite32(0, &rq->ctrl->data_ring);
|
|
}
|
|
|
|
void vnic_rq_init(struct vnic_rq *rq, unsigned int cq_index,
|
|
unsigned int error_interrupt_enable,
|
|
unsigned int error_interrupt_offset)
|
|
{
|
|
uint32_t fetch_index = 0;
|
|
|
|
/* Use current fetch_index as the ring starting point */
|
|
fetch_index = ioread32(&rq->ctrl->fetch_index);
|
|
|
|
if (fetch_index == 0xFFFFFFFF) { /* check for hardware gone */
|
|
/* Hardware surprise removal: reset fetch_index */
|
|
fetch_index = 0;
|
|
}
|
|
|
|
vnic_rq_init_start(rq, cq_index,
|
|
fetch_index, fetch_index,
|
|
error_interrupt_enable,
|
|
error_interrupt_offset);
|
|
rq->rxst_idx = 0;
|
|
rq->tot_pkts = 0;
|
|
rq->pkt_first_seg = NULL;
|
|
rq->pkt_last_seg = NULL;
|
|
}
|
|
|
|
unsigned int vnic_rq_error_status(struct vnic_rq *rq)
|
|
{
|
|
return ioread32(&rq->ctrl->error_status);
|
|
}
|
|
|
|
void vnic_rq_enable(struct vnic_rq *rq)
|
|
{
|
|
iowrite32(1, &rq->ctrl->enable);
|
|
}
|
|
|
|
int vnic_rq_disable(struct vnic_rq *rq)
|
|
{
|
|
unsigned int wait;
|
|
|
|
iowrite32(0, &rq->ctrl->enable);
|
|
|
|
/* Wait for HW to ACK disable request */
|
|
for (wait = 0; wait < 1000; wait++) {
|
|
if (!(ioread32(&rq->ctrl->running)))
|
|
return 0;
|
|
usleep(10);
|
|
}
|
|
|
|
pr_err("Failed to disable RQ[%d]\n", rq->index);
|
|
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
void vnic_rq_clean(struct vnic_rq *rq,
|
|
void (*buf_clean)(struct rte_mbuf **buf))
|
|
{
|
|
struct rte_mbuf **buf;
|
|
uint32_t fetch_index, i;
|
|
unsigned int count = rq->ring.desc_count;
|
|
|
|
buf = &rq->mbuf_ring[0];
|
|
|
|
for (i = 0; i < count; i++) {
|
|
(*buf_clean)(buf);
|
|
buf++;
|
|
}
|
|
rq->ring.desc_avail = count - 1;
|
|
rq->rx_nb_hold = 0;
|
|
|
|
/* Use current fetch_index as the ring starting point */
|
|
fetch_index = ioread32(&rq->ctrl->fetch_index);
|
|
|
|
if (fetch_index == 0xFFFFFFFF) { /* check for hardware gone */
|
|
/* Hardware surprise removal: reset fetch_index */
|
|
fetch_index = 0;
|
|
}
|
|
|
|
iowrite32(fetch_index, &rq->ctrl->posted_index);
|
|
|
|
vnic_dev_clear_desc_ring(&rq->ring);
|
|
}
|