2018-01-08 05:25:17 +00:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
* Copyright(c) 2017 Cavium, Inc
|
2017-07-04 04:53:01 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _EVT_OPTIONS_
|
|
|
|
#define _EVT_OPTIONS_
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
#include <rte_common.h>
|
2018-04-06 15:13:21 +00:00
|
|
|
#include <rte_ethdev.h>
|
2017-07-04 04:53:01 +00:00
|
|
|
#include <rte_eventdev.h>
|
|
|
|
#include <rte_lcore.h>
|
|
|
|
|
|
|
|
#include "evt_common.h"
|
|
|
|
|
2017-07-04 04:53:03 +00:00
|
|
|
#define EVT_BOOL_FMT(x) ((x) ? "true" : "false")
|
|
|
|
|
2017-07-04 04:53:04 +00:00
|
|
|
#define EVT_VERBOSE ("verbose")
|
|
|
|
#define EVT_DEVICE ("dev")
|
|
|
|
#define EVT_TEST ("test")
|
|
|
|
#define EVT_PROD_LCORES ("plcores")
|
|
|
|
#define EVT_WORK_LCORES ("wlcores")
|
|
|
|
#define EVT_NB_FLOWS ("nb_flows")
|
|
|
|
#define EVT_SOCKET_ID ("socket_id")
|
|
|
|
#define EVT_POOL_SZ ("pool_sz")
|
|
|
|
#define EVT_WKR_DEQ_DEP ("worker_deq_depth")
|
|
|
|
#define EVT_NB_PKTS ("nb_pkts")
|
|
|
|
#define EVT_NB_STAGES ("nb_stages")
|
|
|
|
#define EVT_SCHED_TYPE_LIST ("stlist")
|
|
|
|
#define EVT_FWD_LATENCY ("fwd_latency")
|
|
|
|
#define EVT_QUEUE_PRIORITY ("queue_priority")
|
2017-12-11 15:13:39 +00:00
|
|
|
#define EVT_PROD_ETHDEV ("prod_type_ethdev")
|
2018-04-06 15:13:21 +00:00
|
|
|
#define EVT_PROD_TIMERDEV ("prod_type_timerdev")
|
2018-04-06 15:13:22 +00:00
|
|
|
#define EVT_PROD_TIMERDEV_BURST ("prod_type_timerdev_burst")
|
2017-07-04 04:53:04 +00:00
|
|
|
#define EVT_HELP ("help")
|
|
|
|
|
2017-12-11 15:13:39 +00:00
|
|
|
enum evt_prod_type {
|
|
|
|
EVT_PROD_TYPE_NONE,
|
|
|
|
EVT_PROD_TYPE_SYNT, /* Producer type Synthetic i.e. CPU. */
|
|
|
|
EVT_PROD_TYPE_ETH_RX_ADPTR, /* Producer type Eth Rx Adapter. */
|
2018-04-06 15:13:21 +00:00
|
|
|
EVT_PROD_TYPE_EVENT_TIMER_ADPTR, /* Producer type Timer Adapter. */
|
2017-12-11 15:13:39 +00:00
|
|
|
EVT_PROD_TYPE_MAX,
|
|
|
|
};
|
|
|
|
|
2017-07-04 04:53:01 +00:00
|
|
|
struct evt_options {
|
|
|
|
#define EVT_TEST_NAME_MAX_LEN 32
|
|
|
|
char test_name[EVT_TEST_NAME_MAX_LEN];
|
|
|
|
bool plcores[RTE_MAX_LCORE];
|
|
|
|
bool wlcores[RTE_MAX_LCORE];
|
|
|
|
uint8_t sched_type_list[EVT_MAX_STAGES];
|
|
|
|
uint32_t nb_flows;
|
|
|
|
int socket_id;
|
|
|
|
int pool_sz;
|
|
|
|
int nb_stages;
|
|
|
|
int verbose_level;
|
|
|
|
uint64_t nb_pkts;
|
2018-04-06 15:13:21 +00:00
|
|
|
uint8_t nb_timer_adptrs;
|
|
|
|
uint64_t nb_timers;
|
|
|
|
uint64_t timer_tick_nsec;
|
|
|
|
uint64_t optm_timer_tick_nsec;
|
|
|
|
uint64_t max_tmo_nsec;
|
|
|
|
uint64_t expiry_nsec;
|
2017-07-04 04:53:01 +00:00
|
|
|
uint16_t wkr_deq_dep;
|
|
|
|
uint8_t dev_id;
|
|
|
|
uint32_t fwd_latency:1;
|
|
|
|
uint32_t q_priority:1;
|
2017-12-11 15:13:39 +00:00
|
|
|
enum evt_prod_type prod_type;
|
2018-04-06 15:13:22 +00:00
|
|
|
uint8_t timdev_use_burst;
|
2018-04-06 15:13:21 +00:00
|
|
|
uint8_t timdev_cnt;
|
2017-07-04 04:53:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void evt_options_default(struct evt_options *opt);
|
2017-07-04 04:53:04 +00:00
|
|
|
int evt_options_parse(struct evt_options *opt, int argc, char **argv);
|
2017-07-04 04:53:03 +00:00
|
|
|
void evt_options_dump(struct evt_options *opt);
|
2017-07-04 04:53:01 +00:00
|
|
|
|
2017-07-04 04:53:02 +00:00
|
|
|
/* options check helpers */
|
|
|
|
static inline bool
|
|
|
|
evt_lcores_has_overlap(bool lcores[], int lcore)
|
|
|
|
{
|
|
|
|
if (lcores[lcore] == true) {
|
|
|
|
evt_err("lcore overlaps at %d", lcore);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
evt_lcores_has_overlap_multi(bool lcoresx[], bool lcoresy[])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < RTE_MAX_LCORE; i++) {
|
|
|
|
if (lcoresx[i] && lcoresy[i]) {
|
|
|
|
evt_err("lcores overlaps at %d", i);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
evt_has_active_lcore(bool lcores[])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < RTE_MAX_LCORE; i++)
|
|
|
|
if (lcores[i])
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
evt_nr_active_lcores(bool lcores[])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int c = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < RTE_MAX_LCORE; i++)
|
|
|
|
if (lcores[i])
|
|
|
|
c++;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
evt_get_first_active_lcore(bool lcores[])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < RTE_MAX_LCORE; i++)
|
|
|
|
if (lcores[i])
|
|
|
|
return i;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
evt_has_disabled_lcore(bool lcores[])
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < RTE_MAX_LCORE; i++)
|
|
|
|
if ((lcores[i] == true) && !(rte_lcore_is_enabled(i)))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
evt_has_invalid_stage(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
if (!opt->nb_stages) {
|
|
|
|
evt_err("need minimum one stage, check --stlist");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (opt->nb_stages > EVT_MAX_STAGES) {
|
|
|
|
evt_err("requested changes are beyond EVT_MAX_STAGES=%d",
|
|
|
|
EVT_MAX_STAGES);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
evt_has_invalid_sched_type(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < opt->nb_stages; i++) {
|
|
|
|
if (opt->sched_type_list[i] > RTE_SCHED_TYPE_PARALLEL) {
|
|
|
|
evt_err("invalid sched_type %d at %d",
|
|
|
|
opt->sched_type_list[i], i);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-04 04:53:03 +00:00
|
|
|
/* option dump helpers */
|
|
|
|
static inline void
|
|
|
|
evt_dump_worker_lcores(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
|
|
|
evt_dump_begin("worker lcores");
|
|
|
|
for (c = 0; c < RTE_MAX_LCORE; c++) {
|
|
|
|
if (opt->wlcores[c])
|
|
|
|
printf("%d ", c);
|
|
|
|
}
|
|
|
|
evt_dump_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_producer_lcores(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
|
|
|
evt_dump_begin("producer lcores");
|
|
|
|
for (c = 0; c < RTE_MAX_LCORE; c++) {
|
|
|
|
if (opt->plcores[c])
|
|
|
|
printf("%d ", c);
|
|
|
|
}
|
|
|
|
evt_dump_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_nb_flows(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
evt_dump("nb_flows", "%d", opt->nb_flows);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_worker_dequeue_depth(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
evt_dump("worker deq depth", "%d", opt->wkr_deq_dep);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_nb_stages(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
evt_dump("nb_stages", "%d", opt->nb_stages);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_fwd_latency(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
evt_dump("fwd_latency", "%s", EVT_BOOL_FMT(opt->fwd_latency));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_queue_priority(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
evt_dump("queue_priority", "%s", EVT_BOOL_FMT(opt->q_priority));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline const char*
|
|
|
|
evt_sched_type_2_str(uint8_t sched_type)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (sched_type == RTE_SCHED_TYPE_ORDERED)
|
|
|
|
return "O";
|
|
|
|
else if (sched_type == RTE_SCHED_TYPE_ATOMIC)
|
|
|
|
return "A";
|
|
|
|
else if (sched_type == RTE_SCHED_TYPE_PARALLEL)
|
|
|
|
return "P";
|
|
|
|
else
|
|
|
|
return "I";
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
evt_dump_sched_type_list(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
evt_dump_begin("sched_type_list");
|
|
|
|
for (i = 0; i < opt->nb_stages; i++)
|
|
|
|
printf("%s ", evt_sched_type_2_str(opt->sched_type_list[i]));
|
|
|
|
|
|
|
|
evt_dump_end;
|
|
|
|
}
|
2017-07-04 04:53:02 +00:00
|
|
|
|
2017-12-11 15:13:39 +00:00
|
|
|
#define EVT_PROD_MAX_NAME_LEN 50
|
|
|
|
static inline void
|
|
|
|
evt_dump_producer_type(struct evt_options *opt)
|
|
|
|
{
|
|
|
|
char name[EVT_PROD_MAX_NAME_LEN];
|
|
|
|
|
|
|
|
switch (opt->prod_type) {
|
|
|
|
default:
|
|
|
|
case EVT_PROD_TYPE_SYNT:
|
|
|
|
snprintf(name, EVT_PROD_MAX_NAME_LEN,
|
|
|
|
"Synthetic producer lcores");
|
|
|
|
break;
|
|
|
|
case EVT_PROD_TYPE_ETH_RX_ADPTR:
|
|
|
|
snprintf(name, EVT_PROD_MAX_NAME_LEN,
|
|
|
|
"Ethdev Rx Adapter producers");
|
2018-04-06 15:13:21 +00:00
|
|
|
evt_dump("nb_ethdev", "%d", rte_eth_dev_count());
|
|
|
|
break;
|
|
|
|
case EVT_PROD_TYPE_EVENT_TIMER_ADPTR:
|
2018-04-06 15:13:22 +00:00
|
|
|
if (opt->timdev_use_burst)
|
|
|
|
snprintf(name, EVT_PROD_MAX_NAME_LEN,
|
|
|
|
"Event timer adapter burst mode producer");
|
|
|
|
else
|
|
|
|
snprintf(name, EVT_PROD_MAX_NAME_LEN,
|
2018-04-06 15:13:21 +00:00
|
|
|
"Event timer adapter producer");
|
|
|
|
evt_dump("nb_timer_adapters", "%d", opt->nb_timer_adptrs);
|
|
|
|
evt_dump("max_tmo_nsec", "%"PRIu64"", opt->max_tmo_nsec);
|
|
|
|
evt_dump("expiry_nsec", "%"PRIu64"", opt->expiry_nsec);
|
|
|
|
if (opt->optm_timer_tick_nsec)
|
|
|
|
evt_dump("optm_timer_tick_ns", "%"PRIu64"",
|
|
|
|
opt->optm_timer_tick_nsec);
|
|
|
|
else
|
|
|
|
evt_dump("timer_tick_ns", "%"PRIu64"",
|
|
|
|
opt->timer_tick_nsec);
|
2017-12-11 15:13:39 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
evt_dump("prod_type", "%s", name);
|
|
|
|
}
|
|
|
|
|
2017-07-04 04:53:01 +00:00
|
|
|
#endif /* _EVT_OPTIONS_ */
|