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