/* * SPDX-License-Identifier: BSD-3-Clause * Copyright 2016 Intel Corporation. * Copyright 2017 Cavium, Inc. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #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);