mirror of https://github.com/F-Stack/f-stack.git
176 lines
4.1 KiB
C
176 lines
4.1 KiB
C
/*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright 2016 Intel Corporation.
|
|
* Copyright 2017 Cavium, Inc.
|
|
*/
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <rte_eal.h>
|
|
#include <rte_mempool.h>
|
|
#include <rte_mbuf.h>
|
|
#include <rte_launch.h>
|
|
#include <rte_malloc.h>
|
|
#include <rte_random.h>
|
|
#include <rte_cycles.h>
|
|
#include <rte_ethdev.h>
|
|
#include <rte_eventdev.h>
|
|
#include <rte_event_eth_rx_adapter.h>
|
|
#include <rte_event_eth_tx_adapter.h>
|
|
#include <rte_service.h>
|
|
#include <rte_service_component.h>
|
|
|
|
#define MAX_NUM_STAGES 8
|
|
#define BATCH_SIZE 16
|
|
#define MAX_NUM_CORE 64
|
|
|
|
struct worker_data {
|
|
uint8_t dev_id;
|
|
uint8_t port_id;
|
|
} __rte_cache_aligned;
|
|
|
|
typedef int (*worker_loop)(void *);
|
|
typedef void (*schedule_loop)(unsigned int);
|
|
typedef int (*eventdev_setup)(struct worker_data *);
|
|
typedef void (*adapter_setup)(uint16_t nb_ports);
|
|
typedef void (*opt_check)(void);
|
|
|
|
struct setup_data {
|
|
worker_loop worker;
|
|
schedule_loop scheduler;
|
|
eventdev_setup evdev_setup;
|
|
adapter_setup adptr_setup;
|
|
opt_check check_opt;
|
|
};
|
|
|
|
struct fastpath_data {
|
|
volatile int done;
|
|
uint32_t evdev_service_id;
|
|
uint32_t rxadptr_service_id;
|
|
uint32_t txadptr_service_id;
|
|
bool rx_single;
|
|
bool tx_single;
|
|
bool sched_single;
|
|
uint64_t rx_core[MAX_NUM_CORE];
|
|
uint64_t tx_core[MAX_NUM_CORE];
|
|
uint64_t sched_core[MAX_NUM_CORE];
|
|
uint64_t worker_core[MAX_NUM_CORE];
|
|
struct setup_data cap;
|
|
} __rte_cache_aligned;
|
|
|
|
struct config_data {
|
|
unsigned int active_cores;
|
|
unsigned int num_workers;
|
|
int64_t num_packets;
|
|
uint64_t num_mbuf;
|
|
unsigned int num_fids;
|
|
int queue_type;
|
|
int worker_cycles;
|
|
int enable_queue_priorities;
|
|
int quiet;
|
|
int dump_dev;
|
|
int dump_dev_signal;
|
|
int all_type_queues;
|
|
unsigned int num_stages;
|
|
unsigned int worker_cq_depth;
|
|
unsigned int rx_stride;
|
|
/* Use rx stride value to reduce congestion in entry queue when using
|
|
* multiple eth ports by forming multiple event queue pipelines.
|
|
*/
|
|
int16_t next_qid[MAX_NUM_STAGES+2];
|
|
int16_t qid[MAX_NUM_STAGES];
|
|
uint8_t rx_adapter_id;
|
|
uint8_t tx_adapter_id;
|
|
uint8_t tx_queue_id;
|
|
uint64_t worker_lcore_mask;
|
|
uint64_t rx_lcore_mask;
|
|
uint64_t tx_lcore_mask;
|
|
uint64_t sched_lcore_mask;
|
|
};
|
|
|
|
struct port_link {
|
|
uint8_t queue_id;
|
|
uint8_t priority;
|
|
};
|
|
|
|
extern struct fastpath_data *fdata;
|
|
extern struct config_data cdata;
|
|
|
|
static __rte_always_inline void
|
|
exchange_mac(struct rte_mbuf *m)
|
|
{
|
|
struct rte_ether_hdr *eth;
|
|
struct rte_ether_addr addr;
|
|
|
|
/* change mac addresses on packet (to use mbuf data) */
|
|
eth = rte_pktmbuf_mtod(m, struct rte_ether_hdr *);
|
|
rte_ether_addr_copy(ð->dst_addr, &addr);
|
|
rte_ether_addr_copy(&addr, ð->dst_addr);
|
|
}
|
|
|
|
static __rte_always_inline void
|
|
work(void)
|
|
{
|
|
/* do a number of cycles of work per packet */
|
|
volatile uint64_t start_tsc = rte_rdtsc();
|
|
while (rte_rdtsc() < start_tsc + cdata.worker_cycles)
|
|
rte_pause();
|
|
}
|
|
|
|
static __rte_always_inline void
|
|
schedule_devices(unsigned int lcore_id)
|
|
{
|
|
if (fdata->rx_core[lcore_id]) {
|
|
rte_service_run_iter_on_app_lcore(fdata->rxadptr_service_id,
|
|
!fdata->rx_single);
|
|
}
|
|
|
|
if (fdata->sched_core[lcore_id]) {
|
|
rte_service_run_iter_on_app_lcore(fdata->evdev_service_id,
|
|
!fdata->sched_single);
|
|
if (cdata.dump_dev_signal) {
|
|
rte_event_dev_dump(0, stdout);
|
|
cdata.dump_dev_signal = 0;
|
|
}
|
|
}
|
|
|
|
if (fdata->tx_core[lcore_id]) {
|
|
rte_service_run_iter_on_app_lcore(fdata->txadptr_service_id,
|
|
!fdata->tx_single);
|
|
}
|
|
}
|
|
|
|
static void
|
|
event_port_flush(uint8_t dev_id __rte_unused, struct rte_event ev,
|
|
void *args __rte_unused)
|
|
{
|
|
rte_mempool_put(args, ev.event_ptr);
|
|
}
|
|
|
|
static inline void
|
|
worker_cleanup(uint8_t dev_id, uint8_t port_id, struct rte_event events[],
|
|
uint16_t nb_enq, uint16_t nb_deq)
|
|
{
|
|
int i;
|
|
|
|
if (!(nb_deq - nb_enq))
|
|
return;
|
|
|
|
if (nb_deq) {
|
|
for (i = nb_enq; i < nb_deq; i++) {
|
|
if (events[i].op == RTE_EVENT_OP_RELEASE)
|
|
continue;
|
|
rte_pktmbuf_free(events[i].mbuf);
|
|
}
|
|
|
|
for (i = 0; i < nb_deq; i++)
|
|
events[i].op = RTE_EVENT_OP_RELEASE;
|
|
rte_event_enqueue_burst(dev_id, port_id, events, nb_deq);
|
|
}
|
|
|
|
rte_event_port_quiesce(dev_id, port_id, event_port_flush, NULL);
|
|
}
|
|
|
|
void set_worker_generic_setup_data(struct setup_data *caps, bool burst);
|
|
void set_worker_tx_enq_setup_data(struct setup_data *caps, bool burst);
|