mirror of https://github.com/F-Stack/f-stack.git
944 lines
20 KiB
C
944 lines
20 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright(c) 2017 Intel Corporation
|
|
*/
|
|
|
|
#include <inttypes.h>
|
|
#include <string.h>
|
|
|
|
#include <bus_vdev_driver.h>
|
|
#include <rte_errno.h>
|
|
#include <rte_cycles.h>
|
|
#include <rte_memzone.h>
|
|
|
|
#include "opdl_evdev.h"
|
|
#include "opdl_ring.h"
|
|
#include "opdl_log.h"
|
|
|
|
|
|
static __rte_always_inline uint32_t
|
|
enqueue_check(struct opdl_port *p,
|
|
const struct rte_event ev[],
|
|
uint16_t num,
|
|
uint16_t num_events)
|
|
{
|
|
uint16_t i;
|
|
|
|
if (p->opdl->do_validation) {
|
|
|
|
for (i = 0; i < num; i++) {
|
|
if (ev[i].queue_id != p->next_external_qid) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"ERROR - port:[%u] - event wants"
|
|
" to enq to q_id[%u],"
|
|
" but should be [%u]",
|
|
opdl_pmd_dev_id(p->opdl),
|
|
p->id,
|
|
ev[i].queue_id,
|
|
p->next_external_qid);
|
|
rte_errno = EINVAL;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* Stats */
|
|
if (p->p_type == OPDL_PURE_RX_PORT ||
|
|
p->p_type == OPDL_ASYNC_PORT) {
|
|
/* Stats */
|
|
if (num_events) {
|
|
p->port_stat[claim_pkts_requested] += num;
|
|
p->port_stat[claim_pkts_granted] += num_events;
|
|
p->port_stat[claim_non_empty]++;
|
|
p->start_cycles = rte_rdtsc();
|
|
} else {
|
|
p->port_stat[claim_empty]++;
|
|
p->start_cycles = 0;
|
|
}
|
|
} else {
|
|
if (p->start_cycles) {
|
|
uint64_t end_cycles = rte_rdtsc();
|
|
p->port_stat[total_cycles] +=
|
|
end_cycles - p->start_cycles;
|
|
}
|
|
}
|
|
} else {
|
|
if (num > 0 &&
|
|
ev[0].queue_id != p->next_external_qid) {
|
|
rte_errno = EINVAL;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return num;
|
|
}
|
|
|
|
static __rte_always_inline void
|
|
update_on_dequeue(struct opdl_port *p,
|
|
struct rte_event ev[],
|
|
uint16_t num,
|
|
uint16_t num_events)
|
|
{
|
|
if (p->opdl->do_validation) {
|
|
int16_t i;
|
|
for (i = 0; i < num; i++)
|
|
ev[i].queue_id =
|
|
p->opdl->queue[p->queue_id].external_qid;
|
|
|
|
/* Stats */
|
|
if (num_events) {
|
|
p->port_stat[claim_pkts_requested] += num;
|
|
p->port_stat[claim_pkts_granted] += num_events;
|
|
p->port_stat[claim_non_empty]++;
|
|
p->start_cycles = rte_rdtsc();
|
|
} else {
|
|
p->port_stat[claim_empty]++;
|
|
p->start_cycles = 0;
|
|
}
|
|
} else {
|
|
if (num > 0)
|
|
ev[0].queue_id =
|
|
p->opdl->queue[p->queue_id].external_qid;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Error RX enqueue:
|
|
*
|
|
*
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_rx_error_enqueue(struct opdl_port *p,
|
|
const struct rte_event ev[],
|
|
uint16_t num)
|
|
{
|
|
RTE_SET_USED(p);
|
|
RTE_SET_USED(ev);
|
|
RTE_SET_USED(num);
|
|
|
|
rte_errno = ENOSPC;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* RX enqueue:
|
|
*
|
|
* This function handles enqueue for a single input stage_inst with
|
|
* threadsafe disabled or enabled. eg 1 thread using a stage_inst or
|
|
* multiple threads sharing a stage_inst
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_rx_enqueue(struct opdl_port *p,
|
|
const struct rte_event ev[],
|
|
uint16_t num)
|
|
{
|
|
uint16_t enqueued = 0;
|
|
|
|
enqueued = opdl_ring_input(opdl_stage_get_opdl_ring(p->enq_stage_inst),
|
|
ev,
|
|
num,
|
|
false);
|
|
if (!enqueue_check(p, ev, num, enqueued))
|
|
return 0;
|
|
|
|
|
|
if (enqueued < num)
|
|
rte_errno = ENOSPC;
|
|
|
|
return enqueued;
|
|
}
|
|
|
|
/*
|
|
* Error TX handler
|
|
*
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_tx_error_dequeue(struct opdl_port *p,
|
|
struct rte_event ev[],
|
|
uint16_t num)
|
|
{
|
|
RTE_SET_USED(p);
|
|
RTE_SET_USED(ev);
|
|
RTE_SET_USED(num);
|
|
|
|
rte_errno = ENOSPC;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* TX single threaded claim
|
|
*
|
|
* This function handles dequeue for a single worker stage_inst with
|
|
* threadsafe disabled. eg 1 thread using an stage_inst
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_tx_dequeue_single_thread(struct opdl_port *p,
|
|
struct rte_event ev[],
|
|
uint16_t num)
|
|
{
|
|
uint16_t returned;
|
|
|
|
struct opdl_ring *ring;
|
|
|
|
ring = opdl_stage_get_opdl_ring(p->deq_stage_inst);
|
|
|
|
returned = opdl_ring_copy_to_burst(ring,
|
|
p->deq_stage_inst,
|
|
ev,
|
|
num,
|
|
false);
|
|
|
|
update_on_dequeue(p, ev, num, returned);
|
|
|
|
return returned;
|
|
}
|
|
|
|
/*
|
|
* TX multi threaded claim
|
|
*
|
|
* This function handles dequeue for multiple worker stage_inst with
|
|
* threadsafe disabled. eg multiple stage_inst each with its own instance
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_tx_dequeue_multi_inst(struct opdl_port *p,
|
|
struct rte_event ev[],
|
|
uint16_t num)
|
|
{
|
|
uint32_t num_events = 0;
|
|
|
|
num_events = opdl_stage_claim(p->deq_stage_inst,
|
|
(void *)ev,
|
|
num,
|
|
NULL,
|
|
false,
|
|
false);
|
|
|
|
update_on_dequeue(p, ev, num, num_events);
|
|
|
|
return opdl_stage_disclaim(p->deq_stage_inst, num_events, false);
|
|
}
|
|
|
|
|
|
/*
|
|
* Worker thread claim
|
|
*
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_claim(struct opdl_port *p, struct rte_event ev[], uint16_t num)
|
|
{
|
|
uint32_t num_events = 0;
|
|
|
|
if (unlikely(num > MAX_OPDL_CONS_Q_DEPTH)) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"Attempt to dequeue num of events larger than port (%d) max",
|
|
opdl_pmd_dev_id(p->opdl),
|
|
p->id);
|
|
rte_errno = EINVAL;
|
|
return 0;
|
|
}
|
|
|
|
|
|
num_events = opdl_stage_claim(p->deq_stage_inst,
|
|
(void *)ev,
|
|
num,
|
|
NULL,
|
|
false,
|
|
p->atomic_claim);
|
|
|
|
|
|
update_on_dequeue(p, ev, num, num_events);
|
|
|
|
return num_events;
|
|
}
|
|
|
|
/*
|
|
* Worker thread disclaim
|
|
*/
|
|
|
|
static uint16_t
|
|
opdl_disclaim(struct opdl_port *p, const struct rte_event ev[], uint16_t num)
|
|
{
|
|
uint16_t enqueued = 0;
|
|
|
|
uint32_t i = 0;
|
|
|
|
for (i = 0; i < num; i++)
|
|
opdl_ring_cas_slot(p->enq_stage_inst, &ev[i],
|
|
i, p->atomic_claim);
|
|
|
|
enqueued = opdl_stage_disclaim(p->enq_stage_inst,
|
|
num,
|
|
false);
|
|
|
|
return enqueue_check(p, ev, num, enqueued);
|
|
}
|
|
|
|
static __rte_always_inline struct opdl_stage *
|
|
stage_for_port(struct opdl_queue *q, unsigned int i)
|
|
{
|
|
if (q->q_pos == OPDL_Q_POS_START || q->q_pos == OPDL_Q_POS_MIDDLE)
|
|
return q->ports[i]->enq_stage_inst;
|
|
else
|
|
return q->ports[i]->deq_stage_inst;
|
|
}
|
|
|
|
static int opdl_add_deps(struct opdl_evdev *device,
|
|
int q_id,
|
|
int deps_q_id)
|
|
{
|
|
unsigned int i, j;
|
|
int status;
|
|
struct opdl_ring *ring;
|
|
struct opdl_queue *queue = &device->queue[q_id];
|
|
struct opdl_queue *queue_deps = &device->queue[deps_q_id];
|
|
struct opdl_stage *dep_stages[OPDL_PORTS_MAX];
|
|
|
|
/* sanity check that all stages are for same opdl ring */
|
|
for (i = 0; i < queue->nb_ports; i++) {
|
|
struct opdl_ring *r =
|
|
opdl_stage_get_opdl_ring(stage_for_port(queue, i));
|
|
for (j = 0; j < queue_deps->nb_ports; j++) {
|
|
struct opdl_ring *rj =
|
|
opdl_stage_get_opdl_ring(
|
|
stage_for_port(queue_deps, j));
|
|
if (r != rj) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"Stages and dependents"
|
|
" are not for same opdl ring",
|
|
opdl_pmd_dev_id(device));
|
|
uint32_t k;
|
|
for (k = 0; k < device->nb_opdls; k++) {
|
|
opdl_ring_dump(device->opdl[k],
|
|
stdout);
|
|
}
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Gather all stages instance in deps */
|
|
for (i = 0; i < queue_deps->nb_ports; i++)
|
|
dep_stages[i] = stage_for_port(queue_deps, i);
|
|
|
|
|
|
/* Add all deps for each port->stage_inst in this queue */
|
|
for (i = 0; i < queue->nb_ports; i++) {
|
|
|
|
ring = opdl_stage_get_opdl_ring(stage_for_port(queue, i));
|
|
|
|
status = opdl_stage_deps_add(ring,
|
|
stage_for_port(queue, i),
|
|
queue->ports[i]->num_instance,
|
|
queue->ports[i]->instance_id,
|
|
dep_stages,
|
|
queue_deps->nb_ports);
|
|
if (status < 0)
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
opdl_add_event_handlers(struct rte_eventdev *dev)
|
|
{
|
|
int err = 0;
|
|
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < device->max_port_nb; i++) {
|
|
|
|
struct opdl_port *port = &device->ports[i];
|
|
|
|
if (port->configured) {
|
|
if (port->p_type == OPDL_PURE_RX_PORT) {
|
|
port->enq = opdl_rx_enqueue;
|
|
port->deq = opdl_tx_error_dequeue;
|
|
|
|
} else if (port->p_type == OPDL_PURE_TX_PORT) {
|
|
|
|
port->enq = opdl_rx_error_enqueue;
|
|
|
|
if (port->num_instance == 1)
|
|
port->deq =
|
|
opdl_tx_dequeue_single_thread;
|
|
else
|
|
port->deq = opdl_tx_dequeue_multi_inst;
|
|
|
|
} else if (port->p_type == OPDL_REGULAR_PORT) {
|
|
|
|
port->enq = opdl_disclaim;
|
|
port->deq = opdl_claim;
|
|
|
|
} else if (port->p_type == OPDL_ASYNC_PORT) {
|
|
|
|
port->enq = opdl_rx_enqueue;
|
|
|
|
/* Always single instance */
|
|
port->deq = opdl_tx_dequeue_single_thread;
|
|
} else {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"port:[%u] has invalid port type - ",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
port->id);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
port->initialized = 1;
|
|
}
|
|
}
|
|
|
|
if (!err)
|
|
fprintf(stdout, "Success - enqueue/dequeue handler(s) added\n");
|
|
return err;
|
|
}
|
|
|
|
int
|
|
build_all_dependencies(struct rte_eventdev *dev)
|
|
{
|
|
|
|
int err = 0;
|
|
unsigned int i;
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
|
|
uint8_t start_qid = 0;
|
|
|
|
for (i = 0; i < RTE_EVENT_MAX_QUEUES_PER_DEV; i++) {
|
|
struct opdl_queue *queue = &device->queue[i];
|
|
if (!queue->initialized)
|
|
break;
|
|
|
|
if (queue->q_pos == OPDL_Q_POS_START) {
|
|
start_qid = i;
|
|
continue;
|
|
}
|
|
|
|
if (queue->q_pos == OPDL_Q_POS_MIDDLE) {
|
|
err = opdl_add_deps(device, i, i-1);
|
|
if (err < 0) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"dependency addition for queue:[%u] - FAILED",
|
|
dev->data->dev_id,
|
|
queue->external_qid);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (queue->q_pos == OPDL_Q_POS_END) {
|
|
/* Add this dependency */
|
|
err = opdl_add_deps(device, i, i-1);
|
|
if (err < 0) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"dependency addition for queue:[%u] - FAILED",
|
|
dev->data->dev_id,
|
|
queue->external_qid);
|
|
break;
|
|
}
|
|
/* Add dependency for rx on tx */
|
|
err = opdl_add_deps(device, start_qid, i);
|
|
if (err < 0) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"dependency addition for queue:[%u] - FAILED",
|
|
dev->data->dev_id,
|
|
queue->external_qid);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!err)
|
|
fprintf(stdout, "Success - dependencies built\n");
|
|
|
|
return err;
|
|
}
|
|
int
|
|
check_queues_linked(struct rte_eventdev *dev)
|
|
{
|
|
|
|
int err = 0;
|
|
unsigned int i;
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
uint32_t nb_iq = 0;
|
|
|
|
for (i = 0; i < RTE_EVENT_MAX_QUEUES_PER_DEV; i++) {
|
|
struct opdl_queue *queue = &device->queue[i];
|
|
|
|
if (!queue->initialized)
|
|
break;
|
|
|
|
if (queue->external_qid == OPDL_INVALID_QID)
|
|
nb_iq++;
|
|
|
|
if (queue->nb_ports == 0) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"queue:[%u] has no associated ports",
|
|
dev->data->dev_id,
|
|
i);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
}
|
|
if (!err) {
|
|
if ((i - nb_iq) != device->max_queue_nb) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"%u queues counted but should be %u",
|
|
dev->data->dev_id,
|
|
i - nb_iq,
|
|
device->max_queue_nb);
|
|
err = -1;
|
|
}
|
|
|
|
}
|
|
return err;
|
|
}
|
|
|
|
void
|
|
destroy_queues_and_rings(struct rte_eventdev *dev)
|
|
{
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
uint32_t i;
|
|
|
|
for (i = 0; i < device->nb_opdls; i++) {
|
|
if (device->opdl[i])
|
|
opdl_ring_free(device->opdl[i]);
|
|
}
|
|
|
|
memset(&device->queue,
|
|
0,
|
|
sizeof(struct opdl_queue)
|
|
* RTE_EVENT_MAX_QUEUES_PER_DEV);
|
|
}
|
|
|
|
#define OPDL_ID(d)(d->nb_opdls - 1)
|
|
|
|
static __rte_always_inline void
|
|
initialise_queue(struct opdl_evdev *device,
|
|
enum queue_pos pos,
|
|
int32_t i)
|
|
{
|
|
struct opdl_queue *queue = &device->queue[device->nb_queues];
|
|
|
|
if (i == -1) {
|
|
queue->q_type = OPDL_Q_TYPE_ORDERED;
|
|
queue->external_qid = OPDL_INVALID_QID;
|
|
} else {
|
|
queue->q_type = device->q_md[i].type;
|
|
queue->external_qid = device->q_md[i].ext_id;
|
|
/* Add ex->in for queues setup */
|
|
device->q_map_ex_to_in[queue->external_qid] = device->nb_queues;
|
|
}
|
|
queue->opdl_id = OPDL_ID(device);
|
|
queue->q_pos = pos;
|
|
queue->nb_ports = 0;
|
|
queue->configured = 1;
|
|
|
|
device->nb_queues++;
|
|
}
|
|
|
|
|
|
static __rte_always_inline int
|
|
create_opdl(struct opdl_evdev *device)
|
|
{
|
|
int err = 0;
|
|
|
|
char name[RTE_MEMZONE_NAMESIZE];
|
|
|
|
snprintf(name, RTE_MEMZONE_NAMESIZE,
|
|
"%s_%u", device->service_name, device->nb_opdls);
|
|
|
|
device->opdl[device->nb_opdls] =
|
|
opdl_ring_create(name,
|
|
device->nb_events_limit,
|
|
sizeof(struct rte_event),
|
|
device->max_port_nb * 2,
|
|
device->socket);
|
|
|
|
if (!device->opdl[device->nb_opdls]) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"opdl ring %u creation - FAILED",
|
|
opdl_pmd_dev_id(device),
|
|
device->nb_opdls);
|
|
err = -EINVAL;
|
|
} else {
|
|
device->nb_opdls++;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
static __rte_always_inline int
|
|
create_link_opdl(struct opdl_evdev *device, uint32_t index)
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
if (device->q_md[index + 1].type !=
|
|
OPDL_Q_TYPE_SINGLE_LINK) {
|
|
|
|
/* async queue with regular
|
|
* queue following it
|
|
*/
|
|
|
|
/* create a new opdl ring */
|
|
err = create_opdl(device);
|
|
if (!err) {
|
|
/* create an initial
|
|
* dummy queue for new opdl
|
|
*/
|
|
initialise_queue(device,
|
|
OPDL_Q_POS_START,
|
|
-1);
|
|
} else {
|
|
err = -EINVAL;
|
|
}
|
|
} else {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"queue %u, two consecutive"
|
|
" SINGLE_LINK queues, not allowed",
|
|
opdl_pmd_dev_id(device),
|
|
index);
|
|
err = -EINVAL;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
int
|
|
create_queues_and_rings(struct rte_eventdev *dev)
|
|
{
|
|
int err = 0;
|
|
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
|
|
device->nb_queues = 0;
|
|
|
|
if (device->nb_ports != device->max_port_nb) {
|
|
PMD_DRV_LOG(ERR, "Number ports setup:%u NOT EQUAL to max port"
|
|
" number:%u for this device",
|
|
device->nb_ports,
|
|
device->max_port_nb);
|
|
err = -1;
|
|
}
|
|
|
|
if (!err) {
|
|
/* We will have at least one opdl so create it now */
|
|
err = create_opdl(device);
|
|
}
|
|
|
|
if (!err) {
|
|
|
|
/* Create 1st "dummy" queue */
|
|
initialise_queue(device,
|
|
OPDL_Q_POS_START,
|
|
-1);
|
|
|
|
uint32_t i;
|
|
for (i = 0; i < device->nb_q_md; i++) {
|
|
|
|
/* Check */
|
|
if (!device->q_md[i].setup) {
|
|
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"queue meta data slot %u"
|
|
" not setup - FAILING",
|
|
dev->data->dev_id,
|
|
i);
|
|
err = -EINVAL;
|
|
break;
|
|
} else if (device->q_md[i].type !=
|
|
OPDL_Q_TYPE_SINGLE_LINK) {
|
|
|
|
if (!device->q_md[i + 1].setup) {
|
|
/* Create a simple ORDERED/ATOMIC
|
|
* queue at the end
|
|
*/
|
|
initialise_queue(device,
|
|
OPDL_Q_POS_END,
|
|
i);
|
|
|
|
} else {
|
|
/* Create a simple ORDERED/ATOMIC
|
|
* queue in the middle
|
|
*/
|
|
initialise_queue(device,
|
|
OPDL_Q_POS_MIDDLE,
|
|
i);
|
|
}
|
|
} else if (device->q_md[i].type ==
|
|
OPDL_Q_TYPE_SINGLE_LINK) {
|
|
|
|
/* create last queue for this opdl */
|
|
initialise_queue(device,
|
|
OPDL_Q_POS_END,
|
|
i);
|
|
|
|
err = create_link_opdl(device, i);
|
|
|
|
if (err)
|
|
break;
|
|
|
|
|
|
}
|
|
}
|
|
}
|
|
if (err)
|
|
destroy_queues_and_rings(dev);
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
int
|
|
initialise_all_other_ports(struct rte_eventdev *dev)
|
|
{
|
|
int err = 0;
|
|
struct opdl_stage *stage_inst = NULL;
|
|
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
|
|
uint32_t i;
|
|
for (i = 0; i < device->nb_ports; i++) {
|
|
struct opdl_port *port = &device->ports[i];
|
|
struct opdl_queue *queue = &device->queue[port->queue_id];
|
|
|
|
if (port->queue_id == 0) {
|
|
continue;
|
|
} else if (queue->q_type != OPDL_Q_TYPE_SINGLE_LINK) {
|
|
|
|
if (queue->q_pos == OPDL_Q_POS_MIDDLE) {
|
|
|
|
/* Regular port with claim/disclaim */
|
|
stage_inst = opdl_stage_add(
|
|
device->opdl[queue->opdl_id],
|
|
false,
|
|
false);
|
|
port->deq_stage_inst = stage_inst;
|
|
port->enq_stage_inst = stage_inst;
|
|
|
|
if (queue->q_type == OPDL_Q_TYPE_ATOMIC)
|
|
port->atomic_claim = true;
|
|
else
|
|
port->atomic_claim = false;
|
|
|
|
port->p_type = OPDL_REGULAR_PORT;
|
|
|
|
/* Add the port to the queue array of ports */
|
|
queue->ports[queue->nb_ports] = port;
|
|
port->instance_id = queue->nb_ports;
|
|
queue->nb_ports++;
|
|
opdl_stage_set_queue_id(stage_inst,
|
|
port->queue_id);
|
|
|
|
} else if (queue->q_pos == OPDL_Q_POS_END) {
|
|
|
|
/* tx port */
|
|
stage_inst = opdl_stage_add(
|
|
device->opdl[queue->opdl_id],
|
|
false,
|
|
false);
|
|
port->deq_stage_inst = stage_inst;
|
|
port->enq_stage_inst = NULL;
|
|
port->p_type = OPDL_PURE_TX_PORT;
|
|
|
|
/* Add the port to the queue array of ports */
|
|
queue->ports[queue->nb_ports] = port;
|
|
port->instance_id = queue->nb_ports;
|
|
queue->nb_ports++;
|
|
} else {
|
|
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"port %u:, linked incorrectly"
|
|
" to a q_pos START/INVALID %u",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
port->id,
|
|
queue->q_pos);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
} else if (queue->q_type == OPDL_Q_TYPE_SINGLE_LINK) {
|
|
|
|
port->p_type = OPDL_ASYNC_PORT;
|
|
|
|
/* -- tx -- */
|
|
stage_inst = opdl_stage_add(
|
|
device->opdl[queue->opdl_id],
|
|
false,
|
|
false); /* First stage */
|
|
port->deq_stage_inst = stage_inst;
|
|
|
|
/* Add the port to the queue array of ports */
|
|
queue->ports[queue->nb_ports] = port;
|
|
port->instance_id = queue->nb_ports;
|
|
queue->nb_ports++;
|
|
|
|
if (queue->nb_ports > 1) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"queue %u:, setup as SINGLE_LINK"
|
|
" but has more than one port linked",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
queue->external_qid);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
/* -- single instance rx for next opdl -- */
|
|
uint8_t next_qid =
|
|
device->q_map_ex_to_in[queue->external_qid] + 1;
|
|
if (next_qid < RTE_EVENT_MAX_QUEUES_PER_DEV &&
|
|
device->queue[next_qid].configured) {
|
|
|
|
/* Remap the queue */
|
|
queue = &device->queue[next_qid];
|
|
|
|
stage_inst = opdl_stage_add(
|
|
device->opdl[queue->opdl_id],
|
|
false,
|
|
true);
|
|
port->enq_stage_inst = stage_inst;
|
|
|
|
/* Add the port to the queue array of ports */
|
|
queue->ports[queue->nb_ports] = port;
|
|
port->instance_id = queue->nb_ports;
|
|
queue->nb_ports++;
|
|
if (queue->nb_ports > 1) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"dummy queue %u: for "
|
|
"port %u, "
|
|
"SINGLE_LINK but has more "
|
|
"than one port linked",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
next_qid,
|
|
port->id);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
/* Set this queue to initialized as it is never
|
|
* referenced by any ports
|
|
*/
|
|
queue->initialized = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Now that all ports are initialised we need to
|
|
* setup the last bit of stage md
|
|
*/
|
|
if (!err) {
|
|
for (i = 0; i < device->nb_ports; i++) {
|
|
struct opdl_port *port = &device->ports[i];
|
|
struct opdl_queue *queue =
|
|
&device->queue[port->queue_id];
|
|
|
|
if (port->configured &&
|
|
(port->queue_id != OPDL_INVALID_QID)) {
|
|
if (queue->nb_ports == 0) {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"queue:[%u] has no ports"
|
|
" linked to it",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
port->id);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
port->num_instance = queue->nb_ports;
|
|
port->initialized = 1;
|
|
queue->initialized = 1;
|
|
} else {
|
|
PMD_DRV_LOG(ERR, "DEV_ID:[%02d] : "
|
|
"Port:[%u] not configured invalid"
|
|
" queue configuration",
|
|
opdl_pmd_dev_id(port->opdl),
|
|
port->id);
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
int
|
|
initialise_queue_zero_ports(struct rte_eventdev *dev)
|
|
{
|
|
int err = 0;
|
|
uint8_t mt_rx = 0;
|
|
struct opdl_stage *stage_inst = NULL;
|
|
struct opdl_queue *queue = NULL;
|
|
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
|
|
/* Assign queue zero and figure out how many Q0 ports we have */
|
|
uint32_t i;
|
|
for (i = 0; i < device->nb_ports; i++) {
|
|
struct opdl_port *port = &device->ports[i];
|
|
if (port->queue_id == OPDL_INVALID_QID) {
|
|
port->queue_id = 0;
|
|
port->external_qid = OPDL_INVALID_QID;
|
|
port->p_type = OPDL_PURE_RX_PORT;
|
|
mt_rx++;
|
|
}
|
|
}
|
|
|
|
/* Create the stage */
|
|
stage_inst = opdl_stage_add(device->opdl[0],
|
|
(mt_rx > 1 ? true : false),
|
|
true);
|
|
if (stage_inst) {
|
|
|
|
/* Assign the new created input stage to all relevant ports */
|
|
for (i = 0; i < device->nb_ports; i++) {
|
|
struct opdl_port *port = &device->ports[i];
|
|
if (port->queue_id == 0) {
|
|
queue = &device->queue[port->queue_id];
|
|
port->enq_stage_inst = stage_inst;
|
|
port->deq_stage_inst = NULL;
|
|
port->configured = 1;
|
|
port->initialized = 1;
|
|
|
|
queue->ports[queue->nb_ports] = port;
|
|
port->instance_id = queue->nb_ports;
|
|
queue->nb_ports++;
|
|
}
|
|
}
|
|
} else {
|
|
err = -1;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
int
|
|
assign_internal_queue_ids(struct rte_eventdev *dev)
|
|
{
|
|
int err = 0;
|
|
struct opdl_evdev *device = opdl_pmd_priv(dev);
|
|
uint32_t i;
|
|
|
|
for (i = 0; i < device->nb_ports; i++) {
|
|
struct opdl_port *port = &device->ports[i];
|
|
if (port->external_qid != OPDL_INVALID_QID) {
|
|
port->queue_id =
|
|
device->q_map_ex_to_in[port->external_qid];
|
|
|
|
/* Now do the external_qid of the next queue */
|
|
struct opdl_queue *queue =
|
|
&device->queue[port->queue_id];
|
|
if (queue->q_pos == OPDL_Q_POS_END)
|
|
port->next_external_qid =
|
|
device->queue[port->queue_id + 2].external_qid;
|
|
else
|
|
port->next_external_qid =
|
|
device->queue[port->queue_id + 1].external_qid;
|
|
}
|
|
}
|
|
return err;
|
|
}
|