1d30ea906Sjfb8856606 /* SPDX-License-Identifier: BSD-3-Clause
2d30ea906Sjfb8856606 * Copyright(c) 2017 Cavium, Inc
32bfe3f2eSlogwang */
42bfe3f2eSlogwang
52bfe3f2eSlogwang #include "test_perf_common.h"
62bfe3f2eSlogwang
72bfe3f2eSlogwang int
perf_test_result(struct evt_test * test,struct evt_options * opt)82bfe3f2eSlogwang perf_test_result(struct evt_test *test, struct evt_options *opt)
92bfe3f2eSlogwang {
102bfe3f2eSlogwang RTE_SET_USED(opt);
11d30ea906Sjfb8856606 int i;
12d30ea906Sjfb8856606 uint64_t total = 0;
132bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
142bfe3f2eSlogwang
15d30ea906Sjfb8856606 printf("Packet distribution across worker cores :\n");
16d30ea906Sjfb8856606 for (i = 0; i < t->nb_workers; i++)
17d30ea906Sjfb8856606 total += t->worker[i].processed_pkts;
18d30ea906Sjfb8856606 for (i = 0; i < t->nb_workers; i++)
19d30ea906Sjfb8856606 printf("Worker %d packets: "CLGRN"%"PRIx64" "CLNRM"percentage:"
20d30ea906Sjfb8856606 CLGRN" %3.2f\n"CLNRM, i,
21d30ea906Sjfb8856606 t->worker[i].processed_pkts,
22d30ea906Sjfb8856606 (((double)t->worker[i].processed_pkts)/total)
23d30ea906Sjfb8856606 * 100);
24d30ea906Sjfb8856606
252bfe3f2eSlogwang return t->result;
262bfe3f2eSlogwang }
272bfe3f2eSlogwang
282bfe3f2eSlogwang static inline int
perf_producer(void * arg)292bfe3f2eSlogwang perf_producer(void *arg)
302bfe3f2eSlogwang {
314418919fSjohnjiang int i;
322bfe3f2eSlogwang struct prod_data *p = arg;
332bfe3f2eSlogwang struct test_perf *t = p->t;
342bfe3f2eSlogwang struct evt_options *opt = t->opt;
352bfe3f2eSlogwang const uint8_t dev_id = p->dev_id;
362bfe3f2eSlogwang const uint8_t port = p->port_id;
372bfe3f2eSlogwang struct rte_mempool *pool = t->pool;
382bfe3f2eSlogwang const uint64_t nb_pkts = t->nb_pkts;
392bfe3f2eSlogwang const uint32_t nb_flows = t->nb_flows;
402bfe3f2eSlogwang uint32_t flow_counter = 0;
412bfe3f2eSlogwang uint64_t count = 0;
424418919fSjohnjiang struct perf_elt *m[BURST_SIZE + 1] = {NULL};
432bfe3f2eSlogwang struct rte_event ev;
442bfe3f2eSlogwang
452bfe3f2eSlogwang if (opt->verbose_level > 1)
462bfe3f2eSlogwang printf("%s(): lcore %d dev_id %d port=%d queue %d\n", __func__,
472bfe3f2eSlogwang rte_lcore_id(), dev_id, port, p->queue_id);
482bfe3f2eSlogwang
492bfe3f2eSlogwang ev.event = 0;
502bfe3f2eSlogwang ev.op = RTE_EVENT_OP_NEW;
512bfe3f2eSlogwang ev.queue_id = p->queue_id;
522bfe3f2eSlogwang ev.sched_type = t->opt->sched_type_list[0];
532bfe3f2eSlogwang ev.priority = RTE_EVENT_DEV_PRIORITY_NORMAL;
542bfe3f2eSlogwang ev.event_type = RTE_EVENT_TYPE_CPU;
552bfe3f2eSlogwang ev.sub_event_type = 0; /* stage 0 */
562bfe3f2eSlogwang
572bfe3f2eSlogwang while (count < nb_pkts && t->done == false) {
584418919fSjohnjiang if (rte_mempool_get_bulk(pool, (void **)m, BURST_SIZE) < 0)
592bfe3f2eSlogwang continue;
604418919fSjohnjiang for (i = 0; i < BURST_SIZE; i++) {
612bfe3f2eSlogwang ev.flow_id = flow_counter++ % nb_flows;
624418919fSjohnjiang ev.event_ptr = m[i];
634418919fSjohnjiang m[i]->timestamp = rte_get_timer_cycles();
644418919fSjohnjiang while (rte_event_enqueue_burst(dev_id,
654418919fSjohnjiang port, &ev, 1) != 1) {
662bfe3f2eSlogwang if (t->done)
672bfe3f2eSlogwang break;
682bfe3f2eSlogwang rte_pause();
694418919fSjohnjiang m[i]->timestamp = rte_get_timer_cycles();
702bfe3f2eSlogwang }
714418919fSjohnjiang }
724418919fSjohnjiang count += BURST_SIZE;
732bfe3f2eSlogwang }
742bfe3f2eSlogwang
752bfe3f2eSlogwang return 0;
762bfe3f2eSlogwang }
772bfe3f2eSlogwang
78d30ea906Sjfb8856606 static inline int
perf_event_timer_producer(void * arg)79d30ea906Sjfb8856606 perf_event_timer_producer(void *arg)
80d30ea906Sjfb8856606 {
814418919fSjohnjiang int i;
82d30ea906Sjfb8856606 struct prod_data *p = arg;
83d30ea906Sjfb8856606 struct test_perf *t = p->t;
84d30ea906Sjfb8856606 struct evt_options *opt = t->opt;
85d30ea906Sjfb8856606 uint32_t flow_counter = 0;
86d30ea906Sjfb8856606 uint64_t count = 0;
87d30ea906Sjfb8856606 uint64_t arm_latency = 0;
88d30ea906Sjfb8856606 const uint8_t nb_timer_adptrs = opt->nb_timer_adptrs;
89d30ea906Sjfb8856606 const uint32_t nb_flows = t->nb_flows;
90d30ea906Sjfb8856606 const uint64_t nb_timers = opt->nb_timers;
91d30ea906Sjfb8856606 struct rte_mempool *pool = t->pool;
924418919fSjohnjiang struct perf_elt *m[BURST_SIZE + 1] = {NULL};
93d30ea906Sjfb8856606 struct rte_event_timer_adapter **adptr = t->timer_adptr;
94d30ea906Sjfb8856606 struct rte_event_timer tim;
95d30ea906Sjfb8856606 uint64_t timeout_ticks = opt->expiry_nsec / opt->timer_tick_nsec;
96d30ea906Sjfb8856606
97d30ea906Sjfb8856606 memset(&tim, 0, sizeof(struct rte_event_timer));
98d30ea906Sjfb8856606 timeout_ticks = opt->optm_timer_tick_nsec ?
99d30ea906Sjfb8856606 (timeout_ticks * opt->timer_tick_nsec)
100d30ea906Sjfb8856606 / opt->optm_timer_tick_nsec : timeout_ticks;
101d30ea906Sjfb8856606 timeout_ticks += timeout_ticks ? 0 : 1;
102d30ea906Sjfb8856606 tim.ev.event_type = RTE_EVENT_TYPE_TIMER;
103d30ea906Sjfb8856606 tim.ev.op = RTE_EVENT_OP_NEW;
104d30ea906Sjfb8856606 tim.ev.sched_type = t->opt->sched_type_list[0];
105d30ea906Sjfb8856606 tim.ev.queue_id = p->queue_id;
106d30ea906Sjfb8856606 tim.ev.priority = RTE_EVENT_DEV_PRIORITY_NORMAL;
107d30ea906Sjfb8856606 tim.state = RTE_EVENT_TIMER_NOT_ARMED;
108d30ea906Sjfb8856606 tim.timeout_ticks = timeout_ticks;
109d30ea906Sjfb8856606
110d30ea906Sjfb8856606 if (opt->verbose_level > 1)
111d30ea906Sjfb8856606 printf("%s(): lcore %d\n", __func__, rte_lcore_id());
112d30ea906Sjfb8856606
113d30ea906Sjfb8856606 while (count < nb_timers && t->done == false) {
1144418919fSjohnjiang if (rte_mempool_get_bulk(pool, (void **)m, BURST_SIZE) < 0)
115d30ea906Sjfb8856606 continue;
1164418919fSjohnjiang for (i = 0; i < BURST_SIZE; i++) {
1174418919fSjohnjiang rte_prefetch0(m[i + 1]);
1184418919fSjohnjiang m[i]->tim = tim;
1194418919fSjohnjiang m[i]->tim.ev.flow_id = flow_counter++ % nb_flows;
1204418919fSjohnjiang m[i]->tim.ev.event_ptr = m[i];
1214418919fSjohnjiang m[i]->timestamp = rte_get_timer_cycles();
122d30ea906Sjfb8856606 while (rte_event_timer_arm_burst(
123d30ea906Sjfb8856606 adptr[flow_counter % nb_timer_adptrs],
1244418919fSjohnjiang (struct rte_event_timer **)&m[i], 1) != 1) {
125d30ea906Sjfb8856606 if (t->done)
126d30ea906Sjfb8856606 break;
1274418919fSjohnjiang m[i]->timestamp = rte_get_timer_cycles();
128d30ea906Sjfb8856606 }
1294418919fSjohnjiang arm_latency += rte_get_timer_cycles() - m[i]->timestamp;
1304418919fSjohnjiang }
1314418919fSjohnjiang count += BURST_SIZE;
132d30ea906Sjfb8856606 }
133d30ea906Sjfb8856606 fflush(stdout);
134d30ea906Sjfb8856606 rte_delay_ms(1000);
135d30ea906Sjfb8856606 printf("%s(): lcore %d Average event timer arm latency = %.3f us\n",
1364418919fSjohnjiang __func__, rte_lcore_id(),
1374418919fSjohnjiang count ? (float)(arm_latency / count) /
1384418919fSjohnjiang (rte_get_timer_hz() / 1000000) : 0);
139d30ea906Sjfb8856606 return 0;
140d30ea906Sjfb8856606 }
141d30ea906Sjfb8856606
142d30ea906Sjfb8856606 static inline int
perf_event_timer_producer_burst(void * arg)143d30ea906Sjfb8856606 perf_event_timer_producer_burst(void *arg)
144d30ea906Sjfb8856606 {
145d30ea906Sjfb8856606 int i;
146d30ea906Sjfb8856606 struct prod_data *p = arg;
147d30ea906Sjfb8856606 struct test_perf *t = p->t;
148d30ea906Sjfb8856606 struct evt_options *opt = t->opt;
149d30ea906Sjfb8856606 uint32_t flow_counter = 0;
150d30ea906Sjfb8856606 uint64_t count = 0;
151d30ea906Sjfb8856606 uint64_t arm_latency = 0;
152d30ea906Sjfb8856606 const uint8_t nb_timer_adptrs = opt->nb_timer_adptrs;
153d30ea906Sjfb8856606 const uint32_t nb_flows = t->nb_flows;
154d30ea906Sjfb8856606 const uint64_t nb_timers = opt->nb_timers;
155d30ea906Sjfb8856606 struct rte_mempool *pool = t->pool;
156d30ea906Sjfb8856606 struct perf_elt *m[BURST_SIZE + 1] = {NULL};
157d30ea906Sjfb8856606 struct rte_event_timer_adapter **adptr = t->timer_adptr;
158d30ea906Sjfb8856606 struct rte_event_timer tim;
159d30ea906Sjfb8856606 uint64_t timeout_ticks = opt->expiry_nsec / opt->timer_tick_nsec;
160d30ea906Sjfb8856606
161d30ea906Sjfb8856606 memset(&tim, 0, sizeof(struct rte_event_timer));
162d30ea906Sjfb8856606 timeout_ticks = opt->optm_timer_tick_nsec ?
163d30ea906Sjfb8856606 (timeout_ticks * opt->timer_tick_nsec)
164d30ea906Sjfb8856606 / opt->optm_timer_tick_nsec : timeout_ticks;
165d30ea906Sjfb8856606 timeout_ticks += timeout_ticks ? 0 : 1;
166d30ea906Sjfb8856606 tim.ev.event_type = RTE_EVENT_TYPE_TIMER;
167d30ea906Sjfb8856606 tim.ev.op = RTE_EVENT_OP_NEW;
168d30ea906Sjfb8856606 tim.ev.sched_type = t->opt->sched_type_list[0];
169d30ea906Sjfb8856606 tim.ev.queue_id = p->queue_id;
170d30ea906Sjfb8856606 tim.ev.priority = RTE_EVENT_DEV_PRIORITY_NORMAL;
171d30ea906Sjfb8856606 tim.state = RTE_EVENT_TIMER_NOT_ARMED;
172d30ea906Sjfb8856606 tim.timeout_ticks = timeout_ticks;
173d30ea906Sjfb8856606
174d30ea906Sjfb8856606 if (opt->verbose_level > 1)
175d30ea906Sjfb8856606 printf("%s(): lcore %d\n", __func__, rte_lcore_id());
176d30ea906Sjfb8856606
177d30ea906Sjfb8856606 while (count < nb_timers && t->done == false) {
178d30ea906Sjfb8856606 if (rte_mempool_get_bulk(pool, (void **)m, BURST_SIZE) < 0)
179d30ea906Sjfb8856606 continue;
180d30ea906Sjfb8856606 for (i = 0; i < BURST_SIZE; i++) {
181d30ea906Sjfb8856606 rte_prefetch0(m[i + 1]);
182d30ea906Sjfb8856606 m[i]->tim = tim;
183d30ea906Sjfb8856606 m[i]->tim.ev.flow_id = flow_counter++ % nb_flows;
184d30ea906Sjfb8856606 m[i]->tim.ev.event_ptr = m[i];
185d30ea906Sjfb8856606 m[i]->timestamp = rte_get_timer_cycles();
186d30ea906Sjfb8856606 }
187d30ea906Sjfb8856606 rte_event_timer_arm_tmo_tick_burst(
188d30ea906Sjfb8856606 adptr[flow_counter % nb_timer_adptrs],
189d30ea906Sjfb8856606 (struct rte_event_timer **)m,
190d30ea906Sjfb8856606 tim.timeout_ticks,
191d30ea906Sjfb8856606 BURST_SIZE);
192d30ea906Sjfb8856606 arm_latency += rte_get_timer_cycles() - m[i - 1]->timestamp;
193d30ea906Sjfb8856606 count += BURST_SIZE;
194d30ea906Sjfb8856606 }
195d30ea906Sjfb8856606 fflush(stdout);
196d30ea906Sjfb8856606 rte_delay_ms(1000);
197d30ea906Sjfb8856606 printf("%s(): lcore %d Average event timer arm latency = %.3f us\n",
1984418919fSjohnjiang __func__, rte_lcore_id(),
1994418919fSjohnjiang count ? (float)(arm_latency / count) /
2004418919fSjohnjiang (rte_get_timer_hz() / 1000000) : 0);
201d30ea906Sjfb8856606 return 0;
202d30ea906Sjfb8856606 }
203d30ea906Sjfb8856606
204d30ea906Sjfb8856606 static int
perf_producer_wrapper(void * arg)205d30ea906Sjfb8856606 perf_producer_wrapper(void *arg)
206d30ea906Sjfb8856606 {
207d30ea906Sjfb8856606 struct prod_data *p = arg;
208d30ea906Sjfb8856606 struct test_perf *t = p->t;
209d30ea906Sjfb8856606 /* Launch the producer function only in case of synthetic producer. */
210d30ea906Sjfb8856606 if (t->opt->prod_type == EVT_PROD_TYPE_SYNT)
211d30ea906Sjfb8856606 return perf_producer(arg);
212d30ea906Sjfb8856606 else if (t->opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR &&
213d30ea906Sjfb8856606 !t->opt->timdev_use_burst)
214d30ea906Sjfb8856606 return perf_event_timer_producer(arg);
215d30ea906Sjfb8856606 else if (t->opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR &&
216d30ea906Sjfb8856606 t->opt->timdev_use_burst)
217d30ea906Sjfb8856606 return perf_event_timer_producer_burst(arg);
218d30ea906Sjfb8856606 return 0;
219d30ea906Sjfb8856606 }
220d30ea906Sjfb8856606
2212bfe3f2eSlogwang static inline uint64_t
processed_pkts(struct test_perf * t)2222bfe3f2eSlogwang processed_pkts(struct test_perf *t)
2232bfe3f2eSlogwang {
2242bfe3f2eSlogwang uint8_t i;
2252bfe3f2eSlogwang uint64_t total = 0;
2262bfe3f2eSlogwang
2272bfe3f2eSlogwang rte_smp_rmb();
2282bfe3f2eSlogwang for (i = 0; i < t->nb_workers; i++)
2292bfe3f2eSlogwang total += t->worker[i].processed_pkts;
2302bfe3f2eSlogwang
2312bfe3f2eSlogwang return total;
2322bfe3f2eSlogwang }
2332bfe3f2eSlogwang
2342bfe3f2eSlogwang static inline uint64_t
total_latency(struct test_perf * t)2352bfe3f2eSlogwang total_latency(struct test_perf *t)
2362bfe3f2eSlogwang {
2372bfe3f2eSlogwang uint8_t i;
2382bfe3f2eSlogwang uint64_t total = 0;
2392bfe3f2eSlogwang
2402bfe3f2eSlogwang rte_smp_rmb();
2412bfe3f2eSlogwang for (i = 0; i < t->nb_workers; i++)
2422bfe3f2eSlogwang total += t->worker[i].latency;
2432bfe3f2eSlogwang
2442bfe3f2eSlogwang return total;
2452bfe3f2eSlogwang }
2462bfe3f2eSlogwang
2472bfe3f2eSlogwang
2482bfe3f2eSlogwang int
perf_launch_lcores(struct evt_test * test,struct evt_options * opt,int (* worker)(void *))2492bfe3f2eSlogwang perf_launch_lcores(struct evt_test *test, struct evt_options *opt,
2502bfe3f2eSlogwang int (*worker)(void *))
2512bfe3f2eSlogwang {
2522bfe3f2eSlogwang int ret, lcore_id;
2532bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
2542bfe3f2eSlogwang
2552bfe3f2eSlogwang int port_idx = 0;
2562bfe3f2eSlogwang /* launch workers */
257*2d9fd380Sjfb8856606 RTE_LCORE_FOREACH_WORKER(lcore_id) {
2582bfe3f2eSlogwang if (!(opt->wlcores[lcore_id]))
2592bfe3f2eSlogwang continue;
2602bfe3f2eSlogwang
2612bfe3f2eSlogwang ret = rte_eal_remote_launch(worker,
2622bfe3f2eSlogwang &t->worker[port_idx], lcore_id);
2632bfe3f2eSlogwang if (ret) {
2642bfe3f2eSlogwang evt_err("failed to launch worker %d", lcore_id);
2652bfe3f2eSlogwang return ret;
2662bfe3f2eSlogwang }
2672bfe3f2eSlogwang port_idx++;
2682bfe3f2eSlogwang }
2692bfe3f2eSlogwang
2702bfe3f2eSlogwang /* launch producers */
271*2d9fd380Sjfb8856606 RTE_LCORE_FOREACH_WORKER(lcore_id) {
2722bfe3f2eSlogwang if (!(opt->plcores[lcore_id]))
2732bfe3f2eSlogwang continue;
2742bfe3f2eSlogwang
275d30ea906Sjfb8856606 ret = rte_eal_remote_launch(perf_producer_wrapper,
276d30ea906Sjfb8856606 &t->prod[port_idx], lcore_id);
2772bfe3f2eSlogwang if (ret) {
2782bfe3f2eSlogwang evt_err("failed to launch perf_producer %d", lcore_id);
2792bfe3f2eSlogwang return ret;
2802bfe3f2eSlogwang }
2812bfe3f2eSlogwang port_idx++;
2822bfe3f2eSlogwang }
2832bfe3f2eSlogwang
284d30ea906Sjfb8856606 const uint64_t total_pkts = t->outstand_pkts;
2852bfe3f2eSlogwang
2862bfe3f2eSlogwang uint64_t dead_lock_cycles = rte_get_timer_cycles();
2872bfe3f2eSlogwang int64_t dead_lock_remaining = total_pkts;
2882bfe3f2eSlogwang const uint64_t dead_lock_sample = rte_get_timer_hz() * 5;
2892bfe3f2eSlogwang
2902bfe3f2eSlogwang uint64_t perf_cycles = rte_get_timer_cycles();
2912bfe3f2eSlogwang int64_t perf_remaining = total_pkts;
2922bfe3f2eSlogwang const uint64_t perf_sample = rte_get_timer_hz();
2932bfe3f2eSlogwang
2942bfe3f2eSlogwang static float total_mpps;
2952bfe3f2eSlogwang static uint64_t samples;
2962bfe3f2eSlogwang
2972bfe3f2eSlogwang const uint64_t freq_mhz = rte_get_timer_hz() / 1000000;
2982bfe3f2eSlogwang int64_t remaining = t->outstand_pkts - processed_pkts(t);
2992bfe3f2eSlogwang
3002bfe3f2eSlogwang while (t->done == false) {
3012bfe3f2eSlogwang const uint64_t new_cycles = rte_get_timer_cycles();
3022bfe3f2eSlogwang
3032bfe3f2eSlogwang if ((new_cycles - perf_cycles) > perf_sample) {
3042bfe3f2eSlogwang const uint64_t latency = total_latency(t);
3052bfe3f2eSlogwang const uint64_t pkts = processed_pkts(t);
3062bfe3f2eSlogwang
3072bfe3f2eSlogwang remaining = t->outstand_pkts - pkts;
3082bfe3f2eSlogwang float mpps = (float)(perf_remaining-remaining)/1000000;
3092bfe3f2eSlogwang
3102bfe3f2eSlogwang perf_remaining = remaining;
3112bfe3f2eSlogwang perf_cycles = new_cycles;
3122bfe3f2eSlogwang total_mpps += mpps;
3132bfe3f2eSlogwang ++samples;
3142bfe3f2eSlogwang if (opt->fwd_latency && pkts > 0) {
3152bfe3f2eSlogwang printf(CLGRN"\r%.3f mpps avg %.3f mpps [avg fwd latency %.3f us] "CLNRM,
3162bfe3f2eSlogwang mpps, total_mpps/samples,
3172bfe3f2eSlogwang (float)(latency/pkts)/freq_mhz);
3182bfe3f2eSlogwang } else {
3192bfe3f2eSlogwang printf(CLGRN"\r%.3f mpps avg %.3f mpps"CLNRM,
3202bfe3f2eSlogwang mpps, total_mpps/samples);
3212bfe3f2eSlogwang }
3222bfe3f2eSlogwang fflush(stdout);
3232bfe3f2eSlogwang
3242bfe3f2eSlogwang if (remaining <= 0) {
3255af785ecSfengbojiang(姜凤波) t->result = EVT_TEST_SUCCESS;
326d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_SYNT ||
327d30ea906Sjfb8856606 opt->prod_type ==
328d30ea906Sjfb8856606 EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
329d30ea906Sjfb8856606 t->done = true;
3302bfe3f2eSlogwang rte_smp_wmb();
3312bfe3f2eSlogwang break;
3322bfe3f2eSlogwang }
3332bfe3f2eSlogwang }
334d30ea906Sjfb8856606 }
3352bfe3f2eSlogwang
336d30ea906Sjfb8856606 if (new_cycles - dead_lock_cycles > dead_lock_sample &&
3371646932aSjfb8856606 (opt->prod_type == EVT_PROD_TYPE_SYNT ||
3381646932aSjfb8856606 opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR)) {
3392bfe3f2eSlogwang remaining = t->outstand_pkts - processed_pkts(t);
3402bfe3f2eSlogwang if (dead_lock_remaining == remaining) {
3412bfe3f2eSlogwang rte_event_dev_dump(opt->dev_id, stdout);
3422bfe3f2eSlogwang evt_err("No schedules for seconds, deadlock");
3432bfe3f2eSlogwang t->done = true;
3442bfe3f2eSlogwang rte_smp_wmb();
3452bfe3f2eSlogwang break;
3462bfe3f2eSlogwang }
3472bfe3f2eSlogwang dead_lock_remaining = remaining;
3482bfe3f2eSlogwang dead_lock_cycles = new_cycles;
3492bfe3f2eSlogwang }
3502bfe3f2eSlogwang }
3512bfe3f2eSlogwang printf("\n");
3522bfe3f2eSlogwang return 0;
3532bfe3f2eSlogwang }
3542bfe3f2eSlogwang
355d30ea906Sjfb8856606 static int
perf_event_rx_adapter_setup(struct evt_options * opt,uint8_t stride,struct rte_event_port_conf prod_conf)356d30ea906Sjfb8856606 perf_event_rx_adapter_setup(struct evt_options *opt, uint8_t stride,
357d30ea906Sjfb8856606 struct rte_event_port_conf prod_conf)
358d30ea906Sjfb8856606 {
359d30ea906Sjfb8856606 int ret = 0;
360d30ea906Sjfb8856606 uint16_t prod;
361d30ea906Sjfb8856606 struct rte_event_eth_rx_adapter_queue_conf queue_conf;
362d30ea906Sjfb8856606
363d30ea906Sjfb8856606 memset(&queue_conf, 0,
364d30ea906Sjfb8856606 sizeof(struct rte_event_eth_rx_adapter_queue_conf));
365d30ea906Sjfb8856606 queue_conf.ev.sched_type = opt->sched_type_list[0];
366d30ea906Sjfb8856606 RTE_ETH_FOREACH_DEV(prod) {
367d30ea906Sjfb8856606 uint32_t cap;
368d30ea906Sjfb8856606
369d30ea906Sjfb8856606 ret = rte_event_eth_rx_adapter_caps_get(opt->dev_id,
370d30ea906Sjfb8856606 prod, &cap);
371d30ea906Sjfb8856606 if (ret) {
372d30ea906Sjfb8856606 evt_err("failed to get event rx adapter[%d]"
373d30ea906Sjfb8856606 " capabilities",
374d30ea906Sjfb8856606 opt->dev_id);
375d30ea906Sjfb8856606 return ret;
376d30ea906Sjfb8856606 }
377d30ea906Sjfb8856606 queue_conf.ev.queue_id = prod * stride;
378d30ea906Sjfb8856606 ret = rte_event_eth_rx_adapter_create(prod, opt->dev_id,
379d30ea906Sjfb8856606 &prod_conf);
380d30ea906Sjfb8856606 if (ret) {
381d30ea906Sjfb8856606 evt_err("failed to create rx adapter[%d]", prod);
382d30ea906Sjfb8856606 return ret;
383d30ea906Sjfb8856606 }
384d30ea906Sjfb8856606 ret = rte_event_eth_rx_adapter_queue_add(prod, prod, -1,
385d30ea906Sjfb8856606 &queue_conf);
386d30ea906Sjfb8856606 if (ret) {
387d30ea906Sjfb8856606 evt_err("failed to add rx queues to adapter[%d]", prod);
388d30ea906Sjfb8856606 return ret;
389d30ea906Sjfb8856606 }
390d30ea906Sjfb8856606
391d30ea906Sjfb8856606 if (!(cap & RTE_EVENT_ETH_RX_ADAPTER_CAP_INTERNAL_PORT)) {
392d30ea906Sjfb8856606 uint32_t service_id;
393d30ea906Sjfb8856606
394d30ea906Sjfb8856606 rte_event_eth_rx_adapter_service_id_get(prod,
395d30ea906Sjfb8856606 &service_id);
396d30ea906Sjfb8856606 ret = evt_service_setup(service_id);
397d30ea906Sjfb8856606 if (ret) {
398d30ea906Sjfb8856606 evt_err("Failed to setup service core"
399d30ea906Sjfb8856606 " for Rx adapter\n");
400d30ea906Sjfb8856606 return ret;
401d30ea906Sjfb8856606 }
402d30ea906Sjfb8856606 }
403d30ea906Sjfb8856606 }
404d30ea906Sjfb8856606
405d30ea906Sjfb8856606 return ret;
406d30ea906Sjfb8856606 }
407d30ea906Sjfb8856606
408d30ea906Sjfb8856606 static int
perf_event_timer_adapter_setup(struct test_perf * t)409d30ea906Sjfb8856606 perf_event_timer_adapter_setup(struct test_perf *t)
410d30ea906Sjfb8856606 {
411d30ea906Sjfb8856606 int i;
412d30ea906Sjfb8856606 int ret;
413d30ea906Sjfb8856606 struct rte_event_timer_adapter_info adapter_info;
414d30ea906Sjfb8856606 struct rte_event_timer_adapter *wl;
415d30ea906Sjfb8856606 uint8_t nb_producers = evt_nr_active_lcores(t->opt->plcores);
416d30ea906Sjfb8856606 uint8_t flags = RTE_EVENT_TIMER_ADAPTER_F_ADJUST_RES;
417d30ea906Sjfb8856606
418d30ea906Sjfb8856606 if (nb_producers == 1)
419d30ea906Sjfb8856606 flags |= RTE_EVENT_TIMER_ADAPTER_F_SP_PUT;
420d30ea906Sjfb8856606
421d30ea906Sjfb8856606 for (i = 0; i < t->opt->nb_timer_adptrs; i++) {
422d30ea906Sjfb8856606 struct rte_event_timer_adapter_conf config = {
423d30ea906Sjfb8856606 .event_dev_id = t->opt->dev_id,
424d30ea906Sjfb8856606 .timer_adapter_id = i,
425d30ea906Sjfb8856606 .timer_tick_ns = t->opt->timer_tick_nsec,
426d30ea906Sjfb8856606 .max_tmo_ns = t->opt->max_tmo_nsec,
4274418919fSjohnjiang .nb_timers = t->opt->pool_sz,
428d30ea906Sjfb8856606 .flags = flags,
429d30ea906Sjfb8856606 };
430d30ea906Sjfb8856606
431d30ea906Sjfb8856606 wl = rte_event_timer_adapter_create(&config);
432d30ea906Sjfb8856606 if (wl == NULL) {
433d30ea906Sjfb8856606 evt_err("failed to create event timer ring %d", i);
434d30ea906Sjfb8856606 return rte_errno;
435d30ea906Sjfb8856606 }
436d30ea906Sjfb8856606
437d30ea906Sjfb8856606 memset(&adapter_info, 0,
438d30ea906Sjfb8856606 sizeof(struct rte_event_timer_adapter_info));
439d30ea906Sjfb8856606 rte_event_timer_adapter_get_info(wl, &adapter_info);
440d30ea906Sjfb8856606 t->opt->optm_timer_tick_nsec = adapter_info.min_resolution_ns;
441d30ea906Sjfb8856606
442d30ea906Sjfb8856606 if (!(adapter_info.caps &
443d30ea906Sjfb8856606 RTE_EVENT_TIMER_ADAPTER_CAP_INTERNAL_PORT)) {
4444418919fSjohnjiang uint32_t service_id = -1U;
445d30ea906Sjfb8856606
446d30ea906Sjfb8856606 rte_event_timer_adapter_service_id_get(wl,
447d30ea906Sjfb8856606 &service_id);
448d30ea906Sjfb8856606 ret = evt_service_setup(service_id);
449d30ea906Sjfb8856606 if (ret) {
450d30ea906Sjfb8856606 evt_err("Failed to setup service core"
451d30ea906Sjfb8856606 " for timer adapter\n");
452d30ea906Sjfb8856606 return ret;
453d30ea906Sjfb8856606 }
454d30ea906Sjfb8856606 rte_service_runstate_set(service_id, 1);
455d30ea906Sjfb8856606 }
456d30ea906Sjfb8856606 t->timer_adptr[i] = wl;
457d30ea906Sjfb8856606 }
458d30ea906Sjfb8856606 return 0;
459d30ea906Sjfb8856606 }
460d30ea906Sjfb8856606
4612bfe3f2eSlogwang int
perf_event_dev_port_setup(struct evt_test * test,struct evt_options * opt,uint8_t stride,uint8_t nb_queues,const struct rte_event_port_conf * port_conf)4622bfe3f2eSlogwang perf_event_dev_port_setup(struct evt_test *test, struct evt_options *opt,
463d30ea906Sjfb8856606 uint8_t stride, uint8_t nb_queues,
464d30ea906Sjfb8856606 const struct rte_event_port_conf *port_conf)
4652bfe3f2eSlogwang {
4662bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
467d30ea906Sjfb8856606 uint16_t port, prod;
4682bfe3f2eSlogwang int ret = -1;
4692bfe3f2eSlogwang
4702bfe3f2eSlogwang /* setup one port per worker, linking to all queues */
4712bfe3f2eSlogwang for (port = 0; port < evt_nr_active_lcores(opt->wlcores);
4722bfe3f2eSlogwang port++) {
4732bfe3f2eSlogwang struct worker_data *w = &t->worker[port];
4742bfe3f2eSlogwang
4752bfe3f2eSlogwang w->dev_id = opt->dev_id;
4762bfe3f2eSlogwang w->port_id = port;
4772bfe3f2eSlogwang w->t = t;
4782bfe3f2eSlogwang w->processed_pkts = 0;
4792bfe3f2eSlogwang w->latency = 0;
4802bfe3f2eSlogwang
481d30ea906Sjfb8856606 ret = rte_event_port_setup(opt->dev_id, port, port_conf);
4822bfe3f2eSlogwang if (ret) {
4832bfe3f2eSlogwang evt_err("failed to setup port %d", port);
4842bfe3f2eSlogwang return ret;
4852bfe3f2eSlogwang }
4862bfe3f2eSlogwang
4872bfe3f2eSlogwang ret = rte_event_port_link(opt->dev_id, port, NULL, NULL, 0);
4882bfe3f2eSlogwang if (ret != nb_queues) {
4892bfe3f2eSlogwang evt_err("failed to link all queues to port %d", port);
4902bfe3f2eSlogwang return -EINVAL;
4912bfe3f2eSlogwang }
4922bfe3f2eSlogwang }
4932bfe3f2eSlogwang
4942bfe3f2eSlogwang /* port for producers, no links */
495d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_ETH_RX_ADPTR) {
496d30ea906Sjfb8856606 for ( ; port < perf_nb_event_ports(opt); port++) {
497d30ea906Sjfb8856606 struct prod_data *p = &t->prod[port];
498d30ea906Sjfb8856606 p->t = t;
499d30ea906Sjfb8856606 }
500d30ea906Sjfb8856606
501d30ea906Sjfb8856606 ret = perf_event_rx_adapter_setup(opt, stride, *port_conf);
502d30ea906Sjfb8856606 if (ret)
503d30ea906Sjfb8856606 return ret;
504d30ea906Sjfb8856606 } else if (opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
505d30ea906Sjfb8856606 prod = 0;
506d30ea906Sjfb8856606 for ( ; port < perf_nb_event_ports(opt); port++) {
507d30ea906Sjfb8856606 struct prod_data *p = &t->prod[port];
508d30ea906Sjfb8856606 p->queue_id = prod * stride;
509d30ea906Sjfb8856606 p->t = t;
510d30ea906Sjfb8856606 prod++;
511d30ea906Sjfb8856606 }
512d30ea906Sjfb8856606
513d30ea906Sjfb8856606 ret = perf_event_timer_adapter_setup(t);
514d30ea906Sjfb8856606 if (ret)
515d30ea906Sjfb8856606 return ret;
516d30ea906Sjfb8856606 } else {
5172bfe3f2eSlogwang prod = 0;
5182bfe3f2eSlogwang for ( ; port < perf_nb_event_ports(opt); port++) {
5192bfe3f2eSlogwang struct prod_data *p = &t->prod[port];
5202bfe3f2eSlogwang
5212bfe3f2eSlogwang p->dev_id = opt->dev_id;
5222bfe3f2eSlogwang p->port_id = port;
5232bfe3f2eSlogwang p->queue_id = prod * stride;
5242bfe3f2eSlogwang p->t = t;
5252bfe3f2eSlogwang
526d30ea906Sjfb8856606 ret = rte_event_port_setup(opt->dev_id, port,
527d30ea906Sjfb8856606 port_conf);
5282bfe3f2eSlogwang if (ret) {
5292bfe3f2eSlogwang evt_err("failed to setup port %d", port);
5302bfe3f2eSlogwang return ret;
5312bfe3f2eSlogwang }
5322bfe3f2eSlogwang prod++;
5332bfe3f2eSlogwang }
534d30ea906Sjfb8856606 }
5352bfe3f2eSlogwang
5362bfe3f2eSlogwang return ret;
5372bfe3f2eSlogwang }
5382bfe3f2eSlogwang
5392bfe3f2eSlogwang int
perf_opt_check(struct evt_options * opt,uint64_t nb_queues)5402bfe3f2eSlogwang perf_opt_check(struct evt_options *opt, uint64_t nb_queues)
5412bfe3f2eSlogwang {
5422bfe3f2eSlogwang unsigned int lcores;
5432bfe3f2eSlogwang
544*2d9fd380Sjfb8856606 /* N producer + N worker + main when producer cores are used
545*2d9fd380Sjfb8856606 * Else N worker + main when Rx adapter is used
546d30ea906Sjfb8856606 */
547d30ea906Sjfb8856606 lcores = opt->prod_type == EVT_PROD_TYPE_SYNT ? 3 : 2;
5482bfe3f2eSlogwang
5492bfe3f2eSlogwang if (rte_lcore_count() < lcores) {
5502bfe3f2eSlogwang evt_err("test need minimum %d lcores", lcores);
5512bfe3f2eSlogwang return -1;
5522bfe3f2eSlogwang }
5532bfe3f2eSlogwang
5542bfe3f2eSlogwang /* Validate worker lcores */
555*2d9fd380Sjfb8856606 if (evt_lcores_has_overlap(opt->wlcores, rte_get_main_lcore())) {
556*2d9fd380Sjfb8856606 evt_err("worker lcores overlaps with main lcore");
5572bfe3f2eSlogwang return -1;
5582bfe3f2eSlogwang }
5592bfe3f2eSlogwang if (evt_lcores_has_overlap_multi(opt->wlcores, opt->plcores)) {
5602bfe3f2eSlogwang evt_err("worker lcores overlaps producer lcores");
5612bfe3f2eSlogwang return -1;
5622bfe3f2eSlogwang }
5632bfe3f2eSlogwang if (evt_has_disabled_lcore(opt->wlcores)) {
5642bfe3f2eSlogwang evt_err("one or more workers lcores are not enabled");
5652bfe3f2eSlogwang return -1;
5662bfe3f2eSlogwang }
5672bfe3f2eSlogwang if (!evt_has_active_lcore(opt->wlcores)) {
5682bfe3f2eSlogwang evt_err("minimum one worker is required");
5692bfe3f2eSlogwang return -1;
5702bfe3f2eSlogwang }
5712bfe3f2eSlogwang
5724b05018fSfengbojiang if (opt->prod_type == EVT_PROD_TYPE_SYNT ||
5734b05018fSfengbojiang opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
5742bfe3f2eSlogwang /* Validate producer lcores */
575d30ea906Sjfb8856606 if (evt_lcores_has_overlap(opt->plcores,
576*2d9fd380Sjfb8856606 rte_get_main_lcore())) {
577*2d9fd380Sjfb8856606 evt_err("producer lcores overlaps with main lcore");
5782bfe3f2eSlogwang return -1;
5792bfe3f2eSlogwang }
5802bfe3f2eSlogwang if (evt_has_disabled_lcore(opt->plcores)) {
5812bfe3f2eSlogwang evt_err("one or more producer lcores are not enabled");
5822bfe3f2eSlogwang return -1;
5832bfe3f2eSlogwang }
5842bfe3f2eSlogwang if (!evt_has_active_lcore(opt->plcores)) {
5852bfe3f2eSlogwang evt_err("minimum one producer is required");
5862bfe3f2eSlogwang return -1;
5872bfe3f2eSlogwang }
588d30ea906Sjfb8856606 }
5892bfe3f2eSlogwang
5902bfe3f2eSlogwang if (evt_has_invalid_stage(opt))
5912bfe3f2eSlogwang return -1;
5922bfe3f2eSlogwang
5932bfe3f2eSlogwang if (evt_has_invalid_sched_type(opt))
5942bfe3f2eSlogwang return -1;
5952bfe3f2eSlogwang
5962bfe3f2eSlogwang if (nb_queues > EVT_MAX_QUEUES) {
5972bfe3f2eSlogwang evt_err("number of queues exceeds %d", EVT_MAX_QUEUES);
5982bfe3f2eSlogwang return -1;
5992bfe3f2eSlogwang }
6002bfe3f2eSlogwang if (perf_nb_event_ports(opt) > EVT_MAX_PORTS) {
6012bfe3f2eSlogwang evt_err("number of ports exceeds %d", EVT_MAX_PORTS);
6022bfe3f2eSlogwang return -1;
6032bfe3f2eSlogwang }
6042bfe3f2eSlogwang
6052bfe3f2eSlogwang /* Fixups */
606d30ea906Sjfb8856606 if ((opt->nb_stages == 1 &&
607d30ea906Sjfb8856606 opt->prod_type != EVT_PROD_TYPE_EVENT_TIMER_ADPTR) &&
608d30ea906Sjfb8856606 opt->fwd_latency) {
6092bfe3f2eSlogwang evt_info("fwd_latency is valid when nb_stages > 1, disabling");
6102bfe3f2eSlogwang opt->fwd_latency = 0;
6112bfe3f2eSlogwang }
612d30ea906Sjfb8856606
6132bfe3f2eSlogwang if (opt->fwd_latency && !opt->q_priority) {
6142bfe3f2eSlogwang evt_info("enabled queue priority for latency measurement");
6152bfe3f2eSlogwang opt->q_priority = 1;
6162bfe3f2eSlogwang }
6172bfe3f2eSlogwang if (opt->nb_pkts == 0)
6182bfe3f2eSlogwang opt->nb_pkts = INT64_MAX/evt_nr_active_lcores(opt->plcores);
6192bfe3f2eSlogwang
6202bfe3f2eSlogwang return 0;
6212bfe3f2eSlogwang }
6222bfe3f2eSlogwang
6232bfe3f2eSlogwang void
perf_opt_dump(struct evt_options * opt,uint8_t nb_queues)6242bfe3f2eSlogwang perf_opt_dump(struct evt_options *opt, uint8_t nb_queues)
6252bfe3f2eSlogwang {
6262bfe3f2eSlogwang evt_dump("nb_prod_lcores", "%d", evt_nr_active_lcores(opt->plcores));
6272bfe3f2eSlogwang evt_dump_producer_lcores(opt);
6282bfe3f2eSlogwang evt_dump("nb_worker_lcores", "%d", evt_nr_active_lcores(opt->wlcores));
6292bfe3f2eSlogwang evt_dump_worker_lcores(opt);
6302bfe3f2eSlogwang evt_dump_nb_stages(opt);
6312bfe3f2eSlogwang evt_dump("nb_evdev_ports", "%d", perf_nb_event_ports(opt));
6322bfe3f2eSlogwang evt_dump("nb_evdev_queues", "%d", nb_queues);
6332bfe3f2eSlogwang evt_dump_queue_priority(opt);
6342bfe3f2eSlogwang evt_dump_sched_type_list(opt);
635d30ea906Sjfb8856606 evt_dump_producer_type(opt);
6362bfe3f2eSlogwang }
6372bfe3f2eSlogwang
6382bfe3f2eSlogwang void
perf_eventdev_destroy(struct evt_test * test,struct evt_options * opt)6392bfe3f2eSlogwang perf_eventdev_destroy(struct evt_test *test, struct evt_options *opt)
6402bfe3f2eSlogwang {
641d30ea906Sjfb8856606 int i;
642d30ea906Sjfb8856606 struct test_perf *t = evt_test_priv(test);
6432bfe3f2eSlogwang
644d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
645d30ea906Sjfb8856606 for (i = 0; i < opt->nb_timer_adptrs; i++)
646d30ea906Sjfb8856606 rte_event_timer_adapter_stop(t->timer_adptr[i]);
647d30ea906Sjfb8856606 }
6482bfe3f2eSlogwang rte_event_dev_stop(opt->dev_id);
6492bfe3f2eSlogwang rte_event_dev_close(opt->dev_id);
6502bfe3f2eSlogwang }
6512bfe3f2eSlogwang
6522bfe3f2eSlogwang static inline void
perf_elt_init(struct rte_mempool * mp,void * arg __rte_unused,void * obj,unsigned i __rte_unused)6532bfe3f2eSlogwang perf_elt_init(struct rte_mempool *mp, void *arg __rte_unused,
6542bfe3f2eSlogwang void *obj, unsigned i __rte_unused)
6552bfe3f2eSlogwang {
6562bfe3f2eSlogwang memset(obj, 0, mp->elt_size);
6572bfe3f2eSlogwang }
6582bfe3f2eSlogwang
659d30ea906Sjfb8856606 #define NB_RX_DESC 128
660d30ea906Sjfb8856606 #define NB_TX_DESC 512
661d30ea906Sjfb8856606 int
perf_ethdev_setup(struct evt_test * test,struct evt_options * opt)662d30ea906Sjfb8856606 perf_ethdev_setup(struct evt_test *test, struct evt_options *opt)
663d30ea906Sjfb8856606 {
664d30ea906Sjfb8856606 uint16_t i;
6654418919fSjohnjiang int ret;
666d30ea906Sjfb8856606 struct test_perf *t = evt_test_priv(test);
667d30ea906Sjfb8856606 struct rte_eth_conf port_conf = {
668d30ea906Sjfb8856606 .rxmode = {
669d30ea906Sjfb8856606 .mq_mode = ETH_MQ_RX_RSS,
6704418919fSjohnjiang .max_rx_pkt_len = RTE_ETHER_MAX_LEN,
671d30ea906Sjfb8856606 .split_hdr_size = 0,
672d30ea906Sjfb8856606 },
673d30ea906Sjfb8856606 .rx_adv_conf = {
674d30ea906Sjfb8856606 .rss_conf = {
675d30ea906Sjfb8856606 .rss_key = NULL,
676d30ea906Sjfb8856606 .rss_hf = ETH_RSS_IP,
677d30ea906Sjfb8856606 },
678d30ea906Sjfb8856606 },
679d30ea906Sjfb8856606 };
680d30ea906Sjfb8856606
681d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_SYNT ||
682d30ea906Sjfb8856606 opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR)
683d30ea906Sjfb8856606 return 0;
684d30ea906Sjfb8856606
685d30ea906Sjfb8856606 if (!rte_eth_dev_count_avail()) {
686d30ea906Sjfb8856606 evt_err("No ethernet ports found.");
687d30ea906Sjfb8856606 return -ENODEV;
688d30ea906Sjfb8856606 }
689d30ea906Sjfb8856606
690d30ea906Sjfb8856606 RTE_ETH_FOREACH_DEV(i) {
691d30ea906Sjfb8856606 struct rte_eth_dev_info dev_info;
692d30ea906Sjfb8856606 struct rte_eth_conf local_port_conf = port_conf;
693d30ea906Sjfb8856606
6944418919fSjohnjiang ret = rte_eth_dev_info_get(i, &dev_info);
6954418919fSjohnjiang if (ret != 0) {
6964418919fSjohnjiang evt_err("Error during getting device (port %u) info: %s\n",
6974418919fSjohnjiang i, strerror(-ret));
6984418919fSjohnjiang return ret;
6994418919fSjohnjiang }
700d30ea906Sjfb8856606
701d30ea906Sjfb8856606 local_port_conf.rx_adv_conf.rss_conf.rss_hf &=
702d30ea906Sjfb8856606 dev_info.flow_type_rss_offloads;
703d30ea906Sjfb8856606 if (local_port_conf.rx_adv_conf.rss_conf.rss_hf !=
704d30ea906Sjfb8856606 port_conf.rx_adv_conf.rss_conf.rss_hf) {
705d30ea906Sjfb8856606 evt_info("Port %u modified RSS hash function based on hardware support,"
706d30ea906Sjfb8856606 "requested:%#"PRIx64" configured:%#"PRIx64"\n",
707d30ea906Sjfb8856606 i,
708d30ea906Sjfb8856606 port_conf.rx_adv_conf.rss_conf.rss_hf,
709d30ea906Sjfb8856606 local_port_conf.rx_adv_conf.rss_conf.rss_hf);
710d30ea906Sjfb8856606 }
711d30ea906Sjfb8856606
712d30ea906Sjfb8856606 if (rte_eth_dev_configure(i, 1, 1, &local_port_conf) < 0) {
713d30ea906Sjfb8856606 evt_err("Failed to configure eth port [%d]", i);
714d30ea906Sjfb8856606 return -EINVAL;
715d30ea906Sjfb8856606 }
716d30ea906Sjfb8856606
717d30ea906Sjfb8856606 if (rte_eth_rx_queue_setup(i, 0, NB_RX_DESC,
718d30ea906Sjfb8856606 rte_socket_id(), NULL, t->pool) < 0) {
719d30ea906Sjfb8856606 evt_err("Failed to setup eth port [%d] rx_queue: %d.",
720d30ea906Sjfb8856606 i, 0);
721d30ea906Sjfb8856606 return -EINVAL;
722d30ea906Sjfb8856606 }
723d30ea906Sjfb8856606
724d30ea906Sjfb8856606 if (rte_eth_tx_queue_setup(i, 0, NB_TX_DESC,
725d30ea906Sjfb8856606 rte_socket_id(), NULL) < 0) {
726d30ea906Sjfb8856606 evt_err("Failed to setup eth port [%d] tx_queue: %d.",
727d30ea906Sjfb8856606 i, 0);
728d30ea906Sjfb8856606 return -EINVAL;
729d30ea906Sjfb8856606 }
730d30ea906Sjfb8856606
7314418919fSjohnjiang ret = rte_eth_promiscuous_enable(i);
7324418919fSjohnjiang if (ret != 0) {
7334418919fSjohnjiang evt_err("Failed to enable promiscuous mode for eth port [%d]: %s",
7344418919fSjohnjiang i, rte_strerror(-ret));
7354418919fSjohnjiang return ret;
7364418919fSjohnjiang }
737d30ea906Sjfb8856606 }
738d30ea906Sjfb8856606
739d30ea906Sjfb8856606 return 0;
740d30ea906Sjfb8856606 }
741d30ea906Sjfb8856606
perf_ethdev_destroy(struct evt_test * test,struct evt_options * opt)742d30ea906Sjfb8856606 void perf_ethdev_destroy(struct evt_test *test, struct evt_options *opt)
743d30ea906Sjfb8856606 {
744d30ea906Sjfb8856606 uint16_t i;
745d30ea906Sjfb8856606 RTE_SET_USED(test);
746d30ea906Sjfb8856606
747d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_ETH_RX_ADPTR) {
748d30ea906Sjfb8856606 RTE_ETH_FOREACH_DEV(i) {
749d30ea906Sjfb8856606 rte_event_eth_rx_adapter_stop(i);
750d30ea906Sjfb8856606 rte_eth_dev_stop(i);
751d30ea906Sjfb8856606 }
752d30ea906Sjfb8856606 }
753d30ea906Sjfb8856606 }
754d30ea906Sjfb8856606
7552bfe3f2eSlogwang int
perf_mempool_setup(struct evt_test * test,struct evt_options * opt)7562bfe3f2eSlogwang perf_mempool_setup(struct evt_test *test, struct evt_options *opt)
7572bfe3f2eSlogwang {
7582bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
7592bfe3f2eSlogwang
760d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_SYNT ||
761d30ea906Sjfb8856606 opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
7622bfe3f2eSlogwang t->pool = rte_mempool_create(test->name, /* mempool name */
7632bfe3f2eSlogwang opt->pool_sz, /* number of elements*/
7642bfe3f2eSlogwang sizeof(struct perf_elt), /* element size*/
7652bfe3f2eSlogwang 512, /* cache size*/
7662bfe3f2eSlogwang 0, NULL, NULL,
7672bfe3f2eSlogwang perf_elt_init, /* obj constructor */
7682bfe3f2eSlogwang NULL, opt->socket_id, 0); /* flags */
769d30ea906Sjfb8856606 } else {
770d30ea906Sjfb8856606 t->pool = rte_pktmbuf_pool_create(test->name, /* mempool name */
771d30ea906Sjfb8856606 opt->pool_sz, /* number of elements*/
772d30ea906Sjfb8856606 512, /* cache size*/
773d30ea906Sjfb8856606 0,
774d30ea906Sjfb8856606 RTE_MBUF_DEFAULT_BUF_SIZE,
775d30ea906Sjfb8856606 opt->socket_id); /* flags */
776d30ea906Sjfb8856606
777d30ea906Sjfb8856606 }
778d30ea906Sjfb8856606
7792bfe3f2eSlogwang if (t->pool == NULL) {
7802bfe3f2eSlogwang evt_err("failed to create mempool");
7812bfe3f2eSlogwang return -ENOMEM;
7822bfe3f2eSlogwang }
7832bfe3f2eSlogwang
7842bfe3f2eSlogwang return 0;
7852bfe3f2eSlogwang }
7862bfe3f2eSlogwang
7872bfe3f2eSlogwang void
perf_mempool_destroy(struct evt_test * test,struct evt_options * opt)7882bfe3f2eSlogwang perf_mempool_destroy(struct evt_test *test, struct evt_options *opt)
7892bfe3f2eSlogwang {
7902bfe3f2eSlogwang RTE_SET_USED(opt);
7912bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
7922bfe3f2eSlogwang
7932bfe3f2eSlogwang rte_mempool_free(t->pool);
7942bfe3f2eSlogwang }
7952bfe3f2eSlogwang
7962bfe3f2eSlogwang int
perf_test_setup(struct evt_test * test,struct evt_options * opt)7972bfe3f2eSlogwang perf_test_setup(struct evt_test *test, struct evt_options *opt)
7982bfe3f2eSlogwang {
7992bfe3f2eSlogwang void *test_perf;
8002bfe3f2eSlogwang
8012bfe3f2eSlogwang test_perf = rte_zmalloc_socket(test->name, sizeof(struct test_perf),
8022bfe3f2eSlogwang RTE_CACHE_LINE_SIZE, opt->socket_id);
8032bfe3f2eSlogwang if (test_perf == NULL) {
8042bfe3f2eSlogwang evt_err("failed to allocate test_perf memory");
8052bfe3f2eSlogwang goto nomem;
8062bfe3f2eSlogwang }
8072bfe3f2eSlogwang test->test_priv = test_perf;
8082bfe3f2eSlogwang
8092bfe3f2eSlogwang struct test_perf *t = evt_test_priv(test);
8102bfe3f2eSlogwang
811d30ea906Sjfb8856606 if (opt->prod_type == EVT_PROD_TYPE_EVENT_TIMER_ADPTR) {
812d30ea906Sjfb8856606 t->outstand_pkts = opt->nb_timers *
813d30ea906Sjfb8856606 evt_nr_active_lcores(opt->plcores);
814d30ea906Sjfb8856606 t->nb_pkts = opt->nb_timers;
815d30ea906Sjfb8856606 } else {
816d30ea906Sjfb8856606 t->outstand_pkts = opt->nb_pkts *
817d30ea906Sjfb8856606 evt_nr_active_lcores(opt->plcores);
818d30ea906Sjfb8856606 t->nb_pkts = opt->nb_pkts;
819d30ea906Sjfb8856606 }
820d30ea906Sjfb8856606
8212bfe3f2eSlogwang t->nb_workers = evt_nr_active_lcores(opt->wlcores);
8222bfe3f2eSlogwang t->done = false;
8232bfe3f2eSlogwang t->nb_flows = opt->nb_flows;
8242bfe3f2eSlogwang t->result = EVT_TEST_FAILED;
8252bfe3f2eSlogwang t->opt = opt;
8262bfe3f2eSlogwang memcpy(t->sched_type_list, opt->sched_type_list,
8272bfe3f2eSlogwang sizeof(opt->sched_type_list));
8282bfe3f2eSlogwang return 0;
8292bfe3f2eSlogwang nomem:
8302bfe3f2eSlogwang return -ENOMEM;
8312bfe3f2eSlogwang }
8322bfe3f2eSlogwang
8332bfe3f2eSlogwang void
perf_test_destroy(struct evt_test * test,struct evt_options * opt)8342bfe3f2eSlogwang perf_test_destroy(struct evt_test *test, struct evt_options *opt)
8352bfe3f2eSlogwang {
8362bfe3f2eSlogwang RTE_SET_USED(opt);
8372bfe3f2eSlogwang
8382bfe3f2eSlogwang rte_free(test->test_priv);
8392bfe3f2eSlogwang }
840