2017-12-19 15:49:05 +00:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
* Copyright(c) 2010-2014 Intel Corporation
|
2012-09-04 13:54:00 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _TEST_H_
|
|
|
|
#define _TEST_H_
|
2016-07-29 14:55:28 +02:00
|
|
|
|
2015-11-25 13:25:16 +00:00
|
|
|
#include <stddef.h>
|
2014-08-18 13:29:22 +02:00
|
|
|
#include <sys/queue.h>
|
2016-07-29 14:55:28 +02:00
|
|
|
|
2017-12-08 14:21:21 +01:00
|
|
|
#include <rte_hexdump.h>
|
2016-07-29 14:55:28 +02:00
|
|
|
#include <rte_common.h>
|
2014-08-18 13:29:22 +02:00
|
|
|
|
2018-01-11 17:50:34 +00:00
|
|
|
#define TEST_SUCCESS EXIT_SUCCESS
|
|
|
|
#define TEST_FAILED -1
|
|
|
|
#define TEST_SKIPPED 77
|
2015-02-13 11:38:13 +01:00
|
|
|
|
|
|
|
/* Before including test.h file you can define
|
|
|
|
* TEST_TRACE_FAILURE(_file, _line, _func) macro to better trace/debug test
|
|
|
|
* failures. Mostly useful in test development phase. */
|
|
|
|
#ifndef TEST_TRACE_FAILURE
|
|
|
|
# define TEST_TRACE_FAILURE(_file, _line, _func)
|
|
|
|
#endif
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#include <rte_test.h>
|
|
|
|
|
|
|
|
#define TEST_ASSERT RTE_TEST_ASSERT
|
|
|
|
|
|
|
|
#define TEST_ASSERT_EQUAL RTE_TEST_ASSERT_EQUAL
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2016-06-20 15:40:05 +01:00
|
|
|
/* Compare two buffers (length in bytes) */
|
2015-11-25 13:25:16 +00:00
|
|
|
#define TEST_ASSERT_BUFFERS_ARE_EQUAL(a, b, len, msg, ...) do { \
|
|
|
|
if (memcmp(a, b, len)) { \
|
|
|
|
printf("TestCase %s() line %d failed: " \
|
|
|
|
msg "\n", __func__, __LINE__, ##__VA_ARGS__); \
|
|
|
|
TEST_TRACE_FAILURE(__FILE__, __LINE__, __func__); \
|
|
|
|
return TEST_FAILED; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2016-06-20 15:40:05 +01:00
|
|
|
/* Compare two buffers with offset (length and offset in bytes) */
|
|
|
|
#define TEST_ASSERT_BUFFERS_ARE_EQUAL_OFFSET(a, b, len, off, msg, ...) do { \
|
|
|
|
const uint8_t *_a_with_off = (const uint8_t *)a + off; \
|
|
|
|
const uint8_t *_b_with_off = (const uint8_t *)b + off; \
|
|
|
|
TEST_ASSERT_BUFFERS_ARE_EQUAL(_a_with_off, _b_with_off, len, msg); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/* Compare two buffers (length in bits) */
|
|
|
|
#define TEST_ASSERT_BUFFERS_ARE_EQUAL_BIT(a, b, len, msg, ...) do { \
|
|
|
|
uint8_t _last_byte_a, _last_byte_b; \
|
|
|
|
uint8_t _last_byte_mask, _last_byte_bits; \
|
|
|
|
TEST_ASSERT_BUFFERS_ARE_EQUAL(a, b, (len >> 3), msg); \
|
|
|
|
if (len % 8) { \
|
|
|
|
_last_byte_bits = len % 8; \
|
|
|
|
_last_byte_mask = ~((1 << (8 - _last_byte_bits)) - 1); \
|
|
|
|
_last_byte_a = ((const uint8_t *)a)[len >> 3]; \
|
|
|
|
_last_byte_b = ((const uint8_t *)b)[len >> 3]; \
|
|
|
|
_last_byte_a &= _last_byte_mask; \
|
|
|
|
_last_byte_b &= _last_byte_mask; \
|
|
|
|
if (_last_byte_a != _last_byte_b) { \
|
|
|
|
printf("TestCase %s() line %d failed: " \
|
|
|
|
msg "\n", __func__, __LINE__, ##__VA_ARGS__);\
|
|
|
|
TEST_TRACE_FAILURE(__FILE__, __LINE__, __func__); \
|
|
|
|
return TEST_FAILED; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/* Compare two buffers with offset (length and offset in bits) */
|
|
|
|
#define TEST_ASSERT_BUFFERS_ARE_EQUAL_BIT_OFFSET(a, b, len, off, msg, ...) do { \
|
|
|
|
uint8_t _first_byte_a, _first_byte_b; \
|
|
|
|
uint8_t _first_byte_mask, _first_byte_bits; \
|
|
|
|
uint32_t _len_without_first_byte = (off % 8) ? \
|
|
|
|
len - (8 - (off % 8)) : \
|
|
|
|
len; \
|
|
|
|
uint32_t _off_in_bytes = (off % 8) ? (off >> 3) + 1 : (off >> 3); \
|
|
|
|
const uint8_t *_a_with_off = (const uint8_t *)a + _off_in_bytes; \
|
|
|
|
const uint8_t *_b_with_off = (const uint8_t *)b + _off_in_bytes; \
|
|
|
|
TEST_ASSERT_BUFFERS_ARE_EQUAL_BIT(_a_with_off, _b_with_off, \
|
|
|
|
_len_without_first_byte, msg); \
|
|
|
|
if (off % 8) { \
|
|
|
|
_first_byte_bits = 8 - (off % 8); \
|
|
|
|
_first_byte_mask = (1 << _first_byte_bits) - 1; \
|
|
|
|
_first_byte_a = *(_a_with_off - 1); \
|
|
|
|
_first_byte_b = *(_b_with_off - 1); \
|
|
|
|
_first_byte_a &= _first_byte_mask; \
|
|
|
|
_first_byte_b &= _first_byte_mask; \
|
|
|
|
if (_first_byte_a != _first_byte_b) { \
|
|
|
|
printf("TestCase %s() line %d failed: " \
|
|
|
|
msg "\n", __func__, __LINE__, ##__VA_ARGS__); \
|
|
|
|
TEST_TRACE_FAILURE(__FILE__, __LINE__, __func__); \
|
|
|
|
return TEST_FAILED; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
} while (0)
|
2015-11-25 13:25:16 +00:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#define TEST_ASSERT_NOT_EQUAL RTE_TEST_ASSERT_NOT_EQUAL
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#define TEST_ASSERT_SUCCESS RTE_TEST_ASSERT_SUCCESS
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#define TEST_ASSERT_FAIL RTE_TEST_ASSERT_FAIL
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#define TEST_ASSERT_NULL RTE_TEST_ASSERT_NULL
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2018-01-11 15:51:46 +05:30
|
|
|
#define TEST_ASSERT_NOT_NULL RTE_TEST_ASSERT_NOT_NULL
|
2014-07-21 15:52:16 +01:00
|
|
|
|
|
|
|
struct unit_test_case {
|
|
|
|
int (*setup)(void);
|
2015-11-25 13:25:16 +00:00
|
|
|
void (*teardown)(void);
|
2014-07-21 15:52:16 +01:00
|
|
|
int (*testcase)(void);
|
2021-05-12 11:36:55 +00:00
|
|
|
int (*testcase_with_data)(const void *data);
|
2017-03-19 19:07:33 +05:30
|
|
|
const char *name;
|
2015-11-25 13:25:16 +00:00
|
|
|
unsigned enabled;
|
2021-05-12 11:36:55 +00:00
|
|
|
const void *data;
|
2014-07-21 15:52:16 +01:00
|
|
|
};
|
|
|
|
|
2021-05-12 11:36:55 +00:00
|
|
|
#define TEST_CASE(fn) { NULL, NULL, fn, NULL, #fn, 1, NULL }
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2021-05-12 11:36:55 +00:00
|
|
|
#define TEST_CASE_NAMED(name, fn) { NULL, NULL, fn, NULL, name, 1, NULL }
|
2015-02-13 11:38:13 +01:00
|
|
|
|
2017-03-19 19:07:33 +05:30
|
|
|
#define TEST_CASE_ST(setup, teardown, testcase) \
|
2021-05-12 11:36:55 +00:00
|
|
|
{ setup, teardown, testcase, NULL, #testcase, 1, NULL }
|
2015-11-25 13:25:16 +00:00
|
|
|
|
2021-05-12 11:36:55 +00:00
|
|
|
#define TEST_CASE_WITH_DATA(setup, teardown, testcase, data) \
|
|
|
|
{ setup, teardown, NULL, testcase, #testcase, 1, data }
|
2015-11-25 13:25:16 +00:00
|
|
|
|
2021-05-12 11:36:55 +00:00
|
|
|
#define TEST_CASE_DISABLED(fn) { NULL, NULL, fn, NULL, #fn, 0, NULL }
|
2015-11-25 13:25:16 +00:00
|
|
|
|
2017-03-19 19:07:33 +05:30
|
|
|
#define TEST_CASE_ST_DISABLED(setup, teardown, testcase) \
|
2021-05-12 11:36:55 +00:00
|
|
|
{ setup, teardown, testcase, NULL, #testcase, 0, NULL }
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2021-05-12 11:36:55 +00:00
|
|
|
#define TEST_CASES_END() { NULL, NULL, NULL, NULL, NULL, 0, NULL }
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2017-12-08 14:21:21 +01:00
|
|
|
static inline void
|
|
|
|
debug_hexdump(FILE *file, const char *title, const void *buf, size_t len)
|
|
|
|
{
|
|
|
|
if (rte_log_get_global_level() == RTE_LOG_DEBUG)
|
|
|
|
rte_hexdump(file, title, buf, len);
|
|
|
|
}
|
2016-06-15 15:11:20 +01:00
|
|
|
|
2014-07-21 15:52:16 +01:00
|
|
|
struct unit_test_suite {
|
|
|
|
const char *suite_name;
|
|
|
|
int (*setup)(void);
|
2015-11-25 13:25:16 +00:00
|
|
|
void (*teardown)(void);
|
2021-05-12 11:36:49 +00:00
|
|
|
unsigned int total;
|
|
|
|
unsigned int executed;
|
|
|
|
unsigned int succeeded;
|
|
|
|
unsigned int skipped;
|
|
|
|
unsigned int failed;
|
|
|
|
unsigned int unsupported;
|
2021-05-12 11:36:50 +00:00
|
|
|
struct unit_test_suite **unit_test_suites;
|
2014-07-21 15:52:16 +01:00
|
|
|
struct unit_test_case unit_test_cases[];
|
|
|
|
};
|
|
|
|
|
|
|
|
int unit_test_suite_runner(struct unit_test_suite *suite);
|
2018-01-11 17:50:33 +00:00
|
|
|
extern int last_test_result;
|
2014-07-21 15:52:16 +01:00
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
#define RECURSIVE_ENV_VAR "RTE_TEST_RECURSIVE"
|
|
|
|
|
2014-11-12 14:24:35 +08:00
|
|
|
#include <cmdline_parse.h>
|
|
|
|
#include <cmdline_parse_string.h>
|
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
extern const char *prgname;
|
|
|
|
|
2014-08-18 13:29:22 +02:00
|
|
|
int commands_init(void);
|
2021-01-27 17:42:55 +00:00
|
|
|
int command_valid(const char *cmd);
|
2014-08-18 13:29:22 +02:00
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
int test_mp_secondary(void);
|
2019-04-18 16:00:04 -05:00
|
|
|
int test_timer_secondary(void);
|
2012-12-20 00:00:00 +01:00
|
|
|
|
2014-11-12 14:24:35 +08:00
|
|
|
int test_set_rxtx_conf(cmdline_fixed_string_t mode);
|
|
|
|
int test_set_rxtx_anchor(cmdline_fixed_string_t type);
|
|
|
|
int test_set_rxtx_sc(cmdline_fixed_string_t type);
|
2012-09-04 13:54:00 +01:00
|
|
|
|
2014-08-18 13:29:22 +02:00
|
|
|
typedef int (test_callback)(void);
|
|
|
|
TAILQ_HEAD(test_commands_list, test_command);
|
|
|
|
struct test_command {
|
|
|
|
TAILQ_ENTRY(test_command) next;
|
|
|
|
const char *command;
|
|
|
|
test_callback *callback;
|
|
|
|
};
|
|
|
|
|
|
|
|
void add_test_command(struct test_command *t);
|
|
|
|
|
2016-07-13 14:38:13 +02:00
|
|
|
/* Register a test function with its command string */
|
|
|
|
#define REGISTER_TEST_COMMAND(cmd, func) \
|
|
|
|
static struct test_command test_struct_##cmd = { \
|
|
|
|
.command = RTE_STR(cmd), \
|
|
|
|
.callback = func, \
|
|
|
|
}; \
|
2019-03-18 04:15:56 +00:00
|
|
|
RTE_INIT(test_register_##cmd) \
|
2016-07-13 14:38:13 +02:00
|
|
|
{ \
|
|
|
|
add_test_command(&test_struct_##cmd); \
|
|
|
|
}
|
2014-08-18 13:29:22 +02:00
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
#endif
|