xref: /dpdk/app/test/test_pdcp.c (revision d010725b397429a56f445ef0b9523e7f26aa4603)
1bd24488cSAnoob Joseph /* SPDX-License-Identifier: BSD-3-Clause
2bd24488cSAnoob Joseph  * Copyright(C) 2023 Marvell.
3bd24488cSAnoob Joseph  */
4bd24488cSAnoob Joseph 
50eeeb7d0SVolodymyr Fialko #include <rte_bitmap.h>
6bd24488cSAnoob Joseph #include <rte_errno.h>
76f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
86f004629SVolodymyr Fialko #include <rte_eventdev.h>
96f004629SVolodymyr Fialko #include <rte_event_timer_adapter.h>
106f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
11bd24488cSAnoob Joseph #include <rte_malloc.h>
12bd24488cSAnoob Joseph #include <rte_pdcp.h>
13bd24488cSAnoob Joseph #include <rte_pdcp_hdr.h>
146f004629SVolodymyr Fialko #include <rte_timer.h>
15bd24488cSAnoob Joseph 
16bd24488cSAnoob Joseph #include "test.h"
17bd24488cSAnoob Joseph #include "test_cryptodev.h"
1864b12db3SAakash Sasidharan #include "test_cryptodev_security_pdcp_sdap_test_vectors.h"
19bd24488cSAnoob Joseph #include "test_cryptodev_security_pdcp_test_vectors.h"
20bd24488cSAnoob Joseph 
216f004629SVolodymyr Fialko #define NSECPERSEC 1E9
22bd24488cSAnoob Joseph #define NB_DESC 1024
236f004629SVolodymyr Fialko #define TIMER_ADAPTER_ID 0
246f004629SVolodymyr Fialko #define TEST_EV_QUEUE_ID 0
256f004629SVolodymyr Fialko #define TEST_EV_PORT_ID 0
26bd24488cSAnoob Joseph #define CDEV_INVALID_ID UINT8_MAX
2764b12db3SAakash Sasidharan #define NB_BASIC_TESTS RTE_DIM(pdcp_test_params)
2864b12db3SAakash Sasidharan #define NB_SDAP_TESTS RTE_DIM(list_pdcp_sdap_tests)
29bd24488cSAnoob Joseph #define PDCP_IV_LEN 16
30*d010725bSAakash Sasidharan #define PDCP_MBUF_SIZE	(sizeof(struct rte_mbuf) + \
31*d010725bSAakash Sasidharan 			 RTE_PKTMBUF_HEADROOM + RTE_PDCP_CTRL_PDU_SIZE_MAX)
32bd24488cSAnoob Joseph 
33d84ac1b9SVolodymyr Fialko /* Assert that condition is true, or goto the mark */
34d84ac1b9SVolodymyr Fialko #define ASSERT_TRUE_OR_GOTO(cond, mark, ...) do {\
35d84ac1b9SVolodymyr Fialko 	if (!(cond)) { \
36d84ac1b9SVolodymyr Fialko 		RTE_LOG(ERR, USER1, "Error at: %s:%d\n", __func__, __LINE__); \
37d84ac1b9SVolodymyr Fialko 		RTE_LOG(ERR, USER1, __VA_ARGS__); \
38d84ac1b9SVolodymyr Fialko 		goto mark; \
39d84ac1b9SVolodymyr Fialko 	} \
40d84ac1b9SVolodymyr Fialko } while (0)
41d84ac1b9SVolodymyr Fialko 
42e98a7ac0SVolodymyr Fialko /* According to formula(7.2.a Window_Size) */
43e98a7ac0SVolodymyr Fialko #define PDCP_WINDOW_SIZE(sn_size) (1 << (sn_size - 1))
44e98a7ac0SVolodymyr Fialko 
45bd24488cSAnoob Joseph struct pdcp_testsuite_params {
46bd24488cSAnoob Joseph 	struct rte_mempool *mbuf_pool;
47bd24488cSAnoob Joseph 	struct rte_mempool *cop_pool;
48bd24488cSAnoob Joseph 	struct rte_mempool *sess_pool;
49bd24488cSAnoob Joseph 	bool cdevs_used[RTE_CRYPTO_MAX_DEVS];
506f004629SVolodymyr Fialko 	int evdev;
516f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
526f004629SVolodymyr Fialko 	struct rte_event_timer_adapter *timdev;
536f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
546f004629SVolodymyr Fialko 	bool timer_is_running;
556f004629SVolodymyr Fialko 	uint64_t min_resolution_ns;
560eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *status_report;
570eeeb7d0SVolodymyr Fialko 	uint32_t status_report_bitmask_capacity;
580eeeb7d0SVolodymyr Fialko 	uint8_t *ctrl_pdu_buf;
59bd24488cSAnoob Joseph };
60bd24488cSAnoob Joseph 
61bd24488cSAnoob Joseph static struct pdcp_testsuite_params testsuite_params;
62bd24488cSAnoob Joseph 
636f004629SVolodymyr Fialko struct test_rte_timer_args {
646f004629SVolodymyr Fialko 	int status;
656f004629SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
666f004629SVolodymyr Fialko };
676f004629SVolodymyr Fialko 
68bd24488cSAnoob Joseph struct pdcp_test_conf {
69bd24488cSAnoob Joseph 	struct rte_pdcp_entity_conf entity;
70bd24488cSAnoob Joseph 	struct rte_crypto_sym_xform c_xfrm;
71bd24488cSAnoob Joseph 	struct rte_crypto_sym_xform a_xfrm;
72bd24488cSAnoob Joseph 	bool is_integrity_protected;
73bd24488cSAnoob Joseph 	uint8_t input[RTE_PDCP_CTRL_PDU_SIZE_MAX];
74bd24488cSAnoob Joseph 	uint32_t input_len;
75bd24488cSAnoob Joseph 	uint8_t output[RTE_PDCP_CTRL_PDU_SIZE_MAX];
76bd24488cSAnoob Joseph 	uint32_t output_len;
77bd24488cSAnoob Joseph };
78bd24488cSAnoob Joseph 
7964b12db3SAakash Sasidharan enum pdcp_test_suite_type {
8064b12db3SAakash Sasidharan 	PDCP_TEST_SUITE_TY_BASIC,
8164b12db3SAakash Sasidharan 	PDCP_TEST_SUITE_TY_SDAP,
8264b12db3SAakash Sasidharan };
8364b12db3SAakash Sasidharan 
84*d010725bSAakash Sasidharan static bool silent;
85*d010725bSAakash Sasidharan 
8664b12db3SAakash Sasidharan static int create_test_conf_from_index(const int index, struct pdcp_test_conf *conf,
8764b12db3SAakash Sasidharan 				       enum pdcp_test_suite_type suite_type);
88*d010725bSAakash Sasidharan static void test_conf_input_data_modify(struct pdcp_test_conf *conf, int inp_len);
89e98a7ac0SVolodymyr Fialko 
90e98a7ac0SVolodymyr Fialko typedef int (*test_with_conf_t)(struct pdcp_test_conf *conf);
91e98a7ac0SVolodymyr Fialko 
9264b12db3SAakash Sasidharan static uint32_t
nb_tests_get(enum pdcp_test_suite_type type)9364b12db3SAakash Sasidharan nb_tests_get(enum pdcp_test_suite_type type)
9464b12db3SAakash Sasidharan {
9564b12db3SAakash Sasidharan 	uint32_t ret;
9664b12db3SAakash Sasidharan 
9764b12db3SAakash Sasidharan 	switch (type) {
9864b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
9964b12db3SAakash Sasidharan 		ret = NB_BASIC_TESTS;
10064b12db3SAakash Sasidharan 		break;
10164b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
10264b12db3SAakash Sasidharan 		ret = NB_SDAP_TESTS;
10364b12db3SAakash Sasidharan 		break;
10464b12db3SAakash Sasidharan 	default:
10564b12db3SAakash Sasidharan 		return 0;
10664b12db3SAakash Sasidharan 	}
10764b12db3SAakash Sasidharan 
10864b12db3SAakash Sasidharan 	return ret;
10964b12db3SAakash Sasidharan }
11064b12db3SAakash Sasidharan 
11164b12db3SAakash Sasidharan static const char*
pdcp_test_name_get(enum pdcp_test_suite_type type,int idx)11264b12db3SAakash Sasidharan pdcp_test_name_get(enum pdcp_test_suite_type type, int idx)
11364b12db3SAakash Sasidharan {
11464b12db3SAakash Sasidharan 	const char *test_name = NULL;
11564b12db3SAakash Sasidharan 
11664b12db3SAakash Sasidharan 	switch (type) {
11764b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
11864b12db3SAakash Sasidharan 		test_name = pdcp_test_params[idx].name;
11964b12db3SAakash Sasidharan 		break;
12064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
12164b12db3SAakash Sasidharan 		test_name = list_pdcp_sdap_tests[idx].param.name;
12264b12db3SAakash Sasidharan 		break;
12364b12db3SAakash Sasidharan 	default:
12464b12db3SAakash Sasidharan 		return NULL;
12564b12db3SAakash Sasidharan 	}
12664b12db3SAakash Sasidharan 
12764b12db3SAakash Sasidharan 	return test_name;
12864b12db3SAakash Sasidharan }
12964b12db3SAakash Sasidharan 
130e98a7ac0SVolodymyr Fialko static int
run_test_foreach_known_vec(test_with_conf_t test,bool stop_on_first_pass,enum pdcp_test_suite_type suite_type)13164b12db3SAakash Sasidharan run_test_foreach_known_vec(test_with_conf_t test, bool stop_on_first_pass,
13264b12db3SAakash Sasidharan 			   enum pdcp_test_suite_type suite_type)
133e98a7ac0SVolodymyr Fialko {
134e98a7ac0SVolodymyr Fialko 	struct pdcp_test_conf test_conf;
135e98a7ac0SVolodymyr Fialko 	bool all_tests_skipped = true;
13664b12db3SAakash Sasidharan 	uint32_t nb_tests = nb_tests_get(suite_type);
137e98a7ac0SVolodymyr Fialko 	uint32_t i;
138e98a7ac0SVolodymyr Fialko 	int ret;
139e98a7ac0SVolodymyr Fialko 
14064b12db3SAakash Sasidharan 	for (i = 0; i < nb_tests; i++) {
14164b12db3SAakash Sasidharan 		create_test_conf_from_index(i, &test_conf, suite_type);
142e98a7ac0SVolodymyr Fialko 		ret = test(&test_conf);
143e98a7ac0SVolodymyr Fialko 
144e98a7ac0SVolodymyr Fialko 		if (ret == TEST_FAILED) {
14564b12db3SAakash Sasidharan 			printf("[%03i] - %s - failed\n", i,
14664b12db3SAakash Sasidharan 			       pdcp_test_name_get(suite_type, i));
147e98a7ac0SVolodymyr Fialko 			return TEST_FAILED;
148e98a7ac0SVolodymyr Fialko 		}
149e98a7ac0SVolodymyr Fialko 
150e98a7ac0SVolodymyr Fialko 		if ((ret == TEST_SKIPPED) || (ret == -ENOTSUP))
151e98a7ac0SVolodymyr Fialko 			continue;
152e98a7ac0SVolodymyr Fialko 
153e98a7ac0SVolodymyr Fialko 		if (stop_on_first_pass)
154e98a7ac0SVolodymyr Fialko 			return TEST_SUCCESS;
155e98a7ac0SVolodymyr Fialko 
156e98a7ac0SVolodymyr Fialko 		all_tests_skipped = false;
157e98a7ac0SVolodymyr Fialko 	}
158e98a7ac0SVolodymyr Fialko 
159e98a7ac0SVolodymyr Fialko 	if (all_tests_skipped)
160e98a7ac0SVolodymyr Fialko 		return TEST_SKIPPED;
161e98a7ac0SVolodymyr Fialko 
162e98a7ac0SVolodymyr Fialko 	return TEST_SUCCESS;
163e98a7ac0SVolodymyr Fialko }
164e98a7ac0SVolodymyr Fialko 
165e98a7ac0SVolodymyr Fialko static int
run_test_with_all_known_vec(const void * args)166e98a7ac0SVolodymyr Fialko run_test_with_all_known_vec(const void *args)
167e98a7ac0SVolodymyr Fialko {
168e98a7ac0SVolodymyr Fialko 	test_with_conf_t test = args;
169e98a7ac0SVolodymyr Fialko 
17064b12db3SAakash Sasidharan 	return run_test_foreach_known_vec(test, false,
17164b12db3SAakash Sasidharan 					  PDCP_TEST_SUITE_TY_BASIC);
17264b12db3SAakash Sasidharan }
17364b12db3SAakash Sasidharan 
17464b12db3SAakash Sasidharan static int
run_test_with_all_sdap_known_vec(const void * args)17564b12db3SAakash Sasidharan run_test_with_all_sdap_known_vec(const void *args)
17664b12db3SAakash Sasidharan {
17764b12db3SAakash Sasidharan 	test_with_conf_t test = args;
17864b12db3SAakash Sasidharan 
17964b12db3SAakash Sasidharan 	return run_test_foreach_known_vec(test, false,
18064b12db3SAakash Sasidharan 					  PDCP_TEST_SUITE_TY_SDAP);
181e98a7ac0SVolodymyr Fialko }
182e98a7ac0SVolodymyr Fialko 
183d84ac1b9SVolodymyr Fialko static int
run_test_with_all_known_vec_until_first_pass(const void * args)184d84ac1b9SVolodymyr Fialko run_test_with_all_known_vec_until_first_pass(const void *args)
185d84ac1b9SVolodymyr Fialko {
186d84ac1b9SVolodymyr Fialko 	test_with_conf_t test = args;
187d84ac1b9SVolodymyr Fialko 
18864b12db3SAakash Sasidharan 	return run_test_foreach_known_vec(test, true,
18964b12db3SAakash Sasidharan 					  PDCP_TEST_SUITE_TY_BASIC);
190d84ac1b9SVolodymyr Fialko }
191d84ac1b9SVolodymyr Fialko 
192d84ac1b9SVolodymyr Fialko static inline uint32_t
pdcp_sn_mask_get(enum rte_security_pdcp_sn_size sn_size)193d84ac1b9SVolodymyr Fialko pdcp_sn_mask_get(enum rte_security_pdcp_sn_size sn_size)
194d84ac1b9SVolodymyr Fialko {
195d84ac1b9SVolodymyr Fialko 	return (1 << sn_size) - 1;
196d84ac1b9SVolodymyr Fialko }
197d84ac1b9SVolodymyr Fialko 
198d84ac1b9SVolodymyr Fialko static inline uint32_t
pdcp_sn_from_count_get(uint32_t count,enum rte_security_pdcp_sn_size sn_size)199d84ac1b9SVolodymyr Fialko pdcp_sn_from_count_get(uint32_t count, enum rte_security_pdcp_sn_size sn_size)
200d84ac1b9SVolodymyr Fialko {
201d84ac1b9SVolodymyr Fialko 	return (count & pdcp_sn_mask_get(sn_size));
202d84ac1b9SVolodymyr Fialko }
203d84ac1b9SVolodymyr Fialko 
204d84ac1b9SVolodymyr Fialko static inline uint32_t
pdcp_hfn_mask_get(enum rte_security_pdcp_sn_size sn_size)205d84ac1b9SVolodymyr Fialko pdcp_hfn_mask_get(enum rte_security_pdcp_sn_size sn_size)
206d84ac1b9SVolodymyr Fialko {
207d84ac1b9SVolodymyr Fialko 	return ~pdcp_sn_mask_get(sn_size);
208d84ac1b9SVolodymyr Fialko }
209d84ac1b9SVolodymyr Fialko 
210d84ac1b9SVolodymyr Fialko static inline uint32_t
pdcp_hfn_from_count_get(uint32_t count,enum rte_security_pdcp_sn_size sn_size)211d84ac1b9SVolodymyr Fialko pdcp_hfn_from_count_get(uint32_t count, enum rte_security_pdcp_sn_size sn_size)
212d84ac1b9SVolodymyr Fialko {
213d84ac1b9SVolodymyr Fialko 	return (count & pdcp_hfn_mask_get(sn_size)) >> sn_size;
214d84ac1b9SVolodymyr Fialko }
215d84ac1b9SVolodymyr Fialko 
2166f004629SVolodymyr Fialko static void
pdcp_timer_start_cb(void * timer,void * args)2176f004629SVolodymyr Fialko pdcp_timer_start_cb(void *timer, void *args)
2186f004629SVolodymyr Fialko {
2196f004629SVolodymyr Fialko 	bool *is_timer_running = timer;
2206f004629SVolodymyr Fialko 
2216f004629SVolodymyr Fialko 	RTE_SET_USED(args);
2226f004629SVolodymyr Fialko 	*is_timer_running = true;
2236f004629SVolodymyr Fialko }
2246f004629SVolodymyr Fialko 
2256f004629SVolodymyr Fialko static void
pdcp_timer_stop_cb(void * timer,void * args)2266f004629SVolodymyr Fialko pdcp_timer_stop_cb(void *timer, void *args)
2276f004629SVolodymyr Fialko {
2286f004629SVolodymyr Fialko 	bool *is_timer_running = timer;
2296f004629SVolodymyr Fialko 
2306f004629SVolodymyr Fialko 	RTE_SET_USED(args);
2316f004629SVolodymyr Fialko 	*is_timer_running = false;
2326f004629SVolodymyr Fialko }
2336f004629SVolodymyr Fialko 
2346f004629SVolodymyr Fialko static struct rte_pdcp_t_reordering t_reorder_timer = {
2356f004629SVolodymyr Fialko 	.timer = &testsuite_params.timer_is_running,
2366f004629SVolodymyr Fialko 	.start = pdcp_timer_start_cb,
2376f004629SVolodymyr Fialko 	.stop = pdcp_timer_stop_cb,
2386f004629SVolodymyr Fialko };
2396f004629SVolodymyr Fialko 
2400eeeb7d0SVolodymyr Fialko static inline void
bitmask_set_bit(uint8_t * mask,uint32_t bit)2410eeeb7d0SVolodymyr Fialko bitmask_set_bit(uint8_t *mask, uint32_t bit)
2420eeeb7d0SVolodymyr Fialko {
2430eeeb7d0SVolodymyr Fialko 	mask[bit / 8] |= (1 << bit % 8);
2440eeeb7d0SVolodymyr Fialko }
2450eeeb7d0SVolodymyr Fialko 
2460eeeb7d0SVolodymyr Fialko static inline bool
bitmask_is_bit_set(const uint8_t * mask,uint32_t bit)2470eeeb7d0SVolodymyr Fialko bitmask_is_bit_set(const uint8_t *mask, uint32_t bit)
2480eeeb7d0SVolodymyr Fialko {
2490eeeb7d0SVolodymyr Fialko 	return mask[bit / 8] & (1 << (bit % 8));
2500eeeb7d0SVolodymyr Fialko }
2510eeeb7d0SVolodymyr Fialko 
252bd24488cSAnoob Joseph static inline int
pdcp_hdr_size_get(enum rte_security_pdcp_sn_size sn_size)253bd24488cSAnoob Joseph pdcp_hdr_size_get(enum rte_security_pdcp_sn_size sn_size)
254bd24488cSAnoob Joseph {
255bd24488cSAnoob Joseph 	return RTE_ALIGN_MUL_CEIL(sn_size, 8) / 8;
256bd24488cSAnoob Joseph }
257bd24488cSAnoob Joseph 
258bd24488cSAnoob Joseph static int
pktmbuf_read_into(const struct rte_mbuf * m,void * buf,size_t buf_len)259e98a7ac0SVolodymyr Fialko pktmbuf_read_into(const struct rte_mbuf *m, void *buf, size_t buf_len)
260e98a7ac0SVolodymyr Fialko {
261e98a7ac0SVolodymyr Fialko 	if (m->pkt_len > buf_len)
262e98a7ac0SVolodymyr Fialko 		return -ENOMEM;
263e98a7ac0SVolodymyr Fialko 
264e98a7ac0SVolodymyr Fialko 	const void *read = rte_pktmbuf_read(m, 0, m->pkt_len, buf);
265e98a7ac0SVolodymyr Fialko 	if (read != NULL && read != buf)
266e98a7ac0SVolodymyr Fialko 		memcpy(buf, read, m->pkt_len);
267e98a7ac0SVolodymyr Fialko 
268e98a7ac0SVolodymyr Fialko 	return 0;
269e98a7ac0SVolodymyr Fialko }
270e98a7ac0SVolodymyr Fialko 
271e98a7ac0SVolodymyr Fialko static int
cryptodev_init(int dev_id)272bd24488cSAnoob Joseph cryptodev_init(int dev_id)
273bd24488cSAnoob Joseph {
274bd24488cSAnoob Joseph 	struct pdcp_testsuite_params *ts_params = &testsuite_params;
275bd24488cSAnoob Joseph 	struct rte_cryptodev_qp_conf qp_conf;
276bd24488cSAnoob Joseph 	struct rte_cryptodev_info dev_info;
277bd24488cSAnoob Joseph 	struct rte_cryptodev_config config;
278bd24488cSAnoob Joseph 	int ret, socket_id;
279bd24488cSAnoob Joseph 
280bd24488cSAnoob Joseph 	/* Check if device was already initialized */
281bd24488cSAnoob Joseph 	if (ts_params->cdevs_used[dev_id])
282bd24488cSAnoob Joseph 		return 0;
283bd24488cSAnoob Joseph 
284bd24488cSAnoob Joseph 	rte_cryptodev_info_get(dev_id, &dev_info);
285bd24488cSAnoob Joseph 
286bd24488cSAnoob Joseph 	if (dev_info.max_nb_queue_pairs < 1) {
287bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Cryptodev doesn't have sufficient queue pairs available\n");
288bd24488cSAnoob Joseph 		return -ENODEV;
289bd24488cSAnoob Joseph 	}
290bd24488cSAnoob Joseph 
291bd24488cSAnoob Joseph 	socket_id = rte_socket_id();
292bd24488cSAnoob Joseph 
293bd24488cSAnoob Joseph 	memset(&config, 0, sizeof(config));
294bd24488cSAnoob Joseph 	config.nb_queue_pairs = 1;
295bd24488cSAnoob Joseph 	config.socket_id = socket_id;
296bd24488cSAnoob Joseph 
297bd24488cSAnoob Joseph 	ret = rte_cryptodev_configure(dev_id, &config);
298bd24488cSAnoob Joseph 	if (ret < 0) {
299bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not configure cryptodev - %d\n", dev_id);
300bd24488cSAnoob Joseph 		return -ENODEV;
301bd24488cSAnoob Joseph 	}
302bd24488cSAnoob Joseph 
303bd24488cSAnoob Joseph 	memset(&qp_conf, 0, sizeof(qp_conf));
304bd24488cSAnoob Joseph 	qp_conf.nb_descriptors = NB_DESC;
305bd24488cSAnoob Joseph 
306bd24488cSAnoob Joseph 	ret = rte_cryptodev_queue_pair_setup(dev_id, 0, &qp_conf, socket_id);
307bd24488cSAnoob Joseph 	if (ret < 0) {
308bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not configure queue pair\n");
309bd24488cSAnoob Joseph 		return -ENODEV;
310bd24488cSAnoob Joseph 	}
311bd24488cSAnoob Joseph 
312bd24488cSAnoob Joseph 	ret = rte_cryptodev_start(dev_id);
313bd24488cSAnoob Joseph 	if (ret < 0) {
314bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not start cryptodev\n");
315bd24488cSAnoob Joseph 		return -ENODEV;
316bd24488cSAnoob Joseph 	}
317bd24488cSAnoob Joseph 
318bd24488cSAnoob Joseph 	/* Mark device as initialized */
319bd24488cSAnoob Joseph 	ts_params->cdevs_used[dev_id] = true;
320bd24488cSAnoob Joseph 
321bd24488cSAnoob Joseph 	return 0;
322bd24488cSAnoob Joseph }
323bd24488cSAnoob Joseph 
324bd24488cSAnoob Joseph static void
cryptodev_fini(int dev_id)325bd24488cSAnoob Joseph cryptodev_fini(int dev_id)
326bd24488cSAnoob Joseph {
327bd24488cSAnoob Joseph 	rte_cryptodev_stop(dev_id);
328bd24488cSAnoob Joseph }
329bd24488cSAnoob Joseph 
330bd24488cSAnoob Joseph static unsigned int
cryptodev_sess_priv_max_req_get(void)331bd24488cSAnoob Joseph cryptodev_sess_priv_max_req_get(void)
332bd24488cSAnoob Joseph {
333bd24488cSAnoob Joseph 	struct rte_cryptodev_info info;
334bd24488cSAnoob Joseph 	unsigned int sess_priv_sz;
335bd24488cSAnoob Joseph 	int i, nb_dev;
336bd24488cSAnoob Joseph 	void *sec_ctx;
337bd24488cSAnoob Joseph 
338bd24488cSAnoob Joseph 	nb_dev = rte_cryptodev_count();
339bd24488cSAnoob Joseph 
340bd24488cSAnoob Joseph 	sess_priv_sz = 0;
341bd24488cSAnoob Joseph 
342bd24488cSAnoob Joseph 	for (i = 0; i < nb_dev; i++) {
343bd24488cSAnoob Joseph 		rte_cryptodev_info_get(i, &info);
344bd24488cSAnoob Joseph 		sess_priv_sz = RTE_MAX(sess_priv_sz, rte_cryptodev_sym_get_private_session_size(i));
345bd24488cSAnoob Joseph 		if (info.feature_flags & RTE_CRYPTODEV_FF_SECURITY) {
346bd24488cSAnoob Joseph 			sec_ctx = rte_cryptodev_get_sec_ctx(i);
347bd24488cSAnoob Joseph 			sess_priv_sz = RTE_MAX(sess_priv_sz,
348bd24488cSAnoob Joseph 					       rte_security_session_get_size(sec_ctx));
349bd24488cSAnoob Joseph 		}
350bd24488cSAnoob Joseph 	}
351bd24488cSAnoob Joseph 
352bd24488cSAnoob Joseph 	return sess_priv_sz;
353bd24488cSAnoob Joseph }
354bd24488cSAnoob Joseph 
355bd24488cSAnoob Joseph static int
testsuite_setup(void)356bd24488cSAnoob Joseph testsuite_setup(void)
357bd24488cSAnoob Joseph {
358bd24488cSAnoob Joseph 	struct pdcp_testsuite_params *ts_params = &testsuite_params;
359bd24488cSAnoob Joseph 	int nb_cdev, sess_priv_size, nb_sess = 1024;
360bd24488cSAnoob Joseph 
361bd24488cSAnoob Joseph 	RTE_SET_USED(pdcp_test_hfn_threshold);
362bd24488cSAnoob Joseph 
363bd24488cSAnoob Joseph 	nb_cdev = rte_cryptodev_count();
364bd24488cSAnoob Joseph 	if (nb_cdev < 1) {
365bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "No crypto devices found.\n");
366bd24488cSAnoob Joseph 		return TEST_SKIPPED;
367bd24488cSAnoob Joseph 	}
368bd24488cSAnoob Joseph 
369bd24488cSAnoob Joseph 	memset(ts_params, 0, sizeof(*ts_params));
370bd24488cSAnoob Joseph 
371bd24488cSAnoob Joseph 	ts_params->mbuf_pool = rte_pktmbuf_pool_create("mbuf_pool", NUM_MBUFS, MBUF_CACHE_SIZE, 0,
372*d010725bSAakash Sasidharan 						       PDCP_MBUF_SIZE, SOCKET_ID_ANY);
373bd24488cSAnoob Joseph 	if (ts_params->mbuf_pool == NULL) {
374bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not create mbuf pool\n");
375bd24488cSAnoob Joseph 		return TEST_FAILED;
376bd24488cSAnoob Joseph 	}
377bd24488cSAnoob Joseph 
378bd24488cSAnoob Joseph 	ts_params->cop_pool = rte_crypto_op_pool_create("cop_pool", RTE_CRYPTO_OP_TYPE_SYMMETRIC,
379bd24488cSAnoob Joseph 							 NUM_MBUFS, MBUF_CACHE_SIZE,
380bd24488cSAnoob Joseph 							 2 * MAXIMUM_IV_LENGTH, SOCKET_ID_ANY);
381bd24488cSAnoob Joseph 	if (ts_params->cop_pool == NULL) {
382bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not create crypto_op pool\n");
383bd24488cSAnoob Joseph 		goto mbuf_pool_free;
384bd24488cSAnoob Joseph 	}
385bd24488cSAnoob Joseph 
386bd24488cSAnoob Joseph 	/* Get max session priv size required */
387bd24488cSAnoob Joseph 	sess_priv_size = cryptodev_sess_priv_max_req_get();
388bd24488cSAnoob Joseph 
389bd24488cSAnoob Joseph 	ts_params->sess_pool = rte_cryptodev_sym_session_pool_create("sess_pool", nb_sess,
390bd24488cSAnoob Joseph 								     sess_priv_size,
391bd24488cSAnoob Joseph 								     RTE_MEMPOOL_CACHE_MAX_SIZE,
392bd24488cSAnoob Joseph 								     0, SOCKET_ID_ANY);
393bd24488cSAnoob Joseph 	if (ts_params->sess_pool == NULL) {
394bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not create session pool\n");
395bd24488cSAnoob Joseph 		goto cop_pool_free;
396bd24488cSAnoob Joseph 	}
397bd24488cSAnoob Joseph 
3980eeeb7d0SVolodymyr Fialko 	/* Allocate memory for longest possible status report */
3990eeeb7d0SVolodymyr Fialko 	ts_params->status_report_bitmask_capacity = RTE_PDCP_CTRL_PDU_SIZE_MAX -
4000eeeb7d0SVolodymyr Fialko 		sizeof(struct rte_pdcp_up_ctrl_pdu_hdr);
4010eeeb7d0SVolodymyr Fialko 	ts_params->status_report = rte_zmalloc(NULL, RTE_PDCP_CTRL_PDU_SIZE_MAX, 0);
4020eeeb7d0SVolodymyr Fialko 	if (ts_params->status_report == NULL) {
4030eeeb7d0SVolodymyr Fialko 		RTE_LOG(ERR, USER1, "Could not allocate status report\n");
4040eeeb7d0SVolodymyr Fialko 		goto cop_pool_free;
4050eeeb7d0SVolodymyr Fialko 	}
4060eeeb7d0SVolodymyr Fialko 
4070eeeb7d0SVolodymyr Fialko 	ts_params->ctrl_pdu_buf = rte_zmalloc(NULL, RTE_PDCP_CTRL_PDU_SIZE_MAX, 0);
4080eeeb7d0SVolodymyr Fialko 	if (ts_params->ctrl_pdu_buf == NULL) {
4090eeeb7d0SVolodymyr Fialko 		RTE_LOG(ERR, USER1, "Could not allocate status report data\n");
4100eeeb7d0SVolodymyr Fialko 		goto cop_pool_free;
4110eeeb7d0SVolodymyr Fialko 	}
4120eeeb7d0SVolodymyr Fialko 
413bd24488cSAnoob Joseph 	return 0;
414bd24488cSAnoob Joseph 
415bd24488cSAnoob Joseph cop_pool_free:
416bd24488cSAnoob Joseph 	rte_mempool_free(ts_params->cop_pool);
417bd24488cSAnoob Joseph 	ts_params->cop_pool = NULL;
418bd24488cSAnoob Joseph mbuf_pool_free:
419bd24488cSAnoob Joseph 	rte_mempool_free(ts_params->mbuf_pool);
420bd24488cSAnoob Joseph 	ts_params->mbuf_pool = NULL;
4210eeeb7d0SVolodymyr Fialko 	rte_free(ts_params->status_report);
4220eeeb7d0SVolodymyr Fialko 	rte_free(ts_params->ctrl_pdu_buf);
423bd24488cSAnoob Joseph 	return TEST_FAILED;
424bd24488cSAnoob Joseph }
425bd24488cSAnoob Joseph 
426bd24488cSAnoob Joseph static void
testsuite_teardown(void)427bd24488cSAnoob Joseph testsuite_teardown(void)
428bd24488cSAnoob Joseph {
429bd24488cSAnoob Joseph 	struct pdcp_testsuite_params *ts_params = &testsuite_params;
430bd24488cSAnoob Joseph 	uint8_t dev_id;
431bd24488cSAnoob Joseph 
432bd24488cSAnoob Joseph 	for (dev_id = 0; dev_id < RTE_CRYPTO_MAX_DEVS; dev_id++) {
433bd24488cSAnoob Joseph 		if (ts_params->cdevs_used[dev_id])
434bd24488cSAnoob Joseph 			cryptodev_fini(dev_id);
435bd24488cSAnoob Joseph 	}
436bd24488cSAnoob Joseph 
437bd24488cSAnoob Joseph 	rte_mempool_free(ts_params->sess_pool);
438bd24488cSAnoob Joseph 	ts_params->sess_pool = NULL;
439bd24488cSAnoob Joseph 
440bd24488cSAnoob Joseph 	rte_mempool_free(ts_params->cop_pool);
441bd24488cSAnoob Joseph 	ts_params->cop_pool = NULL;
442bd24488cSAnoob Joseph 
443bd24488cSAnoob Joseph 	rte_mempool_free(ts_params->mbuf_pool);
444bd24488cSAnoob Joseph 	ts_params->mbuf_pool = NULL;
4450eeeb7d0SVolodymyr Fialko 
4460eeeb7d0SVolodymyr Fialko 	rte_free(ts_params->status_report);
4470eeeb7d0SVolodymyr Fialko 	rte_free(ts_params->ctrl_pdu_buf);
448bd24488cSAnoob Joseph }
449bd24488cSAnoob Joseph 
450bd24488cSAnoob Joseph static int
ut_setup_pdcp(void)451bd24488cSAnoob Joseph ut_setup_pdcp(void)
452bd24488cSAnoob Joseph {
453bd24488cSAnoob Joseph 	return 0;
454bd24488cSAnoob Joseph }
455bd24488cSAnoob Joseph 
456bd24488cSAnoob Joseph static void
ut_teardown_pdcp(void)457bd24488cSAnoob Joseph ut_teardown_pdcp(void)
458bd24488cSAnoob Joseph {
459bd24488cSAnoob Joseph }
460bd24488cSAnoob Joseph 
461bd24488cSAnoob Joseph static int
crypto_caps_cipher_verify(uint8_t dev_id,const struct rte_crypto_sym_xform * c_xfrm)462bd24488cSAnoob Joseph crypto_caps_cipher_verify(uint8_t dev_id, const struct rte_crypto_sym_xform *c_xfrm)
463bd24488cSAnoob Joseph {
464bd24488cSAnoob Joseph 	const struct rte_cryptodev_symmetric_capability *cap;
465bd24488cSAnoob Joseph 	struct rte_cryptodev_sym_capability_idx cap_idx;
466bd24488cSAnoob Joseph 	int ret;
467bd24488cSAnoob Joseph 
468bd24488cSAnoob Joseph 	cap_idx.type = RTE_CRYPTO_SYM_XFORM_CIPHER;
469bd24488cSAnoob Joseph 	cap_idx.algo.cipher = c_xfrm->cipher.algo;
470bd24488cSAnoob Joseph 
471bd24488cSAnoob Joseph 	cap = rte_cryptodev_sym_capability_get(dev_id, &cap_idx);
472bd24488cSAnoob Joseph 	if (cap == NULL)
473bd24488cSAnoob Joseph 		return -1;
474bd24488cSAnoob Joseph 
475bd24488cSAnoob Joseph 	ret = rte_cryptodev_sym_capability_check_cipher(cap, c_xfrm->cipher.key.length,
476bd24488cSAnoob Joseph 							c_xfrm->cipher.iv.length);
477bd24488cSAnoob Joseph 
478bd24488cSAnoob Joseph 	return ret;
479bd24488cSAnoob Joseph }
480bd24488cSAnoob Joseph 
481bd24488cSAnoob Joseph static int
crypto_caps_auth_verify(uint8_t dev_id,const struct rte_crypto_sym_xform * a_xfrm)482bd24488cSAnoob Joseph crypto_caps_auth_verify(uint8_t dev_id, const struct rte_crypto_sym_xform *a_xfrm)
483bd24488cSAnoob Joseph {
484bd24488cSAnoob Joseph 	const struct rte_cryptodev_symmetric_capability *cap;
485bd24488cSAnoob Joseph 	struct rte_cryptodev_sym_capability_idx cap_idx;
486bd24488cSAnoob Joseph 	int ret;
487bd24488cSAnoob Joseph 
488bd24488cSAnoob Joseph 	cap_idx.type = RTE_CRYPTO_SYM_XFORM_AUTH;
489bd24488cSAnoob Joseph 	cap_idx.algo.auth = a_xfrm->auth.algo;
490bd24488cSAnoob Joseph 
491bd24488cSAnoob Joseph 	cap = rte_cryptodev_sym_capability_get(dev_id, &cap_idx);
492bd24488cSAnoob Joseph 	if (cap == NULL)
493bd24488cSAnoob Joseph 		return -1;
494bd24488cSAnoob Joseph 
495bd24488cSAnoob Joseph 	ret = rte_cryptodev_sym_capability_check_auth(cap, a_xfrm->auth.key.length,
496bd24488cSAnoob Joseph 						      a_xfrm->auth.digest_length,
497bd24488cSAnoob Joseph 						      a_xfrm->auth.iv.length);
498bd24488cSAnoob Joseph 
499bd24488cSAnoob Joseph 	return ret;
500bd24488cSAnoob Joseph }
501bd24488cSAnoob Joseph 
502bd24488cSAnoob Joseph static int
cryptodev_id_get(bool is_integrity_protected,const struct rte_crypto_sym_xform * c_xfrm,const struct rte_crypto_sym_xform * a_xfrm)503bd24488cSAnoob Joseph cryptodev_id_get(bool is_integrity_protected, const struct rte_crypto_sym_xform *c_xfrm,
504bd24488cSAnoob Joseph 		 const struct rte_crypto_sym_xform *a_xfrm)
505bd24488cSAnoob Joseph {
506bd24488cSAnoob Joseph 	int i, nb_devs;
507bd24488cSAnoob Joseph 
508bd24488cSAnoob Joseph 	nb_devs = rte_cryptodev_count();
509bd24488cSAnoob Joseph 
510bd24488cSAnoob Joseph 	/* Check capabilities */
511bd24488cSAnoob Joseph 
512bd24488cSAnoob Joseph 	for (i = 0; i < nb_devs; i++) {
513bd24488cSAnoob Joseph 		if ((crypto_caps_cipher_verify(i, c_xfrm) == 0) &&
514bd24488cSAnoob Joseph 		    (!is_integrity_protected || crypto_caps_auth_verify(i, a_xfrm) == 0))
515bd24488cSAnoob Joseph 			break;
516bd24488cSAnoob Joseph 	}
517bd24488cSAnoob Joseph 
518bd24488cSAnoob Joseph 	if (i == nb_devs)
519bd24488cSAnoob Joseph 		return -1;
520bd24488cSAnoob Joseph 
521bd24488cSAnoob Joseph 	return i;
522bd24488cSAnoob Joseph }
523bd24488cSAnoob Joseph 
524bd24488cSAnoob Joseph static int
pdcp_known_vec_verify(struct rte_mbuf * m,const uint8_t * expected,uint32_t expected_pkt_len)525bd24488cSAnoob Joseph pdcp_known_vec_verify(struct rte_mbuf *m, const uint8_t *expected, uint32_t expected_pkt_len)
526bd24488cSAnoob Joseph {
527bd24488cSAnoob Joseph 	uint8_t *actual = rte_pktmbuf_mtod(m, uint8_t *);
528bd24488cSAnoob Joseph 	uint32_t actual_pkt_len = rte_pktmbuf_pkt_len(m);
529bd24488cSAnoob Joseph 
530*d010725bSAakash Sasidharan 	if (!silent) {
531bd24488cSAnoob Joseph 		debug_hexdump(stdout, "Received:", actual, actual_pkt_len);
532bd24488cSAnoob Joseph 		debug_hexdump(stdout, "Expected:", expected, expected_pkt_len);
533*d010725bSAakash Sasidharan 	}
534bd24488cSAnoob Joseph 
535bd24488cSAnoob Joseph 	TEST_ASSERT_EQUAL(actual_pkt_len, expected_pkt_len,
536bd24488cSAnoob Joseph 			  "Mismatch in packet lengths [expected: %d, received: %d]",
537bd24488cSAnoob Joseph 			  expected_pkt_len, actual_pkt_len);
538bd24488cSAnoob Joseph 
539bd24488cSAnoob Joseph 	TEST_ASSERT_BUFFERS_ARE_EQUAL(actual, expected, expected_pkt_len,
540bd24488cSAnoob Joseph 				     "Generated packet not as expected");
541bd24488cSAnoob Joseph 
542bd24488cSAnoob Joseph 	return 0;
543bd24488cSAnoob Joseph }
544bd24488cSAnoob Joseph 
545bd24488cSAnoob Joseph static struct rte_crypto_op *
process_crypto_request(uint8_t dev_id,struct rte_crypto_op * op)546bd24488cSAnoob Joseph process_crypto_request(uint8_t dev_id, struct rte_crypto_op *op)
547bd24488cSAnoob Joseph {
548bd24488cSAnoob Joseph 	if (rte_cryptodev_enqueue_burst(dev_id, 0, &op, 1) != 1) {
549bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Error sending packet to cryptodev\n");
550bd24488cSAnoob Joseph 		return NULL;
551bd24488cSAnoob Joseph 	}
552bd24488cSAnoob Joseph 
553bd24488cSAnoob Joseph 	op = NULL;
554bd24488cSAnoob Joseph 
555bd24488cSAnoob Joseph 	while (rte_cryptodev_dequeue_burst(dev_id, 0, &op, 1) == 0)
556bd24488cSAnoob Joseph 		rte_pause();
557bd24488cSAnoob Joseph 
558bd24488cSAnoob Joseph 	return op;
559bd24488cSAnoob Joseph }
560bd24488cSAnoob Joseph 
561bd24488cSAnoob Joseph static uint32_t
pdcp_sn_from_raw_get(const void * data,enum rte_security_pdcp_sn_size size)562bd24488cSAnoob Joseph pdcp_sn_from_raw_get(const void *data, enum rte_security_pdcp_sn_size size)
563bd24488cSAnoob Joseph {
564bd24488cSAnoob Joseph 	uint32_t sn = 0;
565bd24488cSAnoob Joseph 
566bd24488cSAnoob Joseph 	if (size == RTE_SECURITY_PDCP_SN_SIZE_12) {
567bd24488cSAnoob Joseph 		sn = rte_cpu_to_be_16(*(const uint16_t *)data);
568bd24488cSAnoob Joseph 		sn = sn & 0xfff;
569bd24488cSAnoob Joseph 	} else if (size == RTE_SECURITY_PDCP_SN_SIZE_18) {
570bd24488cSAnoob Joseph 		sn = rte_cpu_to_be_32(*(const uint32_t *)data);
571bd24488cSAnoob Joseph 		sn = (sn & 0x3ffff00) >> 8;
572bd24488cSAnoob Joseph 	}
573bd24488cSAnoob Joseph 
574bd24488cSAnoob Joseph 	return sn;
575bd24488cSAnoob Joseph }
576bd24488cSAnoob Joseph 
577e98a7ac0SVolodymyr Fialko static void
pdcp_sn_to_raw_set(void * data,uint32_t sn,int size)578e98a7ac0SVolodymyr Fialko pdcp_sn_to_raw_set(void *data, uint32_t sn, int size)
579e98a7ac0SVolodymyr Fialko {
580e98a7ac0SVolodymyr Fialko 	if (size == RTE_SECURITY_PDCP_SN_SIZE_12) {
581e98a7ac0SVolodymyr Fialko 		struct rte_pdcp_up_data_pdu_sn_12_hdr *pdu_hdr = data;
582e98a7ac0SVolodymyr Fialko 		pdu_hdr->sn_11_8 = ((sn & 0xf00) >> 8);
583e98a7ac0SVolodymyr Fialko 		pdu_hdr->sn_7_0 = (sn & 0xff);
584e98a7ac0SVolodymyr Fialko 	} else if (size == RTE_SECURITY_PDCP_SN_SIZE_18) {
585e98a7ac0SVolodymyr Fialko 		struct rte_pdcp_up_data_pdu_sn_18_hdr *pdu_hdr = data;
586e98a7ac0SVolodymyr Fialko 		pdu_hdr->sn_17_16 = ((sn & 0x30000) >> 16);
587e98a7ac0SVolodymyr Fialko 		pdu_hdr->sn_15_8 = ((sn & 0xff00) >> 8);
588e98a7ac0SVolodymyr Fialko 		pdu_hdr->sn_7_0 = (sn & 0xff);
589e98a7ac0SVolodymyr Fialko 	}
590e98a7ac0SVolodymyr Fialko }
591e98a7ac0SVolodymyr Fialko 
59264b12db3SAakash Sasidharan static uint8_t
pdcp_test_bearer_get(enum pdcp_test_suite_type suite_type,const int index)59364b12db3SAakash Sasidharan pdcp_test_bearer_get(enum pdcp_test_suite_type suite_type, const int index)
59464b12db3SAakash Sasidharan {
59564b12db3SAakash Sasidharan 	uint8_t ret;
59664b12db3SAakash Sasidharan 
59764b12db3SAakash Sasidharan 	switch (suite_type) {
59864b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
59964b12db3SAakash Sasidharan 		ret = pdcp_test_bearer[index];
60064b12db3SAakash Sasidharan 		break;
60164b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
60264b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].bearer;
60364b12db3SAakash Sasidharan 		break;
60464b12db3SAakash Sasidharan 	default:
60564b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
60664b12db3SAakash Sasidharan 		ret = -1;
60764b12db3SAakash Sasidharan 
60864b12db3SAakash Sasidharan 	}
60964b12db3SAakash Sasidharan 
61064b12db3SAakash Sasidharan 	return ret;
61164b12db3SAakash Sasidharan }
61264b12db3SAakash Sasidharan 
61364b12db3SAakash Sasidharan static enum rte_security_pdcp_domain
pdcp_test_param_domain_get(enum pdcp_test_suite_type suite_type,const int index)61464b12db3SAakash Sasidharan pdcp_test_param_domain_get(enum pdcp_test_suite_type suite_type, const int index)
61564b12db3SAakash Sasidharan {
61664b12db3SAakash Sasidharan 	enum rte_security_pdcp_domain ret;
61764b12db3SAakash Sasidharan 
61864b12db3SAakash Sasidharan 	switch (suite_type) {
61964b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
62064b12db3SAakash Sasidharan 		ret = pdcp_test_params[index].domain;
62164b12db3SAakash Sasidharan 		break;
62264b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
62364b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].param.domain;
62464b12db3SAakash Sasidharan 		break;
62564b12db3SAakash Sasidharan 	default:
62664b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
62764b12db3SAakash Sasidharan 		ret = -1;
62864b12db3SAakash Sasidharan 	}
62964b12db3SAakash Sasidharan 
63064b12db3SAakash Sasidharan 	return ret;
63164b12db3SAakash Sasidharan }
63264b12db3SAakash Sasidharan 
63364b12db3SAakash Sasidharan static uint8_t
pdcp_test_data_sn_size_get(enum pdcp_test_suite_type suite_type,const int index)63464b12db3SAakash Sasidharan pdcp_test_data_sn_size_get(enum pdcp_test_suite_type suite_type, const int index)
63564b12db3SAakash Sasidharan {
63664b12db3SAakash Sasidharan 	uint8_t ret;
63764b12db3SAakash Sasidharan 
63864b12db3SAakash Sasidharan 	switch (suite_type) {
63964b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
64064b12db3SAakash Sasidharan 		ret = pdcp_test_data_sn_size[index];
64164b12db3SAakash Sasidharan 		break;
64264b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
64364b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].sn_size;
64464b12db3SAakash Sasidharan 		break;
64564b12db3SAakash Sasidharan 	default:
64664b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
64764b12db3SAakash Sasidharan 		return -1;
64864b12db3SAakash Sasidharan 
64964b12db3SAakash Sasidharan 	}
65064b12db3SAakash Sasidharan 
65164b12db3SAakash Sasidharan 	return ret;
65264b12db3SAakash Sasidharan }
65364b12db3SAakash Sasidharan 
65464b12db3SAakash Sasidharan static uint8_t
pdcp_test_packet_direction_get(enum pdcp_test_suite_type suite_type,const int index)65564b12db3SAakash Sasidharan pdcp_test_packet_direction_get(enum pdcp_test_suite_type suite_type, const int index)
65664b12db3SAakash Sasidharan {
65764b12db3SAakash Sasidharan 	uint8_t ret;
65864b12db3SAakash Sasidharan 
65964b12db3SAakash Sasidharan 	switch (suite_type) {
66064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
66164b12db3SAakash Sasidharan 		ret = pdcp_test_packet_direction[index];
66264b12db3SAakash Sasidharan 		break;
66364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
66464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].packet_direction;
66564b12db3SAakash Sasidharan 		break;
66664b12db3SAakash Sasidharan 	default:
66764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
66864b12db3SAakash Sasidharan 		return -1;
66964b12db3SAakash Sasidharan 	}
67064b12db3SAakash Sasidharan 
67164b12db3SAakash Sasidharan 	return ret;
67264b12db3SAakash Sasidharan }
67364b12db3SAakash Sasidharan 
67464b12db3SAakash Sasidharan static enum rte_crypto_cipher_algorithm
pdcp_test_param_cipher_alg_get(enum pdcp_test_suite_type suite_type,const int index)67564b12db3SAakash Sasidharan pdcp_test_param_cipher_alg_get(enum pdcp_test_suite_type suite_type, const int index)
67664b12db3SAakash Sasidharan {
67764b12db3SAakash Sasidharan 	enum rte_crypto_cipher_algorithm ret;
67864b12db3SAakash Sasidharan 
67964b12db3SAakash Sasidharan 	switch (suite_type) {
68064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
68164b12db3SAakash Sasidharan 		ret = pdcp_test_params[index].cipher_alg;
68264b12db3SAakash Sasidharan 		break;
68364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
68464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].param.cipher_alg;
68564b12db3SAakash Sasidharan 		break;
68664b12db3SAakash Sasidharan 	default:
68764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
68864b12db3SAakash Sasidharan 		return 0;
68964b12db3SAakash Sasidharan 	}
69064b12db3SAakash Sasidharan 
69164b12db3SAakash Sasidharan 	return ret;
69264b12db3SAakash Sasidharan }
69364b12db3SAakash Sasidharan 
69464b12db3SAakash Sasidharan static uint8_t
pdcp_test_param_cipher_key_len_get(enum pdcp_test_suite_type suite_type,const int index)69564b12db3SAakash Sasidharan pdcp_test_param_cipher_key_len_get(enum pdcp_test_suite_type suite_type, const int index)
69664b12db3SAakash Sasidharan {
69764b12db3SAakash Sasidharan 	uint8_t ret;
69864b12db3SAakash Sasidharan 
69964b12db3SAakash Sasidharan 	switch (suite_type) {
70064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
70164b12db3SAakash Sasidharan 		ret = pdcp_test_params[index].cipher_key_len;
70264b12db3SAakash Sasidharan 		break;
70364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
70464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].param.cipher_key_len;
70564b12db3SAakash Sasidharan 		break;
70664b12db3SAakash Sasidharan 	default:
70764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
70864b12db3SAakash Sasidharan 		return -1;
70964b12db3SAakash Sasidharan 	}
71064b12db3SAakash Sasidharan 
71164b12db3SAakash Sasidharan 	return ret;
71264b12db3SAakash Sasidharan }
71364b12db3SAakash Sasidharan 
71464b12db3SAakash Sasidharan static const uint8_t*
pdcp_test_crypto_key_get(enum pdcp_test_suite_type suite_type,const int index)71564b12db3SAakash Sasidharan pdcp_test_crypto_key_get(enum pdcp_test_suite_type suite_type, const int index)
71664b12db3SAakash Sasidharan {
71764b12db3SAakash Sasidharan 	const uint8_t *ret;
71864b12db3SAakash Sasidharan 
71964b12db3SAakash Sasidharan 	switch (suite_type) {
72064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
72164b12db3SAakash Sasidharan 		ret = pdcp_test_crypto_key[index];
72264b12db3SAakash Sasidharan 		break;
72364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
72464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].cipher_key;
72564b12db3SAakash Sasidharan 		break;
72664b12db3SAakash Sasidharan 	default:
72764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
72864b12db3SAakash Sasidharan 		return NULL;
72964b12db3SAakash Sasidharan 	}
73064b12db3SAakash Sasidharan 
73164b12db3SAakash Sasidharan 	return ret;
73264b12db3SAakash Sasidharan }
73364b12db3SAakash Sasidharan 
73464b12db3SAakash Sasidharan static enum rte_crypto_auth_algorithm
pdcp_test_param_auth_alg_get(enum pdcp_test_suite_type suite_type,const int index)73564b12db3SAakash Sasidharan pdcp_test_param_auth_alg_get(enum pdcp_test_suite_type suite_type, const int index)
73664b12db3SAakash Sasidharan {
73764b12db3SAakash Sasidharan 	enum rte_crypto_auth_algorithm ret;
73864b12db3SAakash Sasidharan 
73964b12db3SAakash Sasidharan 	switch (suite_type) {
74064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
74164b12db3SAakash Sasidharan 		ret = pdcp_test_params[index].auth_alg;
74264b12db3SAakash Sasidharan 		break;
74364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
74464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].param.auth_alg;
74564b12db3SAakash Sasidharan 		break;
74664b12db3SAakash Sasidharan 	default:
74764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
74864b12db3SAakash Sasidharan 		return 0;
74964b12db3SAakash Sasidharan 	}
75064b12db3SAakash Sasidharan 
75164b12db3SAakash Sasidharan 	return ret;
75264b12db3SAakash Sasidharan }
75364b12db3SAakash Sasidharan 
75464b12db3SAakash Sasidharan static uint8_t
pdcp_test_param_auth_key_len_get(enum pdcp_test_suite_type suite_type,const int index)75564b12db3SAakash Sasidharan pdcp_test_param_auth_key_len_get(enum pdcp_test_suite_type suite_type, const int index)
75664b12db3SAakash Sasidharan {
75764b12db3SAakash Sasidharan 	uint8_t ret;
75864b12db3SAakash Sasidharan 
75964b12db3SAakash Sasidharan 	switch (suite_type) {
76064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
76164b12db3SAakash Sasidharan 		ret = pdcp_test_params[index].auth_key_len;
76264b12db3SAakash Sasidharan 		break;
76364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
76464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].param.auth_key_len;
76564b12db3SAakash Sasidharan 		break;
76664b12db3SAakash Sasidharan 	default:
76764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
76864b12db3SAakash Sasidharan 		return -1;
76964b12db3SAakash Sasidharan 	}
77064b12db3SAakash Sasidharan 
77164b12db3SAakash Sasidharan 	return ret;
77264b12db3SAakash Sasidharan }
77364b12db3SAakash Sasidharan 
77464b12db3SAakash Sasidharan static const uint8_t*
pdcp_test_auth_key_get(enum pdcp_test_suite_type suite_type,const int index)77564b12db3SAakash Sasidharan pdcp_test_auth_key_get(enum pdcp_test_suite_type suite_type, const int index)
77664b12db3SAakash Sasidharan {
77764b12db3SAakash Sasidharan 	const uint8_t *ret;
77864b12db3SAakash Sasidharan 
77964b12db3SAakash Sasidharan 	switch (suite_type) {
78064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
78164b12db3SAakash Sasidharan 		ret = pdcp_test_auth_key[index];
78264b12db3SAakash Sasidharan 		break;
78364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
78464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].auth_key;
78564b12db3SAakash Sasidharan 		break;
78664b12db3SAakash Sasidharan 	default:
78764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
78864b12db3SAakash Sasidharan 		return NULL;
78964b12db3SAakash Sasidharan 	}
79064b12db3SAakash Sasidharan 
79164b12db3SAakash Sasidharan 	return ret;
79264b12db3SAakash Sasidharan }
79364b12db3SAakash Sasidharan 
79464b12db3SAakash Sasidharan static const uint8_t*
pdcp_test_data_in_get(enum pdcp_test_suite_type suite_type,const int index)79564b12db3SAakash Sasidharan pdcp_test_data_in_get(enum pdcp_test_suite_type suite_type, const int index)
79664b12db3SAakash Sasidharan {
79764b12db3SAakash Sasidharan 	const uint8_t *ret;
79864b12db3SAakash Sasidharan 
79964b12db3SAakash Sasidharan 	switch (suite_type) {
80064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
80164b12db3SAakash Sasidharan 		ret = pdcp_test_data_in[index];
80264b12db3SAakash Sasidharan 		break;
80364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
80464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].data_in;
80564b12db3SAakash Sasidharan 		break;
80664b12db3SAakash Sasidharan 	default:
80764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
80864b12db3SAakash Sasidharan 		return NULL;
80964b12db3SAakash Sasidharan 	}
81064b12db3SAakash Sasidharan 
81164b12db3SAakash Sasidharan 	return ret;
81264b12db3SAakash Sasidharan }
81364b12db3SAakash Sasidharan 
81464b12db3SAakash Sasidharan static uint8_t
pdcp_test_data_in_len_get(enum pdcp_test_suite_type suite_type,const int index)81564b12db3SAakash Sasidharan pdcp_test_data_in_len_get(enum pdcp_test_suite_type suite_type, const int index)
81664b12db3SAakash Sasidharan {
81764b12db3SAakash Sasidharan 	uint8_t ret;
81864b12db3SAakash Sasidharan 
81964b12db3SAakash Sasidharan 	switch (suite_type) {
82064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
82164b12db3SAakash Sasidharan 		ret = pdcp_test_data_in_len[index];
82264b12db3SAakash Sasidharan 		break;
82364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
82464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].in_len;
82564b12db3SAakash Sasidharan 		break;
82664b12db3SAakash Sasidharan 	default:
82764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
82864b12db3SAakash Sasidharan 		return -1;
82964b12db3SAakash Sasidharan 	}
83064b12db3SAakash Sasidharan 
83164b12db3SAakash Sasidharan 	return ret;
83264b12db3SAakash Sasidharan }
83364b12db3SAakash Sasidharan 
83464b12db3SAakash Sasidharan static const uint8_t*
pdcp_test_data_out_get(enum pdcp_test_suite_type suite_type,const int index)83564b12db3SAakash Sasidharan pdcp_test_data_out_get(enum pdcp_test_suite_type suite_type, const int index)
83664b12db3SAakash Sasidharan {
83764b12db3SAakash Sasidharan 	const uint8_t *ret;
83864b12db3SAakash Sasidharan 
83964b12db3SAakash Sasidharan 	switch (suite_type) {
84064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
84164b12db3SAakash Sasidharan 		ret = pdcp_test_data_out[index];
84264b12db3SAakash Sasidharan 		break;
84364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
84464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].data_out;
84564b12db3SAakash Sasidharan 		break;
84664b12db3SAakash Sasidharan 	default:
84764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
84864b12db3SAakash Sasidharan 		return NULL;
84964b12db3SAakash Sasidharan 	}
85064b12db3SAakash Sasidharan 
85164b12db3SAakash Sasidharan 	return ret;
85264b12db3SAakash Sasidharan }
85364b12db3SAakash Sasidharan 
85464b12db3SAakash Sasidharan static uint32_t
pdcp_test_hfn_get(enum pdcp_test_suite_type suite_type,const int index)85564b12db3SAakash Sasidharan pdcp_test_hfn_get(enum pdcp_test_suite_type suite_type, const int index)
85664b12db3SAakash Sasidharan {
85764b12db3SAakash Sasidharan 	uint32_t ret;
85864b12db3SAakash Sasidharan 
85964b12db3SAakash Sasidharan 	switch (suite_type) {
86064b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_BASIC:
86164b12db3SAakash Sasidharan 		ret = pdcp_test_hfn[index];
86264b12db3SAakash Sasidharan 		break;
86364b12db3SAakash Sasidharan 	case PDCP_TEST_SUITE_TY_SDAP:
86464b12db3SAakash Sasidharan 		ret = list_pdcp_sdap_tests[index].hfn;
86564b12db3SAakash Sasidharan 		break;
86664b12db3SAakash Sasidharan 	default:
86764b12db3SAakash Sasidharan 		RTE_LOG(ERR, USER1, "Invalid suite_type: %d\n", suite_type);
86864b12db3SAakash Sasidharan 		return -1;
86964b12db3SAakash Sasidharan 	}
87064b12db3SAakash Sasidharan 
87164b12db3SAakash Sasidharan 	return ret;
87264b12db3SAakash Sasidharan }
87364b12db3SAakash Sasidharan 
874bd24488cSAnoob Joseph static int
create_test_conf_from_index(const int index,struct pdcp_test_conf * conf,enum pdcp_test_suite_type suite_type)87564b12db3SAakash Sasidharan create_test_conf_from_index(const int index, struct pdcp_test_conf *conf,
87664b12db3SAakash Sasidharan 			    enum pdcp_test_suite_type suite_type)
877bd24488cSAnoob Joseph {
878bd24488cSAnoob Joseph 	const struct pdcp_testsuite_params *ts_params = &testsuite_params;
879bd24488cSAnoob Joseph 	struct rte_crypto_sym_xform c_xfrm, a_xfrm;
88064b12db3SAakash Sasidharan 	const uint8_t *data, *expected;
881bd24488cSAnoob Joseph 	uint32_t sn, expected_len;
882bd24488cSAnoob Joseph 	int pdcp_hdr_sz;
883bd24488cSAnoob Joseph 
884bd24488cSAnoob Joseph 	memset(conf, 0, sizeof(*conf));
885bd24488cSAnoob Joseph 	memset(&c_xfrm, 0, sizeof(c_xfrm));
886bd24488cSAnoob Joseph 	memset(&a_xfrm, 0, sizeof(a_xfrm));
887bd24488cSAnoob Joseph 
888bd24488cSAnoob Joseph 	conf->entity.sess_mpool = ts_params->sess_pool;
889bd24488cSAnoob Joseph 	conf->entity.cop_pool = ts_params->cop_pool;
890d84ac1b9SVolodymyr Fialko 	conf->entity.ctrl_pdu_pool = ts_params->mbuf_pool;
89164b12db3SAakash Sasidharan 	conf->entity.pdcp_xfrm.bearer = pdcp_test_bearer_get(suite_type, index);
892bd24488cSAnoob Joseph 	conf->entity.pdcp_xfrm.en_ordering = 0;
893bd24488cSAnoob Joseph 	conf->entity.pdcp_xfrm.remove_duplicates = 0;
89464b12db3SAakash Sasidharan 	conf->entity.pdcp_xfrm.domain = pdcp_test_param_domain_get(suite_type, index);
8956f004629SVolodymyr Fialko 	conf->entity.t_reordering = t_reorder_timer;
896bd24488cSAnoob Joseph 
89764b12db3SAakash Sasidharan 	if (pdcp_test_packet_direction_get(suite_type, index) == PDCP_DIR_UPLINK)
898bd24488cSAnoob Joseph 		conf->entity.pdcp_xfrm.pkt_dir = RTE_SECURITY_PDCP_UPLINK;
899bd24488cSAnoob Joseph 	else
900bd24488cSAnoob Joseph 		conf->entity.pdcp_xfrm.pkt_dir = RTE_SECURITY_PDCP_DOWNLINK;
901bd24488cSAnoob Joseph 
90264b12db3SAakash Sasidharan 	conf->entity.pdcp_xfrm.sn_size = pdcp_test_data_sn_size_get(suite_type, index);
903bd24488cSAnoob Joseph 
904bd24488cSAnoob Joseph 	/* Zero initialize unsupported flags */
905bd24488cSAnoob Joseph 	conf->entity.pdcp_xfrm.hfn_threshold = 0;
906bd24488cSAnoob Joseph 	conf->entity.pdcp_xfrm.hfn_ovrd = 0;
90764b12db3SAakash Sasidharan 
90864b12db3SAakash Sasidharan 	conf->entity.pdcp_xfrm.sdap_enabled = (suite_type == PDCP_TEST_SUITE_TY_SDAP);
909bd24488cSAnoob Joseph 
910bd24488cSAnoob Joseph 	c_xfrm.type = RTE_CRYPTO_SYM_XFORM_CIPHER;
91164b12db3SAakash Sasidharan 	c_xfrm.cipher.algo = pdcp_test_param_cipher_alg_get(suite_type, index);
91264b12db3SAakash Sasidharan 	c_xfrm.cipher.key.length = pdcp_test_param_cipher_key_len_get(suite_type, index);
91364b12db3SAakash Sasidharan 	c_xfrm.cipher.key.data = pdcp_test_crypto_key_get(suite_type, index);
914bd24488cSAnoob Joseph 
915bd24488cSAnoob Joseph 	a_xfrm.type = RTE_CRYPTO_SYM_XFORM_AUTH;
916bd24488cSAnoob Joseph 
91764b12db3SAakash Sasidharan 	if (pdcp_test_param_auth_alg_get(suite_type, index) == 0) {
918bd24488cSAnoob Joseph 		conf->is_integrity_protected = false;
919bd24488cSAnoob Joseph 	} else {
92064b12db3SAakash Sasidharan 		a_xfrm.auth.algo = pdcp_test_param_auth_alg_get(suite_type, index);
92164b12db3SAakash Sasidharan 		a_xfrm.auth.key.data = pdcp_test_auth_key_get(suite_type, index);
92264b12db3SAakash Sasidharan 		a_xfrm.auth.key.length = pdcp_test_param_auth_key_len_get(suite_type, index);
923bd24488cSAnoob Joseph 		conf->is_integrity_protected = true;
924bd24488cSAnoob Joseph 	}
925bd24488cSAnoob Joseph 
92664b12db3SAakash Sasidharan 	pdcp_hdr_sz = pdcp_hdr_size_get(pdcp_test_data_sn_size_get(suite_type, index));
927bd24488cSAnoob Joseph 
928bd24488cSAnoob Joseph 	/*
929bd24488cSAnoob Joseph 	 * Uplink means PDCP entity is configured for transmit. Downlink means PDCP entity is
930bd24488cSAnoob Joseph 	 * configured for receive. When integrity protecting is enabled, PDCP always performs
931bd24488cSAnoob Joseph 	 * digest-encrypted or auth-gen-encrypt for uplink (and decrypt-auth-verify for downlink).
932bd24488cSAnoob Joseph 	 * So for uplink, crypto chain would be auth-cipher while for downlink it would be
933bd24488cSAnoob Joseph 	 * cipher-auth.
934bd24488cSAnoob Joseph 	 *
935bd24488cSAnoob Joseph 	 * When integrity protection is not required, xform would be cipher only.
936bd24488cSAnoob Joseph 	 */
937bd24488cSAnoob Joseph 
938bd24488cSAnoob Joseph 	if (conf->is_integrity_protected) {
939bd24488cSAnoob Joseph 		if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK) {
940bd24488cSAnoob Joseph 			conf->entity.crypto_xfrm = &conf->a_xfrm;
941bd24488cSAnoob Joseph 
942bd24488cSAnoob Joseph 			a_xfrm.auth.op = RTE_CRYPTO_AUTH_OP_GENERATE;
943bd24488cSAnoob Joseph 			a_xfrm.next = &conf->c_xfrm;
944bd24488cSAnoob Joseph 
945bd24488cSAnoob Joseph 			c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_ENCRYPT;
946bd24488cSAnoob Joseph 			c_xfrm.next = NULL;
947bd24488cSAnoob Joseph 		} else {
948bd24488cSAnoob Joseph 			conf->entity.crypto_xfrm = &conf->c_xfrm;
949bd24488cSAnoob Joseph 
950bd24488cSAnoob Joseph 			c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_DECRYPT;
951bd24488cSAnoob Joseph 			c_xfrm.next = &conf->a_xfrm;
952bd24488cSAnoob Joseph 
953bd24488cSAnoob Joseph 			a_xfrm.auth.op = RTE_CRYPTO_AUTH_OP_VERIFY;
954bd24488cSAnoob Joseph 			a_xfrm.next = NULL;
955bd24488cSAnoob Joseph 		}
956bd24488cSAnoob Joseph 	} else {
957bd24488cSAnoob Joseph 		conf->entity.crypto_xfrm = &conf->c_xfrm;
958bd24488cSAnoob Joseph 		c_xfrm.next = NULL;
959bd24488cSAnoob Joseph 
960bd24488cSAnoob Joseph 		if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK)
961bd24488cSAnoob Joseph 			c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_ENCRYPT;
962bd24488cSAnoob Joseph 		else
963bd24488cSAnoob Joseph 			c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_DECRYPT;
964bd24488cSAnoob Joseph 	}
965bd24488cSAnoob Joseph 
966bd24488cSAnoob Joseph 	/* Update xforms to match PDCP requirements */
967bd24488cSAnoob Joseph 
968bd24488cSAnoob Joseph 	if ((c_xfrm.cipher.algo == RTE_CRYPTO_CIPHER_AES_CTR) ||
969bd24488cSAnoob Joseph 	    (c_xfrm.cipher.algo == RTE_CRYPTO_CIPHER_ZUC_EEA3 ||
970bd24488cSAnoob Joseph 	    (c_xfrm.cipher.algo == RTE_CRYPTO_CIPHER_SNOW3G_UEA2)))
971bd24488cSAnoob Joseph 		c_xfrm.cipher.iv.length = PDCP_IV_LEN;
972bd24488cSAnoob Joseph 	else
973bd24488cSAnoob Joseph 		c_xfrm.cipher.iv.length = 0;
974bd24488cSAnoob Joseph 
975bd24488cSAnoob Joseph 	if (conf->is_integrity_protected) {
976bd24488cSAnoob Joseph 		if (a_xfrm.auth.algo == RTE_CRYPTO_AUTH_NULL)
977bd24488cSAnoob Joseph 			a_xfrm.auth.digest_length = 0;
978bd24488cSAnoob Joseph 		else
979bd24488cSAnoob Joseph 			a_xfrm.auth.digest_length = RTE_PDCP_MAC_I_LEN;
980bd24488cSAnoob Joseph 
981bd24488cSAnoob Joseph 		if ((a_xfrm.auth.algo == RTE_CRYPTO_AUTH_ZUC_EIA3) ||
982bd24488cSAnoob Joseph 		    (a_xfrm.auth.algo == RTE_CRYPTO_AUTH_SNOW3G_UIA2))
983bd24488cSAnoob Joseph 			a_xfrm.auth.iv.length = PDCP_IV_LEN;
984bd24488cSAnoob Joseph 		else
985bd24488cSAnoob Joseph 			a_xfrm.auth.iv.length = 0;
986bd24488cSAnoob Joseph 	}
987bd24488cSAnoob Joseph 
988bd24488cSAnoob Joseph 	conf->c_xfrm = c_xfrm;
989bd24488cSAnoob Joseph 	conf->a_xfrm = a_xfrm;
990bd24488cSAnoob Joseph 
991bd24488cSAnoob Joseph 	conf->entity.dev_id = (uint8_t)cryptodev_id_get(conf->is_integrity_protected,
992bd24488cSAnoob Joseph 			&conf->c_xfrm, &conf->a_xfrm);
993bd24488cSAnoob Joseph 
99464b12db3SAakash Sasidharan 	if (pdcp_test_param_domain_get(suite_type, index) == RTE_SECURITY_PDCP_MODE_CONTROL ||
99564b12db3SAakash Sasidharan 	    pdcp_test_param_domain_get(suite_type, index) == RTE_SECURITY_PDCP_MODE_DATA) {
99664b12db3SAakash Sasidharan 		data = pdcp_test_data_in_get(suite_type, index);
99764b12db3SAakash Sasidharan 		sn = pdcp_sn_from_raw_get(data, pdcp_test_data_sn_size_get(suite_type, index));
99864b12db3SAakash Sasidharan 		conf->entity.pdcp_xfrm.hfn = pdcp_test_hfn_get(suite_type, index);
999bd24488cSAnoob Joseph 		conf->entity.sn = sn;
1000bd24488cSAnoob Joseph 	}
1001bd24488cSAnoob Joseph 
1002bd24488cSAnoob Joseph 	if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK) {
1003bd24488cSAnoob Joseph #ifdef VEC_DUMP
100464b12db3SAakash Sasidharan 		debug_hexdump(stdout, "Original vector:", pdcp_test_data_in_get(suite_type, index),
100564b12db3SAakash Sasidharan 				pdcp_test_data_in_len_get(suite_type, index));
1006bd24488cSAnoob Joseph #endif
1007bd24488cSAnoob Joseph 		/* Since the vectors available already have PDCP header, trim the same */
100864b12db3SAakash Sasidharan 		conf->input_len = pdcp_test_data_in_len_get(suite_type, index) - pdcp_hdr_sz;
100964b12db3SAakash Sasidharan 		memcpy(conf->input, pdcp_test_data_in_get(suite_type, index) + pdcp_hdr_sz,
101064b12db3SAakash Sasidharan 		       conf->input_len);
1011bd24488cSAnoob Joseph 	} else {
101264b12db3SAakash Sasidharan 		conf->input_len = pdcp_test_data_in_len_get(suite_type, index);
1013bd24488cSAnoob Joseph 
1014bd24488cSAnoob Joseph 		if (conf->is_integrity_protected)
1015bd24488cSAnoob Joseph 			conf->input_len += RTE_PDCP_MAC_I_LEN;
1016bd24488cSAnoob Joseph 
101764b12db3SAakash Sasidharan 		memcpy(conf->input, pdcp_test_data_out_get(suite_type, index), conf->input_len);
1018bd24488cSAnoob Joseph #ifdef VEC_DUMP
1019bd24488cSAnoob Joseph 		debug_hexdump(stdout, "Original vector:", conf->input, conf->input_len);
1020bd24488cSAnoob Joseph #endif
1021bd24488cSAnoob Joseph 	}
1022bd24488cSAnoob Joseph 
1023bd24488cSAnoob Joseph 	if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK)
102464b12db3SAakash Sasidharan 		expected = pdcp_test_data_out_get(suite_type, index);
1025bd24488cSAnoob Joseph 	else
102664b12db3SAakash Sasidharan 		expected = pdcp_test_data_in_get(suite_type, index);
1027bd24488cSAnoob Joseph 
1028bd24488cSAnoob Joseph 	/* Calculate expected packet length */
102964b12db3SAakash Sasidharan 	expected_len = pdcp_test_data_in_len_get(suite_type, index);
1030bd24488cSAnoob Joseph 
1031bd24488cSAnoob Joseph 	/* In DL processing, PDCP header would be stripped */
1032bd24488cSAnoob Joseph 	if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK) {
1033bd24488cSAnoob Joseph 		expected += pdcp_hdr_sz;
1034bd24488cSAnoob Joseph 		expected_len -= pdcp_hdr_sz;
1035bd24488cSAnoob Joseph 	}
1036bd24488cSAnoob Joseph 
1037bd24488cSAnoob Joseph 	/* In UL processing with integrity protection, MAC would be added */
1038bd24488cSAnoob Joseph 	if (conf->is_integrity_protected &&
1039bd24488cSAnoob Joseph 	    conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK)
1040bd24488cSAnoob Joseph 		expected_len += 4;
1041bd24488cSAnoob Joseph 
1042bd24488cSAnoob Joseph 	memcpy(conf->output, expected, expected_len);
1043bd24488cSAnoob Joseph 	conf->output_len = expected_len;
1044bd24488cSAnoob Joseph 
1045bd24488cSAnoob Joseph 	return 0;
1046bd24488cSAnoob Joseph }
1047bd24488cSAnoob Joseph 
1048*d010725bSAakash Sasidharan static void
test_conf_input_data_modify(struct pdcp_test_conf * conf,int inp_len)1049*d010725bSAakash Sasidharan test_conf_input_data_modify(struct pdcp_test_conf *conf, int inp_len)
1050*d010725bSAakash Sasidharan {
1051*d010725bSAakash Sasidharan 	conf->input_len = inp_len;
1052*d010725bSAakash Sasidharan 	memset(conf->input, 0xab, inp_len);
1053*d010725bSAakash Sasidharan }
1054*d010725bSAakash Sasidharan 
1055bd24488cSAnoob Joseph static struct rte_pdcp_entity*
test_entity_create(const struct pdcp_test_conf * t_conf,int * rc)1056bd24488cSAnoob Joseph test_entity_create(const struct pdcp_test_conf *t_conf, int *rc)
1057bd24488cSAnoob Joseph {
1058bd24488cSAnoob Joseph 	struct rte_pdcp_entity *pdcp_entity;
1059bd24488cSAnoob Joseph 	int ret;
1060bd24488cSAnoob Joseph 
1061bd24488cSAnoob Joseph 	if (t_conf->entity.pdcp_xfrm.sn_size != RTE_SECURITY_PDCP_SN_SIZE_12 &&
1062bd24488cSAnoob Joseph 	    t_conf->entity.pdcp_xfrm.sn_size != RTE_SECURITY_PDCP_SN_SIZE_18) {
1063bd24488cSAnoob Joseph 		*rc = -ENOTSUP;
1064bd24488cSAnoob Joseph 		return NULL;
1065bd24488cSAnoob Joseph 	}
1066bd24488cSAnoob Joseph 
1067bd24488cSAnoob Joseph 	if (t_conf->entity.dev_id == CDEV_INVALID_ID) {
1068bd24488cSAnoob Joseph 		RTE_LOG(DEBUG, USER1, "Could not find device with required capabilities\n");
1069bd24488cSAnoob Joseph 		*rc = -ENOTSUP;
1070bd24488cSAnoob Joseph 		return NULL;
1071bd24488cSAnoob Joseph 	}
1072bd24488cSAnoob Joseph 
1073bd24488cSAnoob Joseph 	ret = cryptodev_init(t_conf->entity.dev_id);
1074bd24488cSAnoob Joseph 	if (ret) {
1075bd24488cSAnoob Joseph 		*rc = ret;
1076bd24488cSAnoob Joseph 		RTE_LOG(DEBUG, USER1, "Could not initialize cryptodev\n");
1077bd24488cSAnoob Joseph 		return NULL;
1078bd24488cSAnoob Joseph 	}
1079bd24488cSAnoob Joseph 
1080bd24488cSAnoob Joseph 	rte_errno = 0;
1081bd24488cSAnoob Joseph 
1082bd24488cSAnoob Joseph 	pdcp_entity = rte_pdcp_entity_establish(&t_conf->entity);
1083bd24488cSAnoob Joseph 	if (pdcp_entity == NULL) {
1084bd24488cSAnoob Joseph 		*rc = -rte_errno;
1085bd24488cSAnoob Joseph 		RTE_LOG(DEBUG, USER1, "Could not establish PDCP entity\n");
1086bd24488cSAnoob Joseph 		return NULL;
1087bd24488cSAnoob Joseph 	}
1088bd24488cSAnoob Joseph 
1089bd24488cSAnoob Joseph 	return pdcp_entity;
1090bd24488cSAnoob Joseph }
1091bd24488cSAnoob Joseph 
1092bd24488cSAnoob Joseph static uint16_t
test_process_packets(const struct rte_pdcp_entity * pdcp_entity,uint8_t cdev_id,struct rte_mbuf * in_mb[],uint16_t nb_in,struct rte_mbuf * out_mb[],uint16_t * nb_err)1093bd24488cSAnoob Joseph test_process_packets(const struct rte_pdcp_entity *pdcp_entity, uint8_t cdev_id,
1094bd24488cSAnoob Joseph 		     struct rte_mbuf *in_mb[], uint16_t nb_in,
1095bd24488cSAnoob Joseph 		     struct rte_mbuf *out_mb[], uint16_t *nb_err)
1096bd24488cSAnoob Joseph {
1097bd24488cSAnoob Joseph 	struct rte_crypto_op *cop, *cop_out;
1098bd24488cSAnoob Joseph 	struct rte_pdcp_group grp[1];
1099bd24488cSAnoob Joseph 	uint16_t nb_success, nb_grp;
1100bd24488cSAnoob Joseph 	struct rte_mbuf *mbuf, *mb;
1101bd24488cSAnoob Joseph 
1102bd24488cSAnoob Joseph 	if (nb_in != 1)
1103bd24488cSAnoob Joseph 		return -ENOTSUP;
1104bd24488cSAnoob Joseph 
1105bd24488cSAnoob Joseph 	mbuf = in_mb[0];
1106bd24488cSAnoob Joseph 
1107bd24488cSAnoob Joseph 	nb_success = rte_pdcp_pkt_pre_process(pdcp_entity, &mbuf, &cop_out, 1, nb_err);
1108bd24488cSAnoob Joseph 	if (nb_success != 1 || *nb_err != 0) {
1109bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not pre process PDCP packet\n");
1110bd24488cSAnoob Joseph 		return TEST_FAILED;
1111bd24488cSAnoob Joseph 	}
1112bd24488cSAnoob Joseph 
1113bd24488cSAnoob Joseph #ifdef VEC_DUMP
1114bd24488cSAnoob Joseph 	printf("Pre-processed vector:\n");
1115bd24488cSAnoob Joseph 	rte_pktmbuf_dump(stdout, mbuf, rte_pktmbuf_pkt_len(mbuf));
1116bd24488cSAnoob Joseph #endif
1117bd24488cSAnoob Joseph 
1118bd24488cSAnoob Joseph 	cop = process_crypto_request(cdev_id, cop_out);
1119bd24488cSAnoob Joseph 	if (cop == NULL) {
1120bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not process crypto request\n");
1121bd24488cSAnoob Joseph 		return -EIO;
1122bd24488cSAnoob Joseph 	}
1123bd24488cSAnoob Joseph 
1124bd24488cSAnoob Joseph 	grp[0].id.val = 0;
1125bd24488cSAnoob Joseph 
1126bd24488cSAnoob Joseph 	nb_grp = rte_pdcp_pkt_crypto_group(&cop_out, &mb, grp, 1);
1127bd24488cSAnoob Joseph 	if (nb_grp != 1 || grp[0].cnt != 1) {
1128bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not group PDCP crypto results\n");
1129bd24488cSAnoob Joseph 		return -ENOTRECOVERABLE;
1130bd24488cSAnoob Joseph 	}
1131bd24488cSAnoob Joseph 
1132bd24488cSAnoob Joseph 	if ((uintptr_t)pdcp_entity != grp[0].id.val) {
1133bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "PDCP entity not matching the one from crypto_op\n");
1134bd24488cSAnoob Joseph 		return -ENOTRECOVERABLE;
1135bd24488cSAnoob Joseph 	}
1136bd24488cSAnoob Joseph 
1137bd24488cSAnoob Joseph #ifdef VEC_DUMP
1138bd24488cSAnoob Joseph 	printf("Crypto processed vector:\n");
1139bd24488cSAnoob Joseph 	rte_pktmbuf_dump(stdout, cop->sym->m_dst, rte_pktmbuf_pkt_len(mbuf));
1140bd24488cSAnoob Joseph #endif
1141bd24488cSAnoob Joseph 
1142bd24488cSAnoob Joseph 	return rte_pdcp_pkt_post_process(grp[0].id.ptr, grp[0].m, out_mb, grp[0].cnt, nb_err);
1143bd24488cSAnoob Joseph }
1144bd24488cSAnoob Joseph 
1145bd24488cSAnoob Joseph static struct rte_mbuf*
mbuf_from_data_create(uint8_t * data,uint16_t data_len)1146bd24488cSAnoob Joseph mbuf_from_data_create(uint8_t *data, uint16_t data_len)
1147bd24488cSAnoob Joseph {
1148bd24488cSAnoob Joseph 	const struct pdcp_testsuite_params *ts_params = &testsuite_params;
1149bd24488cSAnoob Joseph 	struct rte_mbuf *mbuf;
1150bd24488cSAnoob Joseph 	uint8_t *input_text;
1151bd24488cSAnoob Joseph 
1152bd24488cSAnoob Joseph 	mbuf = rte_pktmbuf_alloc(ts_params->mbuf_pool);
1153bd24488cSAnoob Joseph 	if (mbuf == NULL) {
1154bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not create mbuf\n");
1155bd24488cSAnoob Joseph 		return NULL;
1156bd24488cSAnoob Joseph 	}
1157bd24488cSAnoob Joseph 
1158bd24488cSAnoob Joseph 	memset(rte_pktmbuf_mtod(mbuf, uint8_t *), 0, rte_pktmbuf_tailroom(mbuf));
1159bd24488cSAnoob Joseph 
1160bd24488cSAnoob Joseph 	input_text = (uint8_t *)rte_pktmbuf_append(mbuf, data_len);
1161bd24488cSAnoob Joseph 	memcpy(input_text, data, data_len);
1162bd24488cSAnoob Joseph 
1163bd24488cSAnoob Joseph 	return mbuf;
1164bd24488cSAnoob Joseph }
1165bd24488cSAnoob Joseph 
1166bd24488cSAnoob Joseph static int
test_attempt_single(struct pdcp_test_conf * t_conf)1167bd24488cSAnoob Joseph test_attempt_single(struct pdcp_test_conf *t_conf)
1168bd24488cSAnoob Joseph {
1169bd24488cSAnoob Joseph 	struct rte_mbuf *mbuf, **out_mb = NULL;
1170bd24488cSAnoob Joseph 	struct rte_pdcp_entity *pdcp_entity;
1171bd24488cSAnoob Joseph 	uint16_t nb_success, nb_err;
1172bd24488cSAnoob Joseph 	int ret = 0, nb_max_out_mb;
1173bd24488cSAnoob Joseph 
1174bd24488cSAnoob Joseph 	pdcp_entity = test_entity_create(t_conf, &ret);
1175bd24488cSAnoob Joseph 	if (pdcp_entity == NULL)
1176bd24488cSAnoob Joseph 		goto exit;
1177bd24488cSAnoob Joseph 
1178bd24488cSAnoob Joseph 	/* Allocate buffer for holding mbufs returned */
1179bd24488cSAnoob Joseph 
1180bd24488cSAnoob Joseph 	/* Max packets that can be cached in entity + burst size */
1181bd24488cSAnoob Joseph 	nb_max_out_mb = pdcp_entity->max_pkt_cache + 1;
1182bd24488cSAnoob Joseph 	out_mb = rte_malloc(NULL, nb_max_out_mb * sizeof(uintptr_t), 0);
1183bd24488cSAnoob Joseph 	if (out_mb == NULL) {
1184bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not allocate buffer for holding out_mb buffers\n");
1185bd24488cSAnoob Joseph 		ret = -ENOMEM;
1186bd24488cSAnoob Joseph 		goto entity_release;
1187bd24488cSAnoob Joseph 	}
1188bd24488cSAnoob Joseph 
1189bd24488cSAnoob Joseph 	mbuf = mbuf_from_data_create(t_conf->input, t_conf->input_len);
1190bd24488cSAnoob Joseph 	if (mbuf == NULL) {
1191bd24488cSAnoob Joseph 		ret = -ENOMEM;
1192bd24488cSAnoob Joseph 		goto entity_release;
1193bd24488cSAnoob Joseph 	}
1194bd24488cSAnoob Joseph 
1195bd24488cSAnoob Joseph #ifdef VEC_DUMP
1196bd24488cSAnoob Joseph 	printf("Adjusted vector:\n");
1197bd24488cSAnoob Joseph 	rte_pktmbuf_dump(stdout, mbuf, t_conf->input_len);
1198bd24488cSAnoob Joseph #endif
1199bd24488cSAnoob Joseph 
1200bd24488cSAnoob Joseph 	nb_success = test_process_packets(pdcp_entity, t_conf->entity.dev_id, &mbuf, 1, out_mb,
1201bd24488cSAnoob Joseph 			&nb_err);
1202bd24488cSAnoob Joseph 	if (nb_success != 1 || nb_err != 0) {
1203bd24488cSAnoob Joseph 		RTE_LOG(ERR, USER1, "Could not process PDCP packet\n");
1204bd24488cSAnoob Joseph 		ret = TEST_FAILED;
1205bd24488cSAnoob Joseph 		goto mbuf_free;
1206bd24488cSAnoob Joseph 	}
1207bd24488cSAnoob Joseph 
1208e98a7ac0SVolodymyr Fialko 	/* If expected output provided - verify, else - store for future use */
1209e98a7ac0SVolodymyr Fialko 	if (t_conf->output_len) {
1210bd24488cSAnoob Joseph 		ret = pdcp_known_vec_verify(mbuf, t_conf->output, t_conf->output_len);
1211bd24488cSAnoob Joseph 		if (ret)
1212bd24488cSAnoob Joseph 			goto mbuf_free;
1213e98a7ac0SVolodymyr Fialko 	} else {
1214e98a7ac0SVolodymyr Fialko 		ret = pktmbuf_read_into(mbuf, t_conf->output, RTE_PDCP_CTRL_PDU_SIZE_MAX);
1215e98a7ac0SVolodymyr Fialko 		if (ret)
1216e98a7ac0SVolodymyr Fialko 			goto mbuf_free;
1217e98a7ac0SVolodymyr Fialko 		t_conf->output_len = mbuf->pkt_len;
1218e98a7ac0SVolodymyr Fialko 	}
1219bd24488cSAnoob Joseph 
1220bd24488cSAnoob Joseph 	ret = rte_pdcp_entity_suspend(pdcp_entity, out_mb);
1221bd24488cSAnoob Joseph 	if (ret) {
1222bd24488cSAnoob Joseph 		RTE_LOG(DEBUG, USER1, "Could not suspend PDCP entity\n");
1223bd24488cSAnoob Joseph 		goto mbuf_free;
1224bd24488cSAnoob Joseph 	}
1225bd24488cSAnoob Joseph 
1226bd24488cSAnoob Joseph mbuf_free:
1227bd24488cSAnoob Joseph 	rte_pktmbuf_free(mbuf);
1228bd24488cSAnoob Joseph entity_release:
1229bd24488cSAnoob Joseph 	rte_pdcp_entity_release(pdcp_entity, out_mb);
1230bd24488cSAnoob Joseph 	rte_free(out_mb);
1231bd24488cSAnoob Joseph exit:
1232bd24488cSAnoob Joseph 	return ret;
1233bd24488cSAnoob Joseph }
1234bd24488cSAnoob Joseph 
1235e98a7ac0SVolodymyr Fialko static void
uplink_to_downlink_convert(const struct pdcp_test_conf * ul_cfg,struct pdcp_test_conf * dl_cfg)1236e98a7ac0SVolodymyr Fialko uplink_to_downlink_convert(const struct pdcp_test_conf *ul_cfg,
1237e98a7ac0SVolodymyr Fialko 			   struct pdcp_test_conf *dl_cfg)
1238e98a7ac0SVolodymyr Fialko {
1239e98a7ac0SVolodymyr Fialko 	assert(ul_cfg->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_UPLINK);
1240e98a7ac0SVolodymyr Fialko 
1241e98a7ac0SVolodymyr Fialko 	memcpy(dl_cfg, ul_cfg, sizeof(*dl_cfg));
1242e98a7ac0SVolodymyr Fialko 	dl_cfg->entity.pdcp_xfrm.pkt_dir = RTE_SECURITY_PDCP_DOWNLINK;
1243e98a7ac0SVolodymyr Fialko 	dl_cfg->entity.reverse_iv_direction = false;
1244e98a7ac0SVolodymyr Fialko 
1245e98a7ac0SVolodymyr Fialko 	if (dl_cfg->is_integrity_protected) {
1246e98a7ac0SVolodymyr Fialko 		dl_cfg->entity.crypto_xfrm = &dl_cfg->c_xfrm;
1247e98a7ac0SVolodymyr Fialko 
1248e98a7ac0SVolodymyr Fialko 		dl_cfg->c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_DECRYPT;
1249e98a7ac0SVolodymyr Fialko 		dl_cfg->c_xfrm.next = &dl_cfg->a_xfrm;
1250e98a7ac0SVolodymyr Fialko 
1251e98a7ac0SVolodymyr Fialko 		dl_cfg->a_xfrm.auth.op = RTE_CRYPTO_AUTH_OP_VERIFY;
1252e98a7ac0SVolodymyr Fialko 		dl_cfg->a_xfrm.next = NULL;
1253e98a7ac0SVolodymyr Fialko 	} else {
1254e98a7ac0SVolodymyr Fialko 		dl_cfg->entity.crypto_xfrm = &dl_cfg->c_xfrm;
1255e98a7ac0SVolodymyr Fialko 		dl_cfg->c_xfrm.next = NULL;
1256e98a7ac0SVolodymyr Fialko 		dl_cfg->c_xfrm.cipher.op = RTE_CRYPTO_CIPHER_OP_DECRYPT;
1257e98a7ac0SVolodymyr Fialko 	}
1258e98a7ac0SVolodymyr Fialko 
1259e98a7ac0SVolodymyr Fialko 	dl_cfg->entity.dev_id = (uint8_t)cryptodev_id_get(dl_cfg->is_integrity_protected,
1260e98a7ac0SVolodymyr Fialko 			&dl_cfg->c_xfrm, &dl_cfg->a_xfrm);
1261e98a7ac0SVolodymyr Fialko 
1262e98a7ac0SVolodymyr Fialko 	memcpy(dl_cfg->input, ul_cfg->output, ul_cfg->output_len);
1263e98a7ac0SVolodymyr Fialko 	dl_cfg->input_len = ul_cfg->output_len;
1264e98a7ac0SVolodymyr Fialko 
1265e98a7ac0SVolodymyr Fialko 	memcpy(dl_cfg->output, ul_cfg->input, ul_cfg->input_len);
1266e98a7ac0SVolodymyr Fialko 	dl_cfg->output_len = ul_cfg->input_len;
1267e98a7ac0SVolodymyr Fialko }
1268e98a7ac0SVolodymyr Fialko 
1269e98a7ac0SVolodymyr Fialko /*
1270e98a7ac0SVolodymyr Fialko  * According to ETSI TS 138 323 V17.1.0, Section 5.2.2.1,
1271e98a7ac0SVolodymyr Fialko  * SN could be divided into following ranges,
1272e98a7ac0SVolodymyr Fialko  * relatively to current value of RX_DELIV state:
1273e98a7ac0SVolodymyr Fialko  * +-------------+-------------+-------------+-------------+
1274e98a7ac0SVolodymyr Fialko  * |  -Outside   |  -Window    |   +Window   |  +Outside   |
1275e98a7ac0SVolodymyr Fialko  * |   (valid)   |  (Invalid)  |   (Valid)   |  (Invalid)  |
1276e98a7ac0SVolodymyr Fialko  * +-------------+-------------^-------------+-------------+
1277e98a7ac0SVolodymyr Fialko  *                             |
1278e98a7ac0SVolodymyr Fialko  *                             v
1279e98a7ac0SVolodymyr Fialko  *                        SN(RX_DELIV)
1280e98a7ac0SVolodymyr Fialko  */
1281e98a7ac0SVolodymyr Fialko enum sn_range_type {
1282e98a7ac0SVolodymyr Fialko 	SN_RANGE_MINUS_OUTSIDE,
1283e98a7ac0SVolodymyr Fialko 	SN_RANGE_MINUS_WINDOW,
1284e98a7ac0SVolodymyr Fialko 	SN_RANGE_PLUS_WINDOW,
1285e98a7ac0SVolodymyr Fialko 	SN_RANGE_PLUS_OUTSIDE,
1286e98a7ac0SVolodymyr Fialko };
1287e98a7ac0SVolodymyr Fialko 
1288e98a7ac0SVolodymyr Fialko #define PDCP_SET_COUNT(hfn, sn, size) ((hfn << size) | (sn & ((1 << size) - 1)))
1289e98a7ac0SVolodymyr Fialko 
1290e98a7ac0SVolodymyr Fialko /*
1291e98a7ac0SVolodymyr Fialko  * Take uplink test case as base, modify RX_DELIV in state and SN in input
1292e98a7ac0SVolodymyr Fialko  */
1293e98a7ac0SVolodymyr Fialko static int
test_sn_range_type(enum sn_range_type type,struct pdcp_test_conf * conf)1294e98a7ac0SVolodymyr Fialko test_sn_range_type(enum sn_range_type type, struct pdcp_test_conf *conf)
1295e98a7ac0SVolodymyr Fialko {
1296e98a7ac0SVolodymyr Fialko 	uint32_t rx_deliv_hfn, rx_deliv_sn, new_hfn, new_sn;
1297e98a7ac0SVolodymyr Fialko 	const int domain = conf->entity.pdcp_xfrm.domain;
1298e98a7ac0SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
1299e98a7ac0SVolodymyr Fialko 	int ret, expected_ret;
1300e98a7ac0SVolodymyr Fialko 
1301e98a7ac0SVolodymyr Fialko 	if (conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
1302e98a7ac0SVolodymyr Fialko 		return TEST_SKIPPED;
1303e98a7ac0SVolodymyr Fialko 
1304e98a7ac0SVolodymyr Fialko 	if (domain != RTE_SECURITY_PDCP_MODE_CONTROL && domain != RTE_SECURITY_PDCP_MODE_DATA)
1305e98a7ac0SVolodymyr Fialko 		return TEST_SKIPPED;
1306e98a7ac0SVolodymyr Fialko 
1307e98a7ac0SVolodymyr Fialko 	const uint32_t sn_size = conf->entity.pdcp_xfrm.sn_size;
1308e98a7ac0SVolodymyr Fialko 	const uint32_t window_size = PDCP_WINDOW_SIZE(sn_size);
1309e98a7ac0SVolodymyr Fialko 	/* Max value of SN that could fit in `sn_size` bits */
1310e98a7ac0SVolodymyr Fialko 	const uint32_t max_sn = (1 << sn_size) - 1;
1311e98a7ac0SVolodymyr Fialko 	const uint32_t shift = (max_sn - window_size) / 2;
1312e98a7ac0SVolodymyr Fialko 	/* Could be any number up to `shift` value */
1313e98a7ac0SVolodymyr Fialko 	const uint32_t default_sn = RTE_MIN(2u, shift);
1314e98a7ac0SVolodymyr Fialko 
1315e98a7ac0SVolodymyr Fialko 	/* Initialize HFN as non zero value, to be able check values before */
1316e98a7ac0SVolodymyr Fialko 	rx_deliv_hfn = 0xa;
1317e98a7ac0SVolodymyr Fialko 
1318e98a7ac0SVolodymyr Fialko 	switch (type) {
1319e98a7ac0SVolodymyr Fialko 	case SN_RANGE_PLUS_WINDOW:
1320e98a7ac0SVolodymyr Fialko 		/* Within window size, HFN stay same */
1321e98a7ac0SVolodymyr Fialko 		new_hfn = rx_deliv_hfn;
1322e98a7ac0SVolodymyr Fialko 		rx_deliv_sn = default_sn;
1323e98a7ac0SVolodymyr Fialko 		new_sn = rx_deliv_sn + 1;
1324e98a7ac0SVolodymyr Fialko 		expected_ret = TEST_SUCCESS;
1325e98a7ac0SVolodymyr Fialko 		break;
1326e98a7ac0SVolodymyr Fialko 	case SN_RANGE_MINUS_WINDOW:
1327e98a7ac0SVolodymyr Fialko 		/* Within window size, HFN stay same */
1328e98a7ac0SVolodymyr Fialko 		new_hfn = rx_deliv_hfn;
1329e98a7ac0SVolodymyr Fialko 		rx_deliv_sn = default_sn;
1330e98a7ac0SVolodymyr Fialko 		new_sn = rx_deliv_sn - 1;
1331e98a7ac0SVolodymyr Fialko 		expected_ret = TEST_FAILED;
1332e98a7ac0SVolodymyr Fialko 		break;
1333e98a7ac0SVolodymyr Fialko 	case SN_RANGE_PLUS_OUTSIDE:
1334e98a7ac0SVolodymyr Fialko 		/* RCVD_SN >= SN(RX_DELIV) + Window_Size */
1335e98a7ac0SVolodymyr Fialko 		new_hfn = rx_deliv_hfn - 1;
1336e98a7ac0SVolodymyr Fialko 		rx_deliv_sn = default_sn;
1337e98a7ac0SVolodymyr Fialko 		new_sn = rx_deliv_sn + window_size;
1338e98a7ac0SVolodymyr Fialko 		expected_ret = TEST_FAILED;
1339e98a7ac0SVolodymyr Fialko 		break;
1340e98a7ac0SVolodymyr Fialko 	case SN_RANGE_MINUS_OUTSIDE:
1341e98a7ac0SVolodymyr Fialko 		/* RCVD_SN < SN(RX_DELIV) - Window_Size */
1342e98a7ac0SVolodymyr Fialko 		new_hfn = rx_deliv_hfn + 1;
1343e98a7ac0SVolodymyr Fialko 		rx_deliv_sn = window_size + default_sn;
1344e98a7ac0SVolodymyr Fialko 		new_sn = rx_deliv_sn - window_size - 1;
1345e98a7ac0SVolodymyr Fialko 		expected_ret = TEST_SUCCESS;
1346e98a7ac0SVolodymyr Fialko 		break;
1347e98a7ac0SVolodymyr Fialko 	default:
1348e98a7ac0SVolodymyr Fialko 		return TEST_FAILED;
1349e98a7ac0SVolodymyr Fialko 	}
1350e98a7ac0SVolodymyr Fialko 
1351e98a7ac0SVolodymyr Fialko 	/* Configure Uplink to generate expected, encrypted packet */
1352e98a7ac0SVolodymyr Fialko 	pdcp_sn_to_raw_set(conf->input, new_sn, conf->entity.pdcp_xfrm.sn_size);
1353d84ac1b9SVolodymyr Fialko 	conf->entity.out_of_order_delivery = true;
1354e98a7ac0SVolodymyr Fialko 	conf->entity.reverse_iv_direction = true;
1355e98a7ac0SVolodymyr Fialko 	conf->entity.pdcp_xfrm.hfn = new_hfn;
1356e98a7ac0SVolodymyr Fialko 	conf->entity.sn = new_sn;
1357e98a7ac0SVolodymyr Fialko 	conf->output_len = 0;
1358e98a7ac0SVolodymyr Fialko 	ret = test_attempt_single(conf);
1359e98a7ac0SVolodymyr Fialko 	if (ret != TEST_SUCCESS)
1360e98a7ac0SVolodymyr Fialko 		return ret;
1361e98a7ac0SVolodymyr Fialko 
1362e98a7ac0SVolodymyr Fialko 	/* Flip configuration to downlink */
1363e98a7ac0SVolodymyr Fialko 	uplink_to_downlink_convert(conf, &dl_conf);
1364e98a7ac0SVolodymyr Fialko 
1365e98a7ac0SVolodymyr Fialko 	/* Modify the rx_deliv to verify the expected behaviour */
1366e98a7ac0SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = rx_deliv_hfn;
1367e98a7ac0SVolodymyr Fialko 	dl_conf.entity.sn = rx_deliv_sn;
1368e98a7ac0SVolodymyr Fialko 	ret = test_attempt_single(&dl_conf);
1369e98a7ac0SVolodymyr Fialko 	if ((ret == TEST_SKIPPED) || (ret == -ENOTSUP))
1370e98a7ac0SVolodymyr Fialko 		return ret;
1371e98a7ac0SVolodymyr Fialko 
1372e98a7ac0SVolodymyr Fialko 	TEST_ASSERT_EQUAL(ret, expected_ret, "Unexpected result");
1373e98a7ac0SVolodymyr Fialko 
1374e98a7ac0SVolodymyr Fialko 	return TEST_SUCCESS;
1375e98a7ac0SVolodymyr Fialko }
1376e98a7ac0SVolodymyr Fialko 
1377e98a7ac0SVolodymyr Fialko static int
test_sn_plus_window(struct pdcp_test_conf * t_conf)1378e98a7ac0SVolodymyr Fialko test_sn_plus_window(struct pdcp_test_conf *t_conf)
1379e98a7ac0SVolodymyr Fialko {
1380e98a7ac0SVolodymyr Fialko 	return test_sn_range_type(SN_RANGE_PLUS_WINDOW, t_conf);
1381e98a7ac0SVolodymyr Fialko }
1382e98a7ac0SVolodymyr Fialko 
1383e98a7ac0SVolodymyr Fialko static int
test_sn_minus_window(struct pdcp_test_conf * t_conf)1384e98a7ac0SVolodymyr Fialko test_sn_minus_window(struct pdcp_test_conf *t_conf)
1385e98a7ac0SVolodymyr Fialko {
1386e98a7ac0SVolodymyr Fialko 	return test_sn_range_type(SN_RANGE_MINUS_WINDOW, t_conf);
1387e98a7ac0SVolodymyr Fialko }
1388e98a7ac0SVolodymyr Fialko 
1389e98a7ac0SVolodymyr Fialko static int
test_sn_plus_outside(struct pdcp_test_conf * t_conf)1390e98a7ac0SVolodymyr Fialko test_sn_plus_outside(struct pdcp_test_conf *t_conf)
1391e98a7ac0SVolodymyr Fialko {
1392e98a7ac0SVolodymyr Fialko 	return test_sn_range_type(SN_RANGE_PLUS_OUTSIDE, t_conf);
1393e98a7ac0SVolodymyr Fialko }
1394e98a7ac0SVolodymyr Fialko 
1395e98a7ac0SVolodymyr Fialko static int
test_sn_minus_outside(struct pdcp_test_conf * t_conf)1396e98a7ac0SVolodymyr Fialko test_sn_minus_outside(struct pdcp_test_conf *t_conf)
1397e98a7ac0SVolodymyr Fialko {
1398e98a7ac0SVolodymyr Fialko 	return test_sn_range_type(SN_RANGE_MINUS_OUTSIDE, t_conf);
1399e98a7ac0SVolodymyr Fialko }
1400e98a7ac0SVolodymyr Fialko 
1401d84ac1b9SVolodymyr Fialko static struct rte_mbuf *
generate_packet_for_dl_with_sn(struct pdcp_test_conf ul_conf,uint32_t count)1402d84ac1b9SVolodymyr Fialko generate_packet_for_dl_with_sn(struct pdcp_test_conf ul_conf, uint32_t count)
1403d84ac1b9SVolodymyr Fialko {
1404d84ac1b9SVolodymyr Fialko 	enum rte_security_pdcp_sn_size sn_size = ul_conf.entity.pdcp_xfrm.sn_size;
1405d84ac1b9SVolodymyr Fialko 	int ret;
1406d84ac1b9SVolodymyr Fialko 
1407d84ac1b9SVolodymyr Fialko 	ul_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(count, sn_size);
1408d84ac1b9SVolodymyr Fialko 	ul_conf.entity.sn = pdcp_sn_from_count_get(count, sn_size);
1409d84ac1b9SVolodymyr Fialko 	ul_conf.entity.out_of_order_delivery = true;
1410d84ac1b9SVolodymyr Fialko 	ul_conf.entity.reverse_iv_direction = true;
1411d84ac1b9SVolodymyr Fialko 	ul_conf.output_len = 0;
1412d84ac1b9SVolodymyr Fialko 
1413d84ac1b9SVolodymyr Fialko 	ret = test_attempt_single(&ul_conf);
1414d84ac1b9SVolodymyr Fialko 	if (ret != TEST_SUCCESS)
1415d84ac1b9SVolodymyr Fialko 		return NULL;
1416d84ac1b9SVolodymyr Fialko 
1417d84ac1b9SVolodymyr Fialko 	return mbuf_from_data_create(ul_conf.output, ul_conf.output_len);
1418d84ac1b9SVolodymyr Fialko }
1419d84ac1b9SVolodymyr Fialko 
1420d84ac1b9SVolodymyr Fialko static bool
array_asc_sorted_check(struct rte_mbuf * m[],uint32_t len,enum rte_security_pdcp_sn_size sn_size)1421d84ac1b9SVolodymyr Fialko array_asc_sorted_check(struct rte_mbuf *m[], uint32_t len, enum rte_security_pdcp_sn_size sn_size)
1422d84ac1b9SVolodymyr Fialko {
1423d84ac1b9SVolodymyr Fialko 	uint32_t i;
1424d84ac1b9SVolodymyr Fialko 
1425d84ac1b9SVolodymyr Fialko 	if (len < 2)
1426d84ac1b9SVolodymyr Fialko 		return true;
1427d84ac1b9SVolodymyr Fialko 
1428d84ac1b9SVolodymyr Fialko 	for (i = 0; i < (len - 1); i++) {
1429d84ac1b9SVolodymyr Fialko 		if (pdcp_sn_from_raw_get(rte_pktmbuf_mtod(m[i], void *), sn_size) >
1430d84ac1b9SVolodymyr Fialko 		    pdcp_sn_from_raw_get(rte_pktmbuf_mtod(m[i + 1], void *), sn_size))
1431d84ac1b9SVolodymyr Fialko 			return false;
1432d84ac1b9SVolodymyr Fialko 	}
1433d84ac1b9SVolodymyr Fialko 
1434d84ac1b9SVolodymyr Fialko 	return true;
1435d84ac1b9SVolodymyr Fialko }
1436d84ac1b9SVolodymyr Fialko 
1437d84ac1b9SVolodymyr Fialko static int
test_reorder_gap_fill(struct pdcp_test_conf * ul_conf)1438d84ac1b9SVolodymyr Fialko test_reorder_gap_fill(struct pdcp_test_conf *ul_conf)
1439d84ac1b9SVolodymyr Fialko {
1440d84ac1b9SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
1441d84ac1b9SVolodymyr Fialko 	struct rte_mbuf *m0 = NULL, *m1 = NULL, *out_mb[2] = {0};
1442d84ac1b9SVolodymyr Fialko 	uint16_t nb_success = 0, nb_err = 0;
1443d84ac1b9SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
1444d84ac1b9SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
1445d84ac1b9SVolodymyr Fialko 	int ret = TEST_FAILED, nb_out;
1446d84ac1b9SVolodymyr Fialko 	uint8_t cdev_id;
1447d84ac1b9SVolodymyr Fialko 
1448d84ac1b9SVolodymyr Fialko 	const int start_count = 0;
1449d84ac1b9SVolodymyr Fialko 
1450d84ac1b9SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
1451d84ac1b9SVolodymyr Fialko 		return TEST_SKIPPED;
1452d84ac1b9SVolodymyr Fialko 
1453d84ac1b9SVolodymyr Fialko 	/* Create configuration for actual testing */
1454d84ac1b9SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
1455d84ac1b9SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(start_count, sn_size);
1456d84ac1b9SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(start_count, sn_size);
1457d84ac1b9SVolodymyr Fialko 
1458d84ac1b9SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
1459d84ac1b9SVolodymyr Fialko 	if (pdcp_entity == NULL)
1460d84ac1b9SVolodymyr Fialko 		return ret;
1461d84ac1b9SVolodymyr Fialko 
1462d84ac1b9SVolodymyr Fialko 	cdev_id = dl_conf.entity.dev_id;
1463d84ac1b9SVolodymyr Fialko 
1464d84ac1b9SVolodymyr Fialko 	/* Send packet with SN > RX_DELIV to create a gap */
1465d84ac1b9SVolodymyr Fialko 	m1 = generate_packet_for_dl_with_sn(*ul_conf, start_count + 1);
1466d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m1 != NULL, exit, "Could not allocate buffer for packet\n");
1467d84ac1b9SVolodymyr Fialko 
1468d84ac1b9SVolodymyr Fialko 	/* Buffered packets after insert [NULL, m1] */
1469d84ac1b9SVolodymyr Fialko 	nb_success = test_process_packets(pdcp_entity, cdev_id, &m1, 1, out_mb, &nb_err);
1470d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet process\n");
1471d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_success == 0, exit, "Packet was not buffered as expected\n");
1472d84ac1b9SVolodymyr Fialko 	m1 = NULL; /* Packet was moved to PDCP lib */
1473d84ac1b9SVolodymyr Fialko 
1474d84ac1b9SVolodymyr Fialko 	/* Generate packet to fill the existing gap */
1475d84ac1b9SVolodymyr Fialko 	m0 = generate_packet_for_dl_with_sn(*ul_conf, start_count);
1476d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m0 != NULL, exit, "Could not allocate buffer for packet\n");
1477d84ac1b9SVolodymyr Fialko 
1478d84ac1b9SVolodymyr Fialko 	/*
1479d84ac1b9SVolodymyr Fialko 	 * Buffered packets after insert [m0, m1]
1480d84ac1b9SVolodymyr Fialko 	 * Gap filled, all packets should be returned
1481d84ac1b9SVolodymyr Fialko 	 */
1482d84ac1b9SVolodymyr Fialko 	nb_success = test_process_packets(pdcp_entity, cdev_id, &m0, 1, out_mb, &nb_err);
1483d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet process\n");
1484d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_success == 2, exit,
1485d84ac1b9SVolodymyr Fialko 			"Packet count mismatch (received: %i, expected: 2)\n", nb_success);
1486d84ac1b9SVolodymyr Fialko 	m0 = NULL; /* Packet was moved to out_mb */
1487d84ac1b9SVolodymyr Fialko 
1488d84ac1b9SVolodymyr Fialko 	/* Check that packets in correct order */
1489d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(array_asc_sorted_check(out_mb, nb_success, sn_size), exit,
1490d84ac1b9SVolodymyr Fialko 			"Error occurred during packet drain\n");
14916f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(testsuite_params.timer_is_running == false, exit,
14926f004629SVolodymyr Fialko 			"Timer should be stopped after full drain\n");
1493d84ac1b9SVolodymyr Fialko 
1494d84ac1b9SVolodymyr Fialko 	ret = TEST_SUCCESS;
1495d84ac1b9SVolodymyr Fialko exit:
1496d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free(m0);
1497d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free(m1);
1498d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_success);
1499d84ac1b9SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
1500d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
1501d84ac1b9SVolodymyr Fialko 	return ret;
1502d84ac1b9SVolodymyr Fialko }
1503d84ac1b9SVolodymyr Fialko 
1504d84ac1b9SVolodymyr Fialko static int
test_reorder_gap_in_reorder_buffer(const struct pdcp_test_conf * ul_conf)150563c6edc5SVolodymyr Fialko test_reorder_gap_in_reorder_buffer(const struct pdcp_test_conf *ul_conf)
150663c6edc5SVolodymyr Fialko {
150763c6edc5SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
150863c6edc5SVolodymyr Fialko 	struct rte_mbuf *m = NULL, *out_mb[2] = {0};
150963c6edc5SVolodymyr Fialko 	uint16_t nb_success = 0, nb_err = 0;
151063c6edc5SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
151163c6edc5SVolodymyr Fialko 	int ret = TEST_FAILED, nb_out, i;
151263c6edc5SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
151363c6edc5SVolodymyr Fialko 	uint8_t cdev_id;
151463c6edc5SVolodymyr Fialko 
151563c6edc5SVolodymyr Fialko 	const int start_count = 0;
151663c6edc5SVolodymyr Fialko 
151763c6edc5SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
151863c6edc5SVolodymyr Fialko 		return TEST_SKIPPED;
151963c6edc5SVolodymyr Fialko 
152063c6edc5SVolodymyr Fialko 	/* Create configuration for actual testing */
152163c6edc5SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
152263c6edc5SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(start_count, sn_size);
152363c6edc5SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(start_count, sn_size);
152463c6edc5SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
152563c6edc5SVolodymyr Fialko 	if (pdcp_entity == NULL)
152663c6edc5SVolodymyr Fialko 		return ret;
152763c6edc5SVolodymyr Fialko 
152863c6edc5SVolodymyr Fialko 	cdev_id = dl_conf.entity.dev_id;
152963c6edc5SVolodymyr Fialko 
153063c6edc5SVolodymyr Fialko 	/* Create two gaps [NULL, m1, NULL, m3]*/
153163c6edc5SVolodymyr Fialko 	for (i = 0; i < 2; i++) {
153263c6edc5SVolodymyr Fialko 		m = generate_packet_for_dl_with_sn(*ul_conf, start_count + 2 * i + 1);
153363c6edc5SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(m != NULL, exit, "Could not allocate buffer for packet\n");
153463c6edc5SVolodymyr Fialko 		nb_success = test_process_packets(pdcp_entity, cdev_id, &m, 1, out_mb, &nb_err);
153563c6edc5SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet process\n");
153663c6edc5SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_success == 0, exit, "Packet was not buffered as expected\n");
153763c6edc5SVolodymyr Fialko 		m = NULL; /* Packet was moved to PDCP lib */
153863c6edc5SVolodymyr Fialko 	}
153963c6edc5SVolodymyr Fialko 
154063c6edc5SVolodymyr Fialko 	/* Generate packet to fill the first gap */
154163c6edc5SVolodymyr Fialko 	m = generate_packet_for_dl_with_sn(*ul_conf, start_count);
154263c6edc5SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m != NULL, exit, "Could not allocate buffer for packet\n");
154363c6edc5SVolodymyr Fialko 
154463c6edc5SVolodymyr Fialko 	/*
154563c6edc5SVolodymyr Fialko 	 * Buffered packets after insert [m0, m1, NULL, m3]
154663c6edc5SVolodymyr Fialko 	 * Only first gap should be filled, timer should be restarted for second gap
154763c6edc5SVolodymyr Fialko 	 */
154863c6edc5SVolodymyr Fialko 	nb_success = test_process_packets(pdcp_entity, cdev_id, &m, 1, out_mb, &nb_err);
154963c6edc5SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet process\n");
155063c6edc5SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_success == 2, exit,
155163c6edc5SVolodymyr Fialko 			"Packet count mismatch (received: %i, expected: 2)\n", nb_success);
155263c6edc5SVolodymyr Fialko 	m = NULL;
155363c6edc5SVolodymyr Fialko 	/* Check that packets in correct order */
155463c6edc5SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(array_asc_sorted_check(out_mb, nb_success, sn_size),
155563c6edc5SVolodymyr Fialko 			exit, "Error occurred during packet drain\n");
155663c6edc5SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(testsuite_params.timer_is_running == true, exit,
155763c6edc5SVolodymyr Fialko 			"Timer should be restarted after partial drain");
155863c6edc5SVolodymyr Fialko 
155963c6edc5SVolodymyr Fialko 
156063c6edc5SVolodymyr Fialko 	ret = TEST_SUCCESS;
156163c6edc5SVolodymyr Fialko exit:
156263c6edc5SVolodymyr Fialko 	rte_pktmbuf_free(m);
156363c6edc5SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_success);
156463c6edc5SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
156563c6edc5SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
156663c6edc5SVolodymyr Fialko 	return ret;
156763c6edc5SVolodymyr Fialko }
156863c6edc5SVolodymyr Fialko 
156963c6edc5SVolodymyr Fialko static int
test_reorder_buffer_full_window_size_sn_12(const struct pdcp_test_conf * ul_conf)1570d84ac1b9SVolodymyr Fialko test_reorder_buffer_full_window_size_sn_12(const struct pdcp_test_conf *ul_conf)
1571d84ac1b9SVolodymyr Fialko {
1572d84ac1b9SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
1573d84ac1b9SVolodymyr Fialko 	const uint32_t window_size = PDCP_WINDOW_SIZE(sn_size);
1574d84ac1b9SVolodymyr Fialko 	struct rte_mbuf *m1 = NULL, **out_mb = NULL;
1575d84ac1b9SVolodymyr Fialko 	uint16_t nb_success = 0, nb_err = 0;
1576d84ac1b9SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
1577d84ac1b9SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
1578d84ac1b9SVolodymyr Fialko 	const int rx_deliv = 0;
1579d84ac1b9SVolodymyr Fialko 	int ret = TEST_FAILED;
1580d84ac1b9SVolodymyr Fialko 	size_t i, nb_out;
1581d84ac1b9SVolodymyr Fialko 	uint8_t cdev_id;
1582d84ac1b9SVolodymyr Fialko 
1583d84ac1b9SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK ||
1584d84ac1b9SVolodymyr Fialko 		sn_size != RTE_SECURITY_PDCP_SN_SIZE_12)
1585d84ac1b9SVolodymyr Fialko 		return TEST_SKIPPED;
1586d84ac1b9SVolodymyr Fialko 
1587d84ac1b9SVolodymyr Fialko 	/* Create configuration for actual testing */
1588d84ac1b9SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
1589d84ac1b9SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(rx_deliv, sn_size);
1590d84ac1b9SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(rx_deliv, sn_size);
1591d84ac1b9SVolodymyr Fialko 
1592d84ac1b9SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
1593d84ac1b9SVolodymyr Fialko 	if (pdcp_entity == NULL)
1594d84ac1b9SVolodymyr Fialko 		return ret;
1595d84ac1b9SVolodymyr Fialko 
1596d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(pdcp_entity->max_pkt_cache >= window_size, exit,
1597d84ac1b9SVolodymyr Fialko 			"PDCP max packet cache is too small");
1598d84ac1b9SVolodymyr Fialko 	cdev_id = dl_conf.entity.dev_id;
1599d84ac1b9SVolodymyr Fialko 	out_mb = rte_zmalloc(NULL, pdcp_entity->max_pkt_cache * sizeof(uintptr_t), 0);
1600d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(out_mb != NULL, exit,
1601d84ac1b9SVolodymyr Fialko 			"Could not allocate buffer for holding out_mb buffers\n");
1602d84ac1b9SVolodymyr Fialko 
1603d84ac1b9SVolodymyr Fialko 	/* Send packets with SN > RX_DELIV to create a gap */
1604d84ac1b9SVolodymyr Fialko 	for (i = rx_deliv + 1; i < window_size; i++) {
1605d84ac1b9SVolodymyr Fialko 		m1 = generate_packet_for_dl_with_sn(*ul_conf, i);
1606d84ac1b9SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(m1 != NULL, exit, "Could not allocate buffer for packet\n");
1607d84ac1b9SVolodymyr Fialko 		/* Buffered packets after insert [NULL, m1] */
1608d84ac1b9SVolodymyr Fialko 		nb_success = test_process_packets(pdcp_entity, cdev_id, &m1, 1, out_mb, &nb_err);
1609d84ac1b9SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet buffering\n");
1610d84ac1b9SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_success == 0, exit, "Packet was not buffered as expected\n");
1611d84ac1b9SVolodymyr Fialko 	}
1612d84ac1b9SVolodymyr Fialko 
1613d84ac1b9SVolodymyr Fialko 	m1 = generate_packet_for_dl_with_sn(*ul_conf, rx_deliv);
1614d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m1 != NULL, exit, "Could not allocate buffer for packet\n");
1615d84ac1b9SVolodymyr Fialko 	/* Insert missing packet */
1616d84ac1b9SVolodymyr Fialko 	nb_success = test_process_packets(pdcp_entity, cdev_id, &m1, 1, out_mb, &nb_err);
1617d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet buffering\n");
1618d84ac1b9SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_success == window_size, exit,
1619d84ac1b9SVolodymyr Fialko 			"Packet count mismatch (received: %i, expected: %i)\n",
1620d84ac1b9SVolodymyr Fialko 			nb_success, window_size);
1621d84ac1b9SVolodymyr Fialko 	m1 = NULL;
1622d84ac1b9SVolodymyr Fialko 
1623d84ac1b9SVolodymyr Fialko 	ret = TEST_SUCCESS;
1624d84ac1b9SVolodymyr Fialko exit:
1625d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free(m1);
1626d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_success);
1627d84ac1b9SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
1628d84ac1b9SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
1629d84ac1b9SVolodymyr Fialko 	rte_free(out_mb);
1630d84ac1b9SVolodymyr Fialko 	return ret;
1631d84ac1b9SVolodymyr Fialko }
1632d84ac1b9SVolodymyr Fialko 
16336f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
16346f004629SVolodymyr Fialko static void
event_timer_start_cb(void * timer,void * args)16356f004629SVolodymyr Fialko event_timer_start_cb(void *timer, void *args)
16366f004629SVolodymyr Fialko {
16376f004629SVolodymyr Fialko 	struct rte_event_timer *evtims = args;
16386f004629SVolodymyr Fialko 	int ret = 0;
16396f004629SVolodymyr Fialko 
16406f004629SVolodymyr Fialko 	ret = rte_event_timer_arm_burst(timer, &evtims, 1);
16416f004629SVolodymyr Fialko 	assert(ret == 1);
16426f004629SVolodymyr Fialko }
16436f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
16446f004629SVolodymyr Fialko 
16456f004629SVolodymyr Fialko static int
test_expiry_with_event_timer(const struct pdcp_test_conf * ul_conf)16466f004629SVolodymyr Fialko test_expiry_with_event_timer(const struct pdcp_test_conf *ul_conf)
16476f004629SVolodymyr Fialko {
16486f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
16496f004629SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
16506f004629SVolodymyr Fialko 	struct rte_mbuf *m1 = NULL, *out_mb[1] = {0};
16516f004629SVolodymyr Fialko 	uint16_t n = 0, nb_err = 0, nb_try = 5;
16526f004629SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
16536f004629SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
16546f004629SVolodymyr Fialko 	int ret = TEST_FAILED, nb_out;
16556f004629SVolodymyr Fialko 	struct rte_event event;
16566f004629SVolodymyr Fialko 
16576f004629SVolodymyr Fialko 	const int start_count = 0;
16586f004629SVolodymyr Fialko 	struct rte_event_timer evtim = {
16596f004629SVolodymyr Fialko 		.ev.op = RTE_EVENT_OP_NEW,
16606f004629SVolodymyr Fialko 		.ev.queue_id = TEST_EV_QUEUE_ID,
16616f004629SVolodymyr Fialko 		.ev.sched_type = RTE_SCHED_TYPE_ATOMIC,
16626f004629SVolodymyr Fialko 		.ev.priority = RTE_EVENT_DEV_PRIORITY_NORMAL,
16636f004629SVolodymyr Fialko 		.ev.event_type =  RTE_EVENT_TYPE_TIMER,
16646f004629SVolodymyr Fialko 		.state = RTE_EVENT_TIMER_NOT_ARMED,
16656f004629SVolodymyr Fialko 		.timeout_ticks = 1,
16666f004629SVolodymyr Fialko 	};
16676f004629SVolodymyr Fialko 
16686f004629SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
16696f004629SVolodymyr Fialko 		return TEST_SKIPPED;
16706f004629SVolodymyr Fialko 
16716f004629SVolodymyr Fialko 	/* Create configuration for actual testing */
16726f004629SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
16736f004629SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(start_count, sn_size);
16746f004629SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(start_count, sn_size);
16756f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.args = &evtim;
16766f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.timer = testsuite_params.timdev;
16776f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.start = event_timer_start_cb;
16786f004629SVolodymyr Fialko 
16796f004629SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
16806f004629SVolodymyr Fialko 	if (pdcp_entity == NULL)
16816f004629SVolodymyr Fialko 		return ret;
16826f004629SVolodymyr Fialko 
16836f004629SVolodymyr Fialko 	evtim.ev.event_ptr = pdcp_entity;
16846f004629SVolodymyr Fialko 
16856f004629SVolodymyr Fialko 	/* Send packet with SN > RX_DELIV to create a gap */
16866f004629SVolodymyr Fialko 	m1 = generate_packet_for_dl_with_sn(*ul_conf, start_count + 1);
16876f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m1 != NULL, exit, "Could not allocate buffer for packet\n");
16886f004629SVolodymyr Fialko 
16896f004629SVolodymyr Fialko 	/* Buffered packets after insert [NULL, m1] */
16906f004629SVolodymyr Fialko 	n = test_process_packets(pdcp_entity, dl_conf.entity.dev_id, &m1, 1, out_mb, &nb_err);
16916f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet buffering\n");
16926f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(n == 0, exit, "Packet was not buffered as expected\n");
16936f004629SVolodymyr Fialko 
16946f004629SVolodymyr Fialko 	m1 = NULL; /* Packet was moved to PDCP lib */
16956f004629SVolodymyr Fialko 
16966f004629SVolodymyr Fialko 	n = rte_event_dequeue_burst(testsuite_params.evdev, TEST_EV_PORT_ID, &event, 1, 0);
16976f004629SVolodymyr Fialko 	while (n != 1) {
16986f004629SVolodymyr Fialko 		rte_delay_us(testsuite_params.min_resolution_ns / 1000);
16996f004629SVolodymyr Fialko 		n = rte_event_dequeue_burst(testsuite_params.evdev, TEST_EV_PORT_ID, &event, 1, 0);
1700bf6b85baSVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_try > 0, exit,
17016f004629SVolodymyr Fialko 				"Dequeued unexpected timer expiry event: %i\n", n);
1702bf6b85baSVolodymyr Fialko 		nb_try--;
17036f004629SVolodymyr Fialko 	}
17046f004629SVolodymyr Fialko 
17056f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(event.event_type == RTE_EVENT_TYPE_TIMER, exit, "Unexpected event type\n");
17066f004629SVolodymyr Fialko 
17076f004629SVolodymyr Fialko 	/* Handle expiry event */
17086f004629SVolodymyr Fialko 	n = rte_pdcp_t_reordering_expiry_handle(event.event_ptr, out_mb);
17096f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(n == 1, exit, "Unexpected number of expired packets :%i\n", n);
17106f004629SVolodymyr Fialko 
17116f004629SVolodymyr Fialko 	ret = TEST_SUCCESS;
17126f004629SVolodymyr Fialko exit:
17136f004629SVolodymyr Fialko 	rte_pktmbuf_free(m1);
17146f004629SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, n);
17156f004629SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
17166f004629SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
17176f004629SVolodymyr Fialko 	return ret;
17186f004629SVolodymyr Fialko #else
17196f004629SVolodymyr Fialko 	RTE_SET_USED(ul_conf);
17206f004629SVolodymyr Fialko 	return TEST_SKIPPED;
17216f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
17226f004629SVolodymyr Fialko }
17236f004629SVolodymyr Fialko 
17246f004629SVolodymyr Fialko static void
test_rte_timer_expiry_handle(struct rte_timer * timer_handle,void * arg)17256f004629SVolodymyr Fialko test_rte_timer_expiry_handle(struct rte_timer *timer_handle, void *arg)
17266f004629SVolodymyr Fialko {
17276f004629SVolodymyr Fialko 	struct test_rte_timer_args *timer_data = arg;
17286f004629SVolodymyr Fialko 	struct rte_mbuf *out_mb[1] = {0};
17296f004629SVolodymyr Fialko 	uint16_t n;
17306f004629SVolodymyr Fialko 
17316f004629SVolodymyr Fialko 	RTE_SET_USED(timer_handle);
17326f004629SVolodymyr Fialko 
17336f004629SVolodymyr Fialko 	n = rte_pdcp_t_reordering_expiry_handle(timer_data->pdcp_entity, out_mb);
17346f004629SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, n);
17356f004629SVolodymyr Fialko 
17366f004629SVolodymyr Fialko 	timer_data->status =  n == 1 ? n : -1;
17376f004629SVolodymyr Fialko }
17386f004629SVolodymyr Fialko 
17396f004629SVolodymyr Fialko static void
test_rte_timer_start_cb(void * timer,void * args)17406f004629SVolodymyr Fialko test_rte_timer_start_cb(void *timer, void *args)
17416f004629SVolodymyr Fialko {
17426f004629SVolodymyr Fialko 	rte_timer_reset_sync(timer, 1, SINGLE, rte_lcore_id(), test_rte_timer_expiry_handle, args);
17436f004629SVolodymyr Fialko }
17446f004629SVolodymyr Fialko 
17456f004629SVolodymyr Fialko static int
test_expiry_with_rte_timer(const struct pdcp_test_conf * ul_conf)17466f004629SVolodymyr Fialko test_expiry_with_rte_timer(const struct pdcp_test_conf *ul_conf)
17476f004629SVolodymyr Fialko {
17486f004629SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
17496f004629SVolodymyr Fialko 	struct rte_mbuf *m1 = NULL, *out_mb[1] = {0};
17506f004629SVolodymyr Fialko 	uint16_t n = 0, nb_err = 0, nb_try = 5;
17516f004629SVolodymyr Fialko 	struct test_rte_timer_args timer_args;
17526f004629SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
17536f004629SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
17546f004629SVolodymyr Fialko 	int ret = TEST_FAILED, nb_out;
17556f004629SVolodymyr Fialko 	struct rte_timer timer = {0};
17566f004629SVolodymyr Fialko 
17576f004629SVolodymyr Fialko 	const int start_count = 0;
17586f004629SVolodymyr Fialko 
17596f004629SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
17606f004629SVolodymyr Fialko 		return TEST_SKIPPED;
17616f004629SVolodymyr Fialko 
17626f004629SVolodymyr Fialko 	/* Set up a timer */
17636f004629SVolodymyr Fialko 	rte_timer_init(&timer);
17646f004629SVolodymyr Fialko 
17656f004629SVolodymyr Fialko 	/* Create configuration for actual testing */
17666f004629SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
17676f004629SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(start_count, sn_size);
17686f004629SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(start_count, sn_size);
17696f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.args = &timer_args;
17706f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.timer = &timer;
17716f004629SVolodymyr Fialko 	dl_conf.entity.t_reordering.start = test_rte_timer_start_cb;
17726f004629SVolodymyr Fialko 
17736f004629SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
17746f004629SVolodymyr Fialko 	if (pdcp_entity == NULL)
17756f004629SVolodymyr Fialko 		return ret;
17766f004629SVolodymyr Fialko 
17776f004629SVolodymyr Fialko 	timer_args.status = 0;
17786f004629SVolodymyr Fialko 	timer_args.pdcp_entity = pdcp_entity;
17796f004629SVolodymyr Fialko 
17806f004629SVolodymyr Fialko 	/* Send packet with SN > RX_DELIV to create a gap */
17816f004629SVolodymyr Fialko 	m1 = generate_packet_for_dl_with_sn(*ul_conf, start_count + 1);
17826f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(m1 != NULL, exit, "Could not allocate buffer for packet\n");
17836f004629SVolodymyr Fialko 
17846f004629SVolodymyr Fialko 	/* Buffered packets after insert [NULL, m1] */
17856f004629SVolodymyr Fialko 	n = test_process_packets(pdcp_entity, dl_conf.entity.dev_id, &m1, 1, out_mb, &nb_err);
17866f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet buffering\n");
17876f004629SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(n == 0, exit, "Packet was not buffered as expected\n");
17886f004629SVolodymyr Fialko 
17896f004629SVolodymyr Fialko 	m1 = NULL; /* Packet was moved to PDCP lib */
17906f004629SVolodymyr Fialko 
17916f004629SVolodymyr Fialko 	/* Verify that expire was handled correctly */
17926f004629SVolodymyr Fialko 	rte_timer_manage();
17936f004629SVolodymyr Fialko 	while (timer_args.status != 1) {
17946f004629SVolodymyr Fialko 		rte_delay_us(1);
17956f004629SVolodymyr Fialko 		rte_timer_manage();
1796bf6b85baSVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_try > 0, exit, "Bad expire handle status %i\n",
17976f004629SVolodymyr Fialko 			timer_args.status);
1798bf6b85baSVolodymyr Fialko 		nb_try--;
17996f004629SVolodymyr Fialko 	}
18006f004629SVolodymyr Fialko 
18016f004629SVolodymyr Fialko 	ret = TEST_SUCCESS;
18026f004629SVolodymyr Fialko exit:
18036f004629SVolodymyr Fialko 	rte_pktmbuf_free(m1);
18046f004629SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, n);
18056f004629SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
18066f004629SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
18076f004629SVolodymyr Fialko 	return ret;
18086f004629SVolodymyr Fialko }
18096f004629SVolodymyr Fialko 
18100eeeb7d0SVolodymyr Fialko static struct rte_pdcp_up_ctrl_pdu_hdr *
pdcp_status_report_init(uint32_t fmc)18110eeeb7d0SVolodymyr Fialko pdcp_status_report_init(uint32_t fmc)
18120eeeb7d0SVolodymyr Fialko {
18130eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = testsuite_params.status_report;
18140eeeb7d0SVolodymyr Fialko 
18150eeeb7d0SVolodymyr Fialko 	hdr->d_c = RTE_PDCP_PDU_TYPE_CTRL;
18160eeeb7d0SVolodymyr Fialko 	hdr->pdu_type = RTE_PDCP_CTRL_PDU_TYPE_STATUS_REPORT;
18170eeeb7d0SVolodymyr Fialko 	hdr->fmc = rte_cpu_to_be_32(fmc);
18180eeeb7d0SVolodymyr Fialko 	hdr->r = 0;
18190eeeb7d0SVolodymyr Fialko 	memset(hdr->bitmap, 0, testsuite_params.status_report_bitmask_capacity);
18200eeeb7d0SVolodymyr Fialko 
18210eeeb7d0SVolodymyr Fialko 	return hdr;
18220eeeb7d0SVolodymyr Fialko }
18230eeeb7d0SVolodymyr Fialko 
18240eeeb7d0SVolodymyr Fialko static uint32_t
pdcp_status_report_len(void)18250eeeb7d0SVolodymyr Fialko pdcp_status_report_len(void)
18260eeeb7d0SVolodymyr Fialko {
18270eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = testsuite_params.status_report;
18280eeeb7d0SVolodymyr Fialko 	uint32_t i;
18290eeeb7d0SVolodymyr Fialko 
18300eeeb7d0SVolodymyr Fialko 	for (i = testsuite_params.status_report_bitmask_capacity; i != 0; i--) {
18310eeeb7d0SVolodymyr Fialko 		if (hdr->bitmap[i - 1])
18320eeeb7d0SVolodymyr Fialko 			return i;
18330eeeb7d0SVolodymyr Fialko 	}
18340eeeb7d0SVolodymyr Fialko 
18350eeeb7d0SVolodymyr Fialko 	return 0;
18360eeeb7d0SVolodymyr Fialko }
18370eeeb7d0SVolodymyr Fialko 
18380eeeb7d0SVolodymyr Fialko static int
pdcp_status_report_verify(struct rte_mbuf * status_report,const struct rte_pdcp_up_ctrl_pdu_hdr * expected_hdr,uint32_t expected_len)18390eeeb7d0SVolodymyr Fialko pdcp_status_report_verify(struct rte_mbuf *status_report,
18400eeeb7d0SVolodymyr Fialko 			 const struct rte_pdcp_up_ctrl_pdu_hdr *expected_hdr, uint32_t expected_len)
18410eeeb7d0SVolodymyr Fialko {
18420eeeb7d0SVolodymyr Fialko 	uint32_t received_len = rte_pktmbuf_pkt_len(status_report);
18430eeeb7d0SVolodymyr Fialko 	uint8_t *received_buf = testsuite_params.ctrl_pdu_buf;
18440eeeb7d0SVolodymyr Fialko 	int ret;
18450eeeb7d0SVolodymyr Fialko 
18460eeeb7d0SVolodymyr Fialko 	ret = pktmbuf_read_into(status_report, received_buf, RTE_PDCP_CTRL_PDU_SIZE_MAX);
18470eeeb7d0SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to copy status report pkt into continuous buffer");
18480eeeb7d0SVolodymyr Fialko 
18490eeeb7d0SVolodymyr Fialko 	debug_hexdump(stdout, "Received:", received_buf, received_len);
18500eeeb7d0SVolodymyr Fialko 	debug_hexdump(stdout, "Expected:", expected_hdr, expected_len);
18510eeeb7d0SVolodymyr Fialko 
18520eeeb7d0SVolodymyr Fialko 	TEST_ASSERT_EQUAL(expected_len, received_len,
18530eeeb7d0SVolodymyr Fialko 			  "Mismatch in packet lengths [expected: %d, received: %d]",
18540eeeb7d0SVolodymyr Fialko 			  expected_len, received_len);
18550eeeb7d0SVolodymyr Fialko 
18560eeeb7d0SVolodymyr Fialko 	TEST_ASSERT_BUFFERS_ARE_EQUAL(received_buf, expected_hdr, expected_len,
18570eeeb7d0SVolodymyr Fialko 				     "Generated packet not as expected");
18580eeeb7d0SVolodymyr Fialko 
18590eeeb7d0SVolodymyr Fialko 	return 0;
18600eeeb7d0SVolodymyr Fialko }
18610eeeb7d0SVolodymyr Fialko 
18620eeeb7d0SVolodymyr Fialko static int
test_status_report_gen(const struct pdcp_test_conf * ul_conf,const struct rte_pdcp_up_ctrl_pdu_hdr * hdr,uint32_t bitmap_len)18630eeeb7d0SVolodymyr Fialko test_status_report_gen(const struct pdcp_test_conf *ul_conf,
18640eeeb7d0SVolodymyr Fialko 		       const struct rte_pdcp_up_ctrl_pdu_hdr *hdr,
18650eeeb7d0SVolodymyr Fialko 		       uint32_t bitmap_len)
18660eeeb7d0SVolodymyr Fialko {
18670eeeb7d0SVolodymyr Fialko 	const enum rte_security_pdcp_sn_size sn_size = ul_conf->entity.pdcp_xfrm.sn_size;
18680eeeb7d0SVolodymyr Fialko 	struct rte_mbuf *status_report = NULL, **out_mb, *m;
18690eeeb7d0SVolodymyr Fialko 	uint16_t nb_success = 0, nb_err = 0;
18700eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_entity *pdcp_entity;
18710eeeb7d0SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
18720eeeb7d0SVolodymyr Fialko 	int ret = TEST_FAILED, nb_out;
18730eeeb7d0SVolodymyr Fialko 	uint32_t nb_pkts = 0, i;
18740eeeb7d0SVolodymyr Fialko 	uint8_t cdev_id;
18750eeeb7d0SVolodymyr Fialko 
18760eeeb7d0SVolodymyr Fialko 	const uint32_t start_count = rte_be_to_cpu_32(hdr->fmc);
18770eeeb7d0SVolodymyr Fialko 
18780eeeb7d0SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
18790eeeb7d0SVolodymyr Fialko 		return TEST_SKIPPED;
18800eeeb7d0SVolodymyr Fialko 
18810eeeb7d0SVolodymyr Fialko 	/* Create configuration for actual testing */
18820eeeb7d0SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
18830eeeb7d0SVolodymyr Fialko 	dl_conf.entity.pdcp_xfrm.hfn = pdcp_hfn_from_count_get(start_count, sn_size);
18840eeeb7d0SVolodymyr Fialko 	dl_conf.entity.sn = pdcp_sn_from_count_get(start_count, sn_size);
18850eeeb7d0SVolodymyr Fialko 	dl_conf.entity.status_report_required = true;
18860eeeb7d0SVolodymyr Fialko 
18870eeeb7d0SVolodymyr Fialko 	pdcp_entity = test_entity_create(&dl_conf, &ret);
18880eeeb7d0SVolodymyr Fialko 	if (pdcp_entity == NULL)
18890eeeb7d0SVolodymyr Fialko 		return ret;
18900eeeb7d0SVolodymyr Fialko 
18910eeeb7d0SVolodymyr Fialko 	cdev_id = dl_conf.entity.dev_id;
18920eeeb7d0SVolodymyr Fialko 	out_mb = calloc(pdcp_entity->max_pkt_cache, sizeof(uintptr_t));
18930eeeb7d0SVolodymyr Fialko 
18940eeeb7d0SVolodymyr Fialko 	for (i = 0; i < bitmap_len * 8; i++) {
18950eeeb7d0SVolodymyr Fialko 		if (!bitmask_is_bit_set(hdr->bitmap, i))
18960eeeb7d0SVolodymyr Fialko 			continue;
18970eeeb7d0SVolodymyr Fialko 
18980eeeb7d0SVolodymyr Fialko 		m = generate_packet_for_dl_with_sn(*ul_conf, start_count + i + 1);
18990eeeb7d0SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(m != NULL, exit, "Could not allocate buffer for packet\n");
19000eeeb7d0SVolodymyr Fialko 
19010eeeb7d0SVolodymyr Fialko 		nb_success = test_process_packets(pdcp_entity, cdev_id, &m, 1, out_mb, &nb_err);
19020eeeb7d0SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_err == 0, exit, "Error occurred during packet buffering\n");
19030eeeb7d0SVolodymyr Fialko 		ASSERT_TRUE_OR_GOTO(nb_success == 0, exit, "Packet was not buffered as expected\n");
19040eeeb7d0SVolodymyr Fialko 
19050eeeb7d0SVolodymyr Fialko 	}
19060eeeb7d0SVolodymyr Fialko 
19070eeeb7d0SVolodymyr Fialko 	m = NULL;
19080eeeb7d0SVolodymyr Fialko 
19090eeeb7d0SVolodymyr Fialko 	/* Check status report */
19100eeeb7d0SVolodymyr Fialko 	status_report = rte_pdcp_control_pdu_create(pdcp_entity,
19110eeeb7d0SVolodymyr Fialko 			RTE_PDCP_CTRL_PDU_TYPE_STATUS_REPORT);
19120eeeb7d0SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(status_report != NULL, exit, "Could not generate status report\n");
19130eeeb7d0SVolodymyr Fialko 
19140eeeb7d0SVolodymyr Fialko 	const uint32_t expected_len = sizeof(struct rte_pdcp_up_ctrl_pdu_hdr) + bitmap_len;
19150eeeb7d0SVolodymyr Fialko 
19160eeeb7d0SVolodymyr Fialko 	ASSERT_TRUE_OR_GOTO(pdcp_status_report_verify(status_report, hdr, expected_len) == 0, exit,
19170eeeb7d0SVolodymyr Fialko 			   "Report verification failure\n");
19180eeeb7d0SVolodymyr Fialko 
19190eeeb7d0SVolodymyr Fialko 	ret = TEST_SUCCESS;
19200eeeb7d0SVolodymyr Fialko exit:
19210eeeb7d0SVolodymyr Fialko 	rte_free(m);
19220eeeb7d0SVolodymyr Fialko 	rte_pktmbuf_free(status_report);
19230eeeb7d0SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_pkts);
19240eeeb7d0SVolodymyr Fialko 	nb_out = rte_pdcp_entity_release(pdcp_entity, out_mb);
19250eeeb7d0SVolodymyr Fialko 	rte_pktmbuf_free_bulk(out_mb, nb_out);
19260eeeb7d0SVolodymyr Fialko 	free(out_mb);
19270eeeb7d0SVolodymyr Fialko 	return ret;
19280eeeb7d0SVolodymyr Fialko }
19290eeeb7d0SVolodymyr Fialko 
19300eeeb7d0SVolodymyr Fialko static void
ctrl_pdu_hdr_packet_set(struct rte_pdcp_up_ctrl_pdu_hdr * hdr,uint32_t pkt_count)19310eeeb7d0SVolodymyr Fialko ctrl_pdu_hdr_packet_set(struct rte_pdcp_up_ctrl_pdu_hdr *hdr, uint32_t pkt_count)
19320eeeb7d0SVolodymyr Fialko {
19330eeeb7d0SVolodymyr Fialko 	bitmask_set_bit(hdr->bitmap, pkt_count - rte_be_to_cpu_32(hdr->fmc) - 1);
19340eeeb7d0SVolodymyr Fialko }
19350eeeb7d0SVolodymyr Fialko 
19360eeeb7d0SVolodymyr Fialko static int
test_status_report_fmc_only(const struct pdcp_test_conf * ul_conf)19370eeeb7d0SVolodymyr Fialko test_status_report_fmc_only(const struct pdcp_test_conf *ul_conf)
19380eeeb7d0SVolodymyr Fialko {
19390eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(42);
19400eeeb7d0SVolodymyr Fialko 
19410eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
19420eeeb7d0SVolodymyr Fialko }
19430eeeb7d0SVolodymyr Fialko 
19440eeeb7d0SVolodymyr Fialko static int
test_status_report_one_pkt_first_slab(const struct pdcp_test_conf * ul_conf)19450eeeb7d0SVolodymyr Fialko test_status_report_one_pkt_first_slab(const struct pdcp_test_conf *ul_conf)
19460eeeb7d0SVolodymyr Fialko {
19470eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(0);
19480eeeb7d0SVolodymyr Fialko 
19490eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, RTE_BITMAP_SLAB_BIT_SIZE / 2 + 1);
19500eeeb7d0SVolodymyr Fialko 
19510eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
19520eeeb7d0SVolodymyr Fialko }
19530eeeb7d0SVolodymyr Fialko 
19540eeeb7d0SVolodymyr Fialko static int
test_status_report_one_pkt_second_slab(const struct pdcp_test_conf * ul_conf)19550eeeb7d0SVolodymyr Fialko test_status_report_one_pkt_second_slab(const struct pdcp_test_conf *ul_conf)
19560eeeb7d0SVolodymyr Fialko {
19570eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(1);
19580eeeb7d0SVolodymyr Fialko 
19590eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, RTE_BITMAP_SLAB_BIT_SIZE + 1);
19600eeeb7d0SVolodymyr Fialko 
19610eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
19620eeeb7d0SVolodymyr Fialko }
19630eeeb7d0SVolodymyr Fialko 
19640eeeb7d0SVolodymyr Fialko static int
test_status_report_full_slab(const struct pdcp_test_conf * ul_conf)19650eeeb7d0SVolodymyr Fialko test_status_report_full_slab(const struct pdcp_test_conf *ul_conf)
19660eeeb7d0SVolodymyr Fialko {
19670eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(1);
19680eeeb7d0SVolodymyr Fialko 	const uint32_t start_offset = RTE_BITMAP_SLAB_BIT_SIZE + 1;
19690eeeb7d0SVolodymyr Fialko 	int i;
19700eeeb7d0SVolodymyr Fialko 
19710eeeb7d0SVolodymyr Fialko 	for (i = 0; i < RTE_BITMAP_SLAB_BIT_SIZE; i++)
19720eeeb7d0SVolodymyr Fialko 		ctrl_pdu_hdr_packet_set(hdr, start_offset + i);
19730eeeb7d0SVolodymyr Fialko 
19740eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
19750eeeb7d0SVolodymyr Fialko }
19760eeeb7d0SVolodymyr Fialko 
19770eeeb7d0SVolodymyr Fialko static int
test_status_report_two_sequential_slabs(const struct pdcp_test_conf * ul_conf)19780eeeb7d0SVolodymyr Fialko test_status_report_two_sequential_slabs(const struct pdcp_test_conf *ul_conf)
19790eeeb7d0SVolodymyr Fialko {
19800eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(0);
19810eeeb7d0SVolodymyr Fialko 	const uint32_t start_offset = RTE_BITMAP_SLAB_BIT_SIZE / 2 + 1;
19820eeeb7d0SVolodymyr Fialko 
19830eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, start_offset);
19840eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, start_offset + RTE_BITMAP_SLAB_BIT_SIZE);
19850eeeb7d0SVolodymyr Fialko 
19860eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
19870eeeb7d0SVolodymyr Fialko }
19880eeeb7d0SVolodymyr Fialko 
19890eeeb7d0SVolodymyr Fialko static int
test_status_report_two_non_sequential_slabs(const struct pdcp_test_conf * ul_conf)19900eeeb7d0SVolodymyr Fialko test_status_report_two_non_sequential_slabs(const struct pdcp_test_conf *ul_conf)
19910eeeb7d0SVolodymyr Fialko {
19920eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(0);
19930eeeb7d0SVolodymyr Fialko 	const uint32_t start_offset = RTE_BITMAP_SLAB_BIT_SIZE / 2 + 1;
19940eeeb7d0SVolodymyr Fialko 
19950eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, start_offset);
19960eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, start_offset + RTE_BITMAP_SLAB_BIT_SIZE);
19970eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, 3 * RTE_BITMAP_SLAB_BIT_SIZE);
19980eeeb7d0SVolodymyr Fialko 
19990eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
20000eeeb7d0SVolodymyr Fialko }
20010eeeb7d0SVolodymyr Fialko 
20020eeeb7d0SVolodymyr Fialko static int
test_status_report_max_length_sn_12(const struct pdcp_test_conf * ul_conf)20030eeeb7d0SVolodymyr Fialko test_status_report_max_length_sn_12(const struct pdcp_test_conf *ul_conf)
20040eeeb7d0SVolodymyr Fialko {
20050eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr;
20060eeeb7d0SVolodymyr Fialko 	const uint32_t fmc = 0;
20070eeeb7d0SVolodymyr Fialko 	uint32_t i;
20080eeeb7d0SVolodymyr Fialko 
20090eeeb7d0SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK ||
20100eeeb7d0SVolodymyr Fialko 		ul_conf->entity.pdcp_xfrm.sn_size != RTE_SECURITY_PDCP_SN_SIZE_12)
20110eeeb7d0SVolodymyr Fialko 		return TEST_SKIPPED;
20120eeeb7d0SVolodymyr Fialko 
20130eeeb7d0SVolodymyr Fialko 	hdr = pdcp_status_report_init(fmc);
20140eeeb7d0SVolodymyr Fialko 
20150eeeb7d0SVolodymyr Fialko 	const uint32_t max_count = RTE_MIN((RTE_PDCP_CTRL_PDU_SIZE_MAX - sizeof(hdr)) * 8,
20160eeeb7d0SVolodymyr Fialko 			(uint32_t)PDCP_WINDOW_SIZE(RTE_SECURITY_PDCP_SN_SIZE_12));
20170eeeb7d0SVolodymyr Fialko 
20180eeeb7d0SVolodymyr Fialko 	i = fmc + 2; /* set first count to have a gap, to enable packet buffering */
20190eeeb7d0SVolodymyr Fialko 
20200eeeb7d0SVolodymyr Fialko 	for (; i < max_count; i++)
20210eeeb7d0SVolodymyr Fialko 		ctrl_pdu_hdr_packet_set(hdr, i);
20220eeeb7d0SVolodymyr Fialko 
20230eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
20240eeeb7d0SVolodymyr Fialko }
20250eeeb7d0SVolodymyr Fialko 
20260eeeb7d0SVolodymyr Fialko static int
test_status_report_overlap_different_slabs(const struct pdcp_test_conf * ul_conf)20270eeeb7d0SVolodymyr Fialko test_status_report_overlap_different_slabs(const struct pdcp_test_conf *ul_conf)
20280eeeb7d0SVolodymyr Fialko {
20290eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(63);
20300eeeb7d0SVolodymyr Fialko 	const uint32_t sn_size = 12;
20310eeeb7d0SVolodymyr Fialko 
20320eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, 64 + 1);
20330eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, PDCP_WINDOW_SIZE(sn_size) + 1);
20340eeeb7d0SVolodymyr Fialko 
20350eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
20360eeeb7d0SVolodymyr Fialko }
20370eeeb7d0SVolodymyr Fialko 
20380eeeb7d0SVolodymyr Fialko static int
test_status_report_overlap_same_slab(const struct pdcp_test_conf * ul_conf)20390eeeb7d0SVolodymyr Fialko test_status_report_overlap_same_slab(const struct pdcp_test_conf *ul_conf)
20400eeeb7d0SVolodymyr Fialko {
20410eeeb7d0SVolodymyr Fialko 	struct rte_pdcp_up_ctrl_pdu_hdr *hdr = pdcp_status_report_init(2);
20420eeeb7d0SVolodymyr Fialko 	const uint32_t sn_size = 12;
20430eeeb7d0SVolodymyr Fialko 
20440eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, 4);
20450eeeb7d0SVolodymyr Fialko 	ctrl_pdu_hdr_packet_set(hdr, PDCP_WINDOW_SIZE(sn_size) + 1);
20460eeeb7d0SVolodymyr Fialko 
20470eeeb7d0SVolodymyr Fialko 	return test_status_report_gen(ul_conf, hdr, pdcp_status_report_len());
20480eeeb7d0SVolodymyr Fialko }
20490eeeb7d0SVolodymyr Fialko 
2050e98a7ac0SVolodymyr Fialko static int
test_combined(struct pdcp_test_conf * ul_conf)2051e98a7ac0SVolodymyr Fialko test_combined(struct pdcp_test_conf *ul_conf)
2052e98a7ac0SVolodymyr Fialko {
2053e98a7ac0SVolodymyr Fialko 	struct pdcp_test_conf dl_conf;
2054e98a7ac0SVolodymyr Fialko 	int ret;
2055e98a7ac0SVolodymyr Fialko 
2056e98a7ac0SVolodymyr Fialko 	if (ul_conf->entity.pdcp_xfrm.pkt_dir == RTE_SECURITY_PDCP_DOWNLINK)
2057e98a7ac0SVolodymyr Fialko 		return TEST_SKIPPED;
2058e98a7ac0SVolodymyr Fialko 
2059e98a7ac0SVolodymyr Fialko 	ul_conf->entity.reverse_iv_direction = true;
2060e98a7ac0SVolodymyr Fialko 	ul_conf->output_len = 0;
2061e98a7ac0SVolodymyr Fialko 
2062e98a7ac0SVolodymyr Fialko 	ret = test_attempt_single(ul_conf);
2063e98a7ac0SVolodymyr Fialko 	if (ret != TEST_SUCCESS)
2064e98a7ac0SVolodymyr Fialko 		return ret;
2065e98a7ac0SVolodymyr Fialko 
2066e98a7ac0SVolodymyr Fialko 	uplink_to_downlink_convert(ul_conf, &dl_conf);
2067e98a7ac0SVolodymyr Fialko 	ret = test_attempt_single(&dl_conf);
2068e98a7ac0SVolodymyr Fialko 
2069e98a7ac0SVolodymyr Fialko 	return ret;
2070e98a7ac0SVolodymyr Fialko }
2071e98a7ac0SVolodymyr Fialko 
2072*d010725bSAakash Sasidharan #define MIN_DATA_LEN 0
2073*d010725bSAakash Sasidharan #define MAX_DATA_LEN 9000
2074*d010725bSAakash Sasidharan 
2075*d010725bSAakash Sasidharan static int
test_combined_data_walkthrough(struct pdcp_test_conf * test_conf)2076*d010725bSAakash Sasidharan test_combined_data_walkthrough(struct pdcp_test_conf *test_conf)
2077*d010725bSAakash Sasidharan {
2078*d010725bSAakash Sasidharan 	uint32_t data_len;
2079*d010725bSAakash Sasidharan 	int ret;
2080*d010725bSAakash Sasidharan 
2081*d010725bSAakash Sasidharan 	ret = test_combined(test_conf);
2082*d010725bSAakash Sasidharan 	if (ret != TEST_SUCCESS)
2083*d010725bSAakash Sasidharan 		return ret;
2084*d010725bSAakash Sasidharan 
2085*d010725bSAakash Sasidharan 	if (!silent)
2086*d010725bSAakash Sasidharan 		silent = true;
2087*d010725bSAakash Sasidharan 
2088*d010725bSAakash Sasidharan 	/* With the passing config, perform a data walkthrough test. */
2089*d010725bSAakash Sasidharan 	for (data_len = MIN_DATA_LEN; data_len <= MAX_DATA_LEN; data_len++) {
2090*d010725bSAakash Sasidharan 		test_conf_input_data_modify(test_conf, data_len);
2091*d010725bSAakash Sasidharan 		ret = test_combined(test_conf);
2092*d010725bSAakash Sasidharan 
2093*d010725bSAakash Sasidharan 		if (ret == TEST_FAILED) {
2094*d010725bSAakash Sasidharan 			printf("Data walkthrough failed for input len: %d\n", data_len);
2095*d010725bSAakash Sasidharan 			return TEST_FAILED;
2096*d010725bSAakash Sasidharan 		}
2097*d010725bSAakash Sasidharan 	}
2098*d010725bSAakash Sasidharan 
2099*d010725bSAakash Sasidharan 	silent = false;
2100*d010725bSAakash Sasidharan 
2101*d010725bSAakash Sasidharan 	return TEST_SUCCESS;
2102*d010725bSAakash Sasidharan }
2103*d010725bSAakash Sasidharan 
21046f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
21056f004629SVolodymyr Fialko static inline void
eventdev_conf_default_set(struct rte_event_dev_config * dev_conf,struct rte_event_dev_info * info)21066f004629SVolodymyr Fialko eventdev_conf_default_set(struct rte_event_dev_config *dev_conf, struct rte_event_dev_info *info)
21076f004629SVolodymyr Fialko {
21086f004629SVolodymyr Fialko 	memset(dev_conf, 0, sizeof(struct rte_event_dev_config));
21096f004629SVolodymyr Fialko 	dev_conf->dequeue_timeout_ns = info->min_dequeue_timeout_ns;
21106f004629SVolodymyr Fialko 	dev_conf->nb_event_ports = 1;
21116f004629SVolodymyr Fialko 	dev_conf->nb_event_queues = 1;
21126f004629SVolodymyr Fialko 	dev_conf->nb_event_queue_flows = info->max_event_queue_flows;
21136f004629SVolodymyr Fialko 	dev_conf->nb_event_port_dequeue_depth = info->max_event_port_dequeue_depth;
21146f004629SVolodymyr Fialko 	dev_conf->nb_event_port_enqueue_depth = info->max_event_port_enqueue_depth;
21156f004629SVolodymyr Fialko 	dev_conf->nb_event_port_enqueue_depth = info->max_event_port_enqueue_depth;
21166f004629SVolodymyr Fialko 	dev_conf->nb_events_limit = info->max_num_events;
21176f004629SVolodymyr Fialko }
21186f004629SVolodymyr Fialko 
21196f004629SVolodymyr Fialko static inline int
eventdev_setup(void)21206f004629SVolodymyr Fialko eventdev_setup(void)
21216f004629SVolodymyr Fialko {
21226f004629SVolodymyr Fialko 	struct rte_event_dev_config dev_conf;
21236f004629SVolodymyr Fialko 	struct rte_event_dev_info info;
21246f004629SVolodymyr Fialko 	int ret, evdev = 0;
21256f004629SVolodymyr Fialko 
21266f004629SVolodymyr Fialko 	if (!rte_event_dev_count())
21276f004629SVolodymyr Fialko 		return TEST_SKIPPED;
21286f004629SVolodymyr Fialko 
21296f004629SVolodymyr Fialko 	ret = rte_event_dev_info_get(evdev, &info);
21306f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to get event dev info");
21316f004629SVolodymyr Fialko 	TEST_ASSERT(info.max_num_events < 0 || info.max_num_events >= 1,
21326f004629SVolodymyr Fialko 			"ERROR max_num_events=%d < max_events=%d", info.max_num_events, 1);
21336f004629SVolodymyr Fialko 
21346f004629SVolodymyr Fialko 	eventdev_conf_default_set(&dev_conf, &info);
21356f004629SVolodymyr Fialko 	ret = rte_event_dev_configure(evdev, &dev_conf);
21366f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to configure eventdev");
21376f004629SVolodymyr Fialko 
21386f004629SVolodymyr Fialko 	ret = rte_event_queue_setup(evdev, TEST_EV_QUEUE_ID, NULL);
21396f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to setup queue=%d", TEST_EV_QUEUE_ID);
21406f004629SVolodymyr Fialko 
21416f004629SVolodymyr Fialko 	/* Configure event port */
21426f004629SVolodymyr Fialko 	ret = rte_event_port_setup(evdev, TEST_EV_PORT_ID, NULL);
21436f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to setup port=%d", TEST_EV_PORT_ID);
21446f004629SVolodymyr Fialko 	ret = rte_event_port_link(evdev, TEST_EV_PORT_ID, NULL, NULL, 0);
21456f004629SVolodymyr Fialko 	TEST_ASSERT(ret >= 0, "Failed to link all queues port=%d", TEST_EV_PORT_ID);
21466f004629SVolodymyr Fialko 
21476f004629SVolodymyr Fialko 	ret = rte_event_dev_start(evdev);
21486f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(ret, "Failed to start device");
21496f004629SVolodymyr Fialko 
21506f004629SVolodymyr Fialko 	testsuite_params.evdev = evdev;
21516f004629SVolodymyr Fialko 
21526f004629SVolodymyr Fialko 	return TEST_SUCCESS;
21536f004629SVolodymyr Fialko }
21546f004629SVolodymyr Fialko 
21556f004629SVolodymyr Fialko static int
event_timer_setup(void)21566f004629SVolodymyr Fialko event_timer_setup(void)
21576f004629SVolodymyr Fialko {
21586f004629SVolodymyr Fialko 	struct rte_event_timer_adapter_info info;
21596f004629SVolodymyr Fialko 	struct rte_event_timer_adapter *timdev;
21606f004629SVolodymyr Fialko 	uint32_t caps = 0;
21616f004629SVolodymyr Fialko 
21626f004629SVolodymyr Fialko 	struct rte_event_timer_adapter_conf config = {
21636f004629SVolodymyr Fialko 		.event_dev_id = testsuite_params.evdev,
21646f004629SVolodymyr Fialko 		.timer_adapter_id = TIMER_ADAPTER_ID,
21656f004629SVolodymyr Fialko 		.timer_tick_ns = NSECPERSEC,
21666f004629SVolodymyr Fialko 		.max_tmo_ns = 10 * NSECPERSEC,
21676f004629SVolodymyr Fialko 		.nb_timers = 10,
21686f004629SVolodymyr Fialko 		.flags = 0,
21696f004629SVolodymyr Fialko 	};
21706f004629SVolodymyr Fialko 
21716f004629SVolodymyr Fialko 	TEST_ASSERT_SUCCESS(rte_event_timer_adapter_caps_get(testsuite_params.evdev, &caps),
21726f004629SVolodymyr Fialko 				"Failed to get adapter capabilities");
21736f004629SVolodymyr Fialko 
21746f004629SVolodymyr Fialko 	if (!(caps & RTE_EVENT_TIMER_ADAPTER_CAP_INTERNAL_PORT))
21756f004629SVolodymyr Fialko 		return TEST_SKIPPED;
21766f004629SVolodymyr Fialko 
21776f004629SVolodymyr Fialko 	timdev = rte_event_timer_adapter_create(&config);
21786f004629SVolodymyr Fialko 
21796f004629SVolodymyr Fialko 	TEST_ASSERT_NOT_NULL(timdev, "Failed to create event timer ring");
21806f004629SVolodymyr Fialko 
21816f004629SVolodymyr Fialko 	testsuite_params.timdev = timdev;
21826f004629SVolodymyr Fialko 
21836f004629SVolodymyr Fialko 	TEST_ASSERT_EQUAL(rte_event_timer_adapter_start(timdev), 0,
21846f004629SVolodymyr Fialko 			"Failed to start event timer adapter");
21856f004629SVolodymyr Fialko 
21866f004629SVolodymyr Fialko 	rte_event_timer_adapter_get_info(timdev, &info);
21876f004629SVolodymyr Fialko 	testsuite_params.min_resolution_ns = info.min_resolution_ns;
21886f004629SVolodymyr Fialko 
21896f004629SVolodymyr Fialko 	return TEST_SUCCESS;
21906f004629SVolodymyr Fialko }
21916f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
21926f004629SVolodymyr Fialko 
21936f004629SVolodymyr Fialko static int
ut_setup_pdcp_event_timer(void)21946f004629SVolodymyr Fialko ut_setup_pdcp_event_timer(void)
21956f004629SVolodymyr Fialko {
21966f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
21976f004629SVolodymyr Fialko 	int ret;
21986f004629SVolodymyr Fialko 
21996f004629SVolodymyr Fialko 	ret = eventdev_setup();
22006f004629SVolodymyr Fialko 	if (ret)
22016f004629SVolodymyr Fialko 		return ret;
22026f004629SVolodymyr Fialko 
22036f004629SVolodymyr Fialko 	return event_timer_setup();
22046f004629SVolodymyr Fialko #else
22056f004629SVolodymyr Fialko 	return TEST_SKIPPED;
22066f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
22076f004629SVolodymyr Fialko }
22086f004629SVolodymyr Fialko 
22096f004629SVolodymyr Fialko static void
ut_teardown_pdcp_event_timer(void)22106f004629SVolodymyr Fialko ut_teardown_pdcp_event_timer(void)
22116f004629SVolodymyr Fialko {
22126f004629SVolodymyr Fialko #ifdef RTE_LIB_EVENTDEV
22136f004629SVolodymyr Fialko 	struct rte_event_timer_adapter *timdev = testsuite_params.timdev;
22146f004629SVolodymyr Fialko 	int evdev = testsuite_params.evdev;
22156f004629SVolodymyr Fialko 
22166f004629SVolodymyr Fialko 	rte_event_dev_stop(evdev);
22176f004629SVolodymyr Fialko 	rte_event_dev_close(evdev);
22186f004629SVolodymyr Fialko 
22196f004629SVolodymyr Fialko 	rte_event_timer_adapter_stop(timdev);
22206f004629SVolodymyr Fialko 	rte_event_timer_adapter_free(timdev);
22216f004629SVolodymyr Fialko #endif /* RTE_LIB_EVENTDEV */
22226f004629SVolodymyr Fialko }
22236f004629SVolodymyr Fialko 
2224bd24488cSAnoob Joseph static int
run_test_for_one_known_vec(const void * arg)2225bd24488cSAnoob Joseph run_test_for_one_known_vec(const void *arg)
2226bd24488cSAnoob Joseph {
2227bd24488cSAnoob Joseph 	struct pdcp_test_conf test_conf;
2228bd24488cSAnoob Joseph 	int i = *(const uint32_t *)arg;
2229bd24488cSAnoob Joseph 
223064b12db3SAakash Sasidharan 	create_test_conf_from_index(i, &test_conf, PDCP_TEST_SUITE_TY_BASIC);
2231bd24488cSAnoob Joseph 	return test_attempt_single(&test_conf);
2232bd24488cSAnoob Joseph }
2233bd24488cSAnoob Joseph 
2234e98a7ac0SVolodymyr Fialko static struct unit_test_suite combined_mode_cases  = {
2235e98a7ac0SVolodymyr Fialko 	.suite_name = "PDCP combined mode",
2236e98a7ac0SVolodymyr Fialko 	.unit_test_cases = {
2237e98a7ac0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("combined mode", ut_setup_pdcp, ut_teardown_pdcp,
2238e98a7ac0SVolodymyr Fialko 			run_test_with_all_known_vec, test_combined),
2239*d010725bSAakash Sasidharan 		TEST_CASE_NAMED_WITH_DATA("combined mode data walkthrough",
2240*d010725bSAakash Sasidharan 			ut_setup_pdcp, ut_teardown_pdcp,
2241*d010725bSAakash Sasidharan 			run_test_with_all_known_vec, test_combined_data_walkthrough),
2242e98a7ac0SVolodymyr Fialko 		TEST_CASES_END() /**< NULL terminate unit test array */
2243e98a7ac0SVolodymyr Fialko 	}
2244e98a7ac0SVolodymyr Fialko };
2245e98a7ac0SVolodymyr Fialko 
2246e98a7ac0SVolodymyr Fialko static struct unit_test_suite hfn_sn_test_cases  = {
2247e98a7ac0SVolodymyr Fialko 	.suite_name = "PDCP HFN/SN",
2248e98a7ac0SVolodymyr Fialko 	.unit_test_cases = {
2249e98a7ac0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("SN plus window", ut_setup_pdcp, ut_teardown_pdcp,
2250e98a7ac0SVolodymyr Fialko 			run_test_with_all_known_vec, test_sn_plus_window),
2251e98a7ac0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("SN minus window", ut_setup_pdcp, ut_teardown_pdcp,
2252e98a7ac0SVolodymyr Fialko 			run_test_with_all_known_vec, test_sn_minus_window),
2253e98a7ac0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("SN plus outside", ut_setup_pdcp, ut_teardown_pdcp,
2254e98a7ac0SVolodymyr Fialko 			run_test_with_all_known_vec, test_sn_plus_outside),
2255e98a7ac0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("SN minus outside", ut_setup_pdcp, ut_teardown_pdcp,
2256e98a7ac0SVolodymyr Fialko 			run_test_with_all_known_vec, test_sn_minus_outside),
2257e98a7ac0SVolodymyr Fialko 		TEST_CASES_END() /**< NULL terminate unit test array */
2258e98a7ac0SVolodymyr Fialko 	}
2259e98a7ac0SVolodymyr Fialko };
2260e98a7ac0SVolodymyr Fialko 
2261d84ac1b9SVolodymyr Fialko static struct unit_test_suite reorder_test_cases  = {
2262d84ac1b9SVolodymyr Fialko 	.suite_name = "PDCP reorder",
2263d84ac1b9SVolodymyr Fialko 	.unit_test_cases = {
2264d84ac1b9SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_reorder_gap_fill",
2265d84ac1b9SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
2266d84ac1b9SVolodymyr Fialko 			run_test_with_all_known_vec, test_reorder_gap_fill),
226763c6edc5SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_reorder_gap_in_reorder_buffer",
226863c6edc5SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
226963c6edc5SVolodymyr Fialko 			run_test_with_all_known_vec, test_reorder_gap_in_reorder_buffer),
2270d84ac1b9SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_reorder_buffer_full_window_size_sn_12",
2271d84ac1b9SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
2272d84ac1b9SVolodymyr Fialko 			run_test_with_all_known_vec_until_first_pass,
2273d84ac1b9SVolodymyr Fialko 			test_reorder_buffer_full_window_size_sn_12),
22746f004629SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_expire_with_event_timer",
22756f004629SVolodymyr Fialko 			ut_setup_pdcp_event_timer, ut_teardown_pdcp_event_timer,
22766f004629SVolodymyr Fialko 			run_test_with_all_known_vec_until_first_pass,
22776f004629SVolodymyr Fialko 			test_expiry_with_event_timer),
22786f004629SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_expire_with_rte_timer",
22796f004629SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
22806f004629SVolodymyr Fialko 			run_test_with_all_known_vec_until_first_pass,
22816f004629SVolodymyr Fialko 			test_expiry_with_rte_timer),
2282d84ac1b9SVolodymyr Fialko 		TEST_CASES_END() /**< NULL terminate unit test array */
2283d84ac1b9SVolodymyr Fialko 	}
2284d84ac1b9SVolodymyr Fialko };
2285d84ac1b9SVolodymyr Fialko 
22860eeeb7d0SVolodymyr Fialko static struct unit_test_suite status_report_test_cases  = {
22870eeeb7d0SVolodymyr Fialko 	.suite_name = "PDCP status report",
22880eeeb7d0SVolodymyr Fialko 	.unit_test_cases = {
22890eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_fmc_only",
22900eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
22910eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_fmc_only),
22920eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_one_pkt_first_slab",
22930eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
22940eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_one_pkt_first_slab),
22950eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_one_pkt_second_slab",
22960eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
22970eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_one_pkt_second_slab),
22980eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_full_slab",
22990eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23000eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_full_slab),
23010eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_two_sequential_slabs",
23020eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23030eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_two_sequential_slabs),
23040eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_two_non_sequential_slabs",
23050eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23060eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_two_non_sequential_slabs),
23070eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_max_length_sn_12",
23080eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23090eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec_until_first_pass,
23100eeeb7d0SVolodymyr Fialko 			test_status_report_max_length_sn_12),
23110eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_overlap_different_slabs",
23120eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23130eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_overlap_different_slabs),
23140eeeb7d0SVolodymyr Fialko 		TEST_CASE_NAMED_WITH_DATA("test_status_report_overlap_same_slab",
23150eeeb7d0SVolodymyr Fialko 			ut_setup_pdcp, ut_teardown_pdcp,
23160eeeb7d0SVolodymyr Fialko 			run_test_with_all_known_vec, test_status_report_overlap_same_slab),
23170eeeb7d0SVolodymyr Fialko 		TEST_CASES_END() /**< NULL terminate unit test array */
23180eeeb7d0SVolodymyr Fialko 	}
23190eeeb7d0SVolodymyr Fialko };
23200eeeb7d0SVolodymyr Fialko 
232164b12db3SAakash Sasidharan static struct unit_test_suite sdap_test_cases  = {
232264b12db3SAakash Sasidharan 	.suite_name = "PDCP SDAP",
232364b12db3SAakash Sasidharan 	.unit_test_cases = {
232464b12db3SAakash Sasidharan 		TEST_CASE_NAMED_WITH_DATA("SDAP Known vector cases",
232564b12db3SAakash Sasidharan 			ut_setup_pdcp, ut_teardown_pdcp,
232664b12db3SAakash Sasidharan 			run_test_with_all_sdap_known_vec, test_attempt_single),
232764b12db3SAakash Sasidharan 		TEST_CASE_NAMED_WITH_DATA("SDAP combined mode",
232864b12db3SAakash Sasidharan 			ut_setup_pdcp, ut_teardown_pdcp,
232964b12db3SAakash Sasidharan 			run_test_with_all_sdap_known_vec, test_combined),
233064b12db3SAakash Sasidharan 		TEST_CASES_END() /**< NULL terminate unit test array */
233164b12db3SAakash Sasidharan 	}
233264b12db3SAakash Sasidharan };
2333bd24488cSAnoob Joseph struct unit_test_suite *test_suites[] = {
2334bd24488cSAnoob Joseph 	NULL, /* Place holder for known_vector_cases */
233564b12db3SAakash Sasidharan 	&sdap_test_cases,
2336e98a7ac0SVolodymyr Fialko 	&combined_mode_cases,
2337e98a7ac0SVolodymyr Fialko 	&hfn_sn_test_cases,
2338d84ac1b9SVolodymyr Fialko 	&reorder_test_cases,
23390eeeb7d0SVolodymyr Fialko 	&status_report_test_cases,
2340bd24488cSAnoob Joseph 	NULL /* End of suites list */
2341bd24488cSAnoob Joseph };
2342bd24488cSAnoob Joseph 
2343bd24488cSAnoob Joseph static struct unit_test_suite pdcp_testsuite  = {
2344bd24488cSAnoob Joseph 	.suite_name = "PDCP Unit Test Suite",
2345bd24488cSAnoob Joseph 	.unit_test_cases = {TEST_CASES_END()},
2346bd24488cSAnoob Joseph 	.setup = testsuite_setup,
2347bd24488cSAnoob Joseph 	.teardown = testsuite_teardown,
2348bd24488cSAnoob Joseph 	.unit_test_suites = test_suites,
2349bd24488cSAnoob Joseph };
2350bd24488cSAnoob Joseph 
2351bd24488cSAnoob Joseph static int
test_pdcp(void)2352bd24488cSAnoob Joseph test_pdcp(void)
2353bd24488cSAnoob Joseph {
2354bd24488cSAnoob Joseph 	struct unit_test_suite *known_vector_cases;
235564b12db3SAakash Sasidharan 	uint32_t nb_tests = nb_tests_get(PDCP_TEST_SUITE_TY_BASIC);
235664b12db3SAakash Sasidharan 	int ret, index[nb_tests];
2357bd24488cSAnoob Joseph 	uint32_t i, size;
2358bd24488cSAnoob Joseph 
2359bd24488cSAnoob Joseph 	size = sizeof(struct unit_test_suite);
236064b12db3SAakash Sasidharan 	size += (nb_tests + 1) * sizeof(struct unit_test_case);
2361bd24488cSAnoob Joseph 
2362bd24488cSAnoob Joseph 	known_vector_cases = rte_zmalloc(NULL, size, 0);
2363bd24488cSAnoob Joseph 	if (known_vector_cases == NULL)
2364bd24488cSAnoob Joseph 		return TEST_FAILED;
2365bd24488cSAnoob Joseph 
2366bd24488cSAnoob Joseph 	known_vector_cases->suite_name = "Known vector cases";
2367bd24488cSAnoob Joseph 
236864b12db3SAakash Sasidharan 	for (i = 0; i < nb_tests; i++) {
2369bd24488cSAnoob Joseph 		index[i] = i;
2370bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].name = pdcp_test_params[i].name;
2371bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].data = (void *)&index[i];
2372bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].enabled = 1;
2373bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].setup = ut_setup_pdcp;
2374bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].teardown = ut_teardown_pdcp;
2375bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].testcase = NULL;
2376bd24488cSAnoob Joseph 		known_vector_cases->unit_test_cases[i].testcase_with_data
2377bd24488cSAnoob Joseph 				= run_test_for_one_known_vec;
2378bd24488cSAnoob Joseph 	}
2379bd24488cSAnoob Joseph 
2380bd24488cSAnoob Joseph 	known_vector_cases->unit_test_cases[i].testcase = NULL;
2381bd24488cSAnoob Joseph 	known_vector_cases->unit_test_cases[i].testcase_with_data = NULL;
2382bd24488cSAnoob Joseph 
2383bd24488cSAnoob Joseph 	test_suites[0] = known_vector_cases;
2384bd24488cSAnoob Joseph 
2385bd24488cSAnoob Joseph 	ret = unit_test_suite_runner(&pdcp_testsuite);
2386bd24488cSAnoob Joseph 
2387bd24488cSAnoob Joseph 	rte_free(known_vector_cases);
2388bd24488cSAnoob Joseph 	return ret;
2389bd24488cSAnoob Joseph }
2390bd24488cSAnoob Joseph 
2391d83fb967SDavid Marchand REGISTER_FAST_TEST(pdcp_autotest, false, true, test_pdcp);
2392