app/regex: add RegEx test application
Following the new RegEx class. There is a need to create a dedicated test application in order to validate this class and PMD. Unlike net device this application loads data from a file. This commit introduces the new RegEx test app. The basic app flow: 1. Configure the RegEx device to use one queue, and set the rule database, using precompiled file. 2. Allocate mbufs based on the requested number of jobs, each job will i get one mbuf. 3. Enqueue as much as possible jobs. 4. Dequeue jobs. 5. if the number of dequeue jobs < requested number of jobs job to step Signed-off-by: Ori Kam <orika@mellanox.com> Signed-off-by: Yuval Avnery <yuvalav@mellanox.com>
This commit is contained in:
parent
76e821a303
commit
de06137cb2
@ -453,6 +453,7 @@ F: doc/guides/compressdevs/features/default.ini
|
||||
RegEx API - EXPERIMENTAL
|
||||
M: Ori Kam <orika@mellanox.com>
|
||||
F: lib/librte_regexdev/
|
||||
F: app/test-regex/
|
||||
F: doc/guides/prog_guide/regexdev.rst
|
||||
F: doc/guides/regexdevs/features/default.ini
|
||||
|
||||
|
@ -13,6 +13,7 @@ DIRS-$(CONFIG_RTE_LIBRTE_FIB) += test-fib
|
||||
DIRS-$(CONFIG_RTE_TEST_FLOW_PERF) += test-flow-perf
|
||||
DIRS-$(CONFIG_RTE_LIBRTE_PIPELINE) += test-pipeline
|
||||
DIRS-$(CONFIG_RTE_LIBRTE_IPSEC) += test-sad
|
||||
DIRS-$(CONFIG_RTE_LIBRTE_REGEXDEV) += test-regex
|
||||
|
||||
ifeq ($(CONFIG_RTE_LIBRTE_BBDEV),y)
|
||||
DIRS-$(CONFIG_RTE_TEST_BBDEV) += test-bbdev
|
||||
|
@ -18,6 +18,7 @@ apps = [
|
||||
'test-flow-perf',
|
||||
'test-pipeline',
|
||||
'test-pmd',
|
||||
'test-regex',
|
||||
'test-sad']
|
||||
|
||||
# for BSD only
|
||||
|
13
app/test-regex/Makefile
Normal file
13
app/test-regex/Makefile
Normal file
@ -0,0 +1,13 @@
|
||||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
# Copyright 2020 Mellanox Technologies, Ltd
|
||||
|
||||
include $(RTE_SDK)/mk/rte.vars.mk
|
||||
|
||||
APP = dpdk-test-regex
|
||||
|
||||
CFLAGS += -O3
|
||||
CFLAGS += $(WERROR_FLAGS)
|
||||
|
||||
SRCS-y := main.c
|
||||
|
||||
include $(RTE_SDK)/mk/rte.app.mk
|
433
app/test-regex/main.c
Normal file
433
app/test-regex/main.c
Normal file
@ -0,0 +1,433 @@
|
||||
/* SPDX-License-Identifier: BSD-3-Clause
|
||||
* Copyright 2020 Mellanox Technologies, Ltd
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdarg.h>
|
||||
#include <ctype.h>
|
||||
#include <errno.h>
|
||||
#include <getopt.h>
|
||||
#include <signal.h>
|
||||
|
||||
#include <rte_eal.h>
|
||||
#include <rte_common.h>
|
||||
#include <rte_malloc.h>
|
||||
#include <rte_mempool.h>
|
||||
#include <rte_mbuf.h>
|
||||
#include <rte_cycles.h>
|
||||
#include <rte_regexdev.h>
|
||||
|
||||
#define MAX_FILE_NAME 255
|
||||
#define MBUF_CACHE_SIZE 256
|
||||
#define MBUF_SIZE (1 << 8)
|
||||
#define START_BURST_SIZE 32u
|
||||
|
||||
enum app_args {
|
||||
ARG_HELP,
|
||||
ARG_RULES_FILE_NAME,
|
||||
ARG_DATA_FILE_NAME,
|
||||
ARG_NUM_OF_JOBS,
|
||||
ARG_PERF_MODE,
|
||||
ARG_NUM_OF_ITERATIONS,
|
||||
};
|
||||
|
||||
static void
|
||||
usage(const char *prog_name)
|
||||
{
|
||||
printf("%s [EAL options] --\n"
|
||||
" --rules NAME: precompiled rules file\n"
|
||||
" --data NAME: data file to use\n"
|
||||
" --nb_jobs: number of jobs to use\n"
|
||||
" --perf N: only outputs the performance data\n"
|
||||
" --nb_iter N: number of iteration to run\n",
|
||||
prog_name);
|
||||
}
|
||||
|
||||
static void
|
||||
args_parse(int argc, char **argv, char *rules_file, char *data_file,
|
||||
uint32_t *nb_jobs, bool *perf_mode, uint32_t *nb_iterations)
|
||||
{
|
||||
char **argvopt;
|
||||
int opt;
|
||||
int opt_idx;
|
||||
size_t len;
|
||||
static struct option lgopts[] = {
|
||||
{ "help", 0, 0, ARG_HELP},
|
||||
/* Rules database file to load. */
|
||||
{ "rules", 1, 0, ARG_RULES_FILE_NAME},
|
||||
/* Data file to load. */
|
||||
{ "data", 1, 0, ARG_DATA_FILE_NAME},
|
||||
/* Number of jobs to create. */
|
||||
{ "nb_jobs", 1, 0, ARG_NUM_OF_JOBS},
|
||||
/* Perf test only */
|
||||
{ "perf", 0, 0, ARG_PERF_MODE},
|
||||
/* Number of iterations to run with perf test */
|
||||
{ "nb_iter", 1, 0, ARG_NUM_OF_ITERATIONS}
|
||||
};
|
||||
|
||||
argvopt = argv;
|
||||
while ((opt = getopt_long(argc, argvopt, "",
|
||||
lgopts, &opt_idx)) != EOF) {
|
||||
switch (opt) {
|
||||
case ARG_RULES_FILE_NAME:
|
||||
len = strnlen(optarg, MAX_FILE_NAME - 1);
|
||||
if (len == MAX_FILE_NAME)
|
||||
rte_exit(EXIT_FAILURE,
|
||||
"Rule file name to long max %d\n",
|
||||
MAX_FILE_NAME - 1);
|
||||
strncpy(rules_file, optarg, MAX_FILE_NAME - 1);
|
||||
break;
|
||||
case ARG_DATA_FILE_NAME:
|
||||
len = strnlen(optarg, MAX_FILE_NAME - 1);
|
||||
if (len == MAX_FILE_NAME)
|
||||
rte_exit(EXIT_FAILURE,
|
||||
"Data file name to long max %d\n",
|
||||
MAX_FILE_NAME - 1);
|
||||
strncpy(data_file, optarg, MAX_FILE_NAME - 1);
|
||||
break;
|
||||
case ARG_NUM_OF_JOBS:
|
||||
*nb_jobs = atoi(optarg);
|
||||
break;
|
||||
case ARG_PERF_MODE:
|
||||
*perf_mode = true;
|
||||
break;
|
||||
case ARG_NUM_OF_ITERATIONS:
|
||||
*nb_iterations = atoi(optarg);
|
||||
break;
|
||||
case ARG_HELP:
|
||||
usage("RegEx test app");
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "Invalid option: %s\n", argv[optind]);
|
||||
usage("RegEx test app");
|
||||
rte_exit(EXIT_FAILURE, "Invalid option\n");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!perf_mode)
|
||||
*nb_iterations = 1;
|
||||
}
|
||||
|
||||
static long
|
||||
read_file(char *file, char **buf)
|
||||
{
|
||||
FILE *fp;
|
||||
long buf_len = 0;
|
||||
size_t read_len;
|
||||
int res = 0;
|
||||
|
||||
fp = fopen(file, "r");
|
||||
if (!fp)
|
||||
return -EIO;
|
||||
if (fseek(fp, 0L, SEEK_END) == 0) {
|
||||
buf_len = ftell(fp);
|
||||
if (buf_len == -1) {
|
||||
res = EIO;
|
||||
goto error;
|
||||
}
|
||||
*buf = rte_malloc(NULL, sizeof(char) * (buf_len + 1), 4096);
|
||||
if (!*buf) {
|
||||
res = ENOMEM;
|
||||
goto error;
|
||||
}
|
||||
if (fseek(fp, 0L, SEEK_SET) != 0) {
|
||||
res = EIO;
|
||||
goto error;
|
||||
}
|
||||
read_len = fread(*buf, sizeof(char), buf_len, fp);
|
||||
if (read_len != (unsigned long)buf_len) {
|
||||
res = EIO;
|
||||
goto error;
|
||||
}
|
||||
}
|
||||
fclose(fp);
|
||||
return buf_len;
|
||||
error:
|
||||
printf("Error, can't open file %s\n, err = %d", file, res);
|
||||
if (fp)
|
||||
fclose(fp);
|
||||
if (*buf)
|
||||
rte_free(*buf);
|
||||
return -res;
|
||||
}
|
||||
|
||||
static int
|
||||
init_port(struct rte_mempool **mbuf_mp, uint32_t nb_jobs,
|
||||
uint16_t *nb_max_payload, char *rules_file, uint8_t *nb_max_matches)
|
||||
{
|
||||
uint16_t id;
|
||||
uint16_t num_devs;
|
||||
char *rules = NULL;
|
||||
long rules_len;
|
||||
struct rte_regexdev_info info;
|
||||
struct rte_regexdev_config dev_conf = {
|
||||
.nb_queue_pairs = 1,
|
||||
.nb_groups = 1,
|
||||
};
|
||||
struct rte_regexdev_qp_conf qp_conf = {
|
||||
.nb_desc = 1024,
|
||||
.qp_conf_flags = RTE_REGEX_QUEUE_PAIR_CFG_OOS_F,
|
||||
};
|
||||
int res = 0;
|
||||
|
||||
num_devs = rte_regexdev_count();
|
||||
if (num_devs == 0) {
|
||||
printf("Error, no devices detected.\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
*mbuf_mp = rte_pktmbuf_pool_create("mbuf_pool", nb_jobs, 0,
|
||||
0, MBUF_SIZE, rte_socket_id());
|
||||
if (*mbuf_mp == NULL) {
|
||||
printf("Error, can't create memory pool\n");
|
||||
res = -ENOMEM;
|
||||
goto error;
|
||||
}
|
||||
|
||||
rules_len = read_file(rules_file, &rules);
|
||||
if (rules_len < 0) {
|
||||
printf("Error, can't read rules files.\n");
|
||||
res = -EIO;
|
||||
goto error;
|
||||
}
|
||||
|
||||
for (id = 0; id < num_devs; id++) {
|
||||
res = rte_regexdev_info_get(id, &info);
|
||||
if (res != 0) {
|
||||
printf("Error, can't get device info.\n");
|
||||
goto error;
|
||||
}
|
||||
printf(":: initializing dev: %d\n", id);
|
||||
*nb_max_matches = info.max_matches;
|
||||
*nb_max_payload = info.max_payload_size;
|
||||
if (info.regexdev_capa & RTE_REGEXDEV_SUPP_MATCH_AS_END_F)
|
||||
dev_conf.dev_cfg_flags |= RTE_REGEXDEV_CFG_MATCH_AS_END_F;
|
||||
dev_conf.nb_max_matches = info.max_matches;
|
||||
dev_conf.nb_rules_per_group = info.max_rules_per_group;
|
||||
dev_conf.rule_db_len = rules_len;
|
||||
dev_conf.rule_db = rules;
|
||||
res = rte_regexdev_configure(id, &dev_conf);
|
||||
if (res < 0) {
|
||||
printf("Error, can't configure device %d.\n", id);
|
||||
goto error;
|
||||
}
|
||||
res = rte_regexdev_queue_pair_setup(id, 0, &qp_conf);
|
||||
if (res < 0) {
|
||||
printf("Error, can't setup queue pair for device %d.\n",
|
||||
id);
|
||||
goto error;
|
||||
}
|
||||
printf(":: initializing device: %d done\n", id);
|
||||
}
|
||||
rte_free(rules);
|
||||
return 0;
|
||||
error:
|
||||
if (rules)
|
||||
rte_free(rules);
|
||||
if (*mbuf_mp)
|
||||
rte_mempool_free(*mbuf_mp);
|
||||
return res;
|
||||
}
|
||||
|
||||
static void
|
||||
extbuf_free_cb(void *addr __rte_unused, void *fcb_opaque __rte_unused)
|
||||
{
|
||||
}
|
||||
|
||||
static int
|
||||
run_regex(struct rte_mempool *mbuf_mp, uint32_t nb_jobs,
|
||||
uint16_t nb_max_payload, bool perf_mode, uint32_t nb_iterations,
|
||||
char *data_file, uint8_t nb_max_matches)
|
||||
{
|
||||
char *buf = NULL;
|
||||
long buf_len;
|
||||
long job_len;
|
||||
uint32_t actual_jobs = 0;
|
||||
uint32_t i;
|
||||
struct rte_regex_ops **ops;
|
||||
uint16_t dev_id = 0;
|
||||
uint16_t qp_id = 0;
|
||||
uint8_t nb_matches;
|
||||
struct rte_regexdev_match *match;
|
||||
long pos = 0;
|
||||
unsigned long d_ind = 0;
|
||||
struct rte_mbuf_ext_shared_info shinfo;
|
||||
uint32_t total_enqueue = 0;
|
||||
uint32_t total_dequeue = 0;
|
||||
uint32_t total_matches = 0;
|
||||
int res = 0;
|
||||
time_t start;
|
||||
time_t end;
|
||||
double time;
|
||||
|
||||
shinfo.free_cb = extbuf_free_cb;
|
||||
|
||||
ops = rte_malloc(NULL, sizeof(*ops) * nb_jobs, 0);
|
||||
if (!ops) {
|
||||
printf("Error, can't allocate memory for ops.\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
/* Allocate the jobs and assign each job with an mbuf. */
|
||||
for (i = 0; i < nb_jobs; i++) {
|
||||
ops[i] = rte_malloc(NULL, sizeof(*ops[0]) + nb_max_matches *
|
||||
sizeof(struct rte_regexdev_match), 0);
|
||||
if (!ops[i]) {
|
||||
printf("Error, can't allocate memory for op.\n");
|
||||
res = -ENOMEM;
|
||||
goto end;
|
||||
}
|
||||
ops[i]->mbuf = rte_pktmbuf_alloc(mbuf_mp);
|
||||
if (!ops[i]->mbuf) {
|
||||
printf("Error, can't attach mbuf.\n");
|
||||
res = -ENOMEM;
|
||||
goto end;
|
||||
}
|
||||
}
|
||||
|
||||
buf_len = read_file(data_file, &buf);
|
||||
if (buf_len <= 0) {
|
||||
printf("Error, can't read file, or file is empty.\n");
|
||||
res = -EXIT_FAILURE;
|
||||
goto end;
|
||||
}
|
||||
|
||||
job_len = buf_len / nb_jobs;
|
||||
if (job_len == 0) {
|
||||
printf("Error, To many jobs, for the given input.\n");
|
||||
res = -EXIT_FAILURE;
|
||||
goto end;
|
||||
}
|
||||
|
||||
if (job_len > nb_max_payload) {
|
||||
printf("Error, not enough jobs to cover input.\n");
|
||||
res = -EXIT_FAILURE;
|
||||
goto end;
|
||||
}
|
||||
|
||||
/* Assign each mbuf with the data to handle. */
|
||||
for (i = 0; (pos < buf_len) && (i < nb_jobs) ; i++) {
|
||||
long act_job_len = RTE_MIN(job_len, buf_len - pos);
|
||||
rte_pktmbuf_attach_extbuf(ops[i]->mbuf, &buf[pos], 0,
|
||||
act_job_len, &shinfo);
|
||||
ops[i]->mbuf->data_len = job_len;
|
||||
ops[i]->mbuf->pkt_len = act_job_len;
|
||||
ops[i]->user_id = i;
|
||||
ops[i]->group_id0 = 1;
|
||||
pos += act_job_len;
|
||||
actual_jobs++;
|
||||
}
|
||||
|
||||
start = clock();
|
||||
for (i = 0; i < nb_iterations; i++) {
|
||||
total_enqueue = 0;
|
||||
total_dequeue = 0;
|
||||
while (total_dequeue < actual_jobs) {
|
||||
struct rte_regex_ops **cur_ops_to_enqueue = ops +
|
||||
total_enqueue;
|
||||
struct rte_regex_ops **cur_ops_to_dequeue = ops +
|
||||
total_dequeue;
|
||||
|
||||
if (actual_jobs - total_enqueue)
|
||||
total_enqueue += rte_regexdev_enqueue_burst
|
||||
(dev_id, qp_id, cur_ops_to_enqueue,
|
||||
actual_jobs - total_enqueue);
|
||||
|
||||
total_dequeue += rte_regexdev_dequeue_burst
|
||||
(dev_id, qp_id, cur_ops_to_dequeue,
|
||||
total_enqueue - total_dequeue);
|
||||
}
|
||||
}
|
||||
end = clock();
|
||||
time = ((double)end - start) / CLOCKS_PER_SEC;
|
||||
printf("Job len = %ld Bytes\n", job_len);
|
||||
printf("Time = %lf sec\n", time);
|
||||
printf("Perf = %lf Gbps\n",
|
||||
(((double)actual_jobs * job_len * nb_iterations * 8) / time) /
|
||||
1000000000.0);
|
||||
|
||||
if (!perf_mode) {
|
||||
/* Log results per job. */
|
||||
for (d_ind = 0; d_ind < total_dequeue; d_ind++) {
|
||||
nb_matches = ops[d_ind % actual_jobs]->nb_matches;
|
||||
printf("Job id %"PRIu64" number of matches = %d\n",
|
||||
ops[d_ind]->user_id, nb_matches);
|
||||
total_matches += nb_matches;
|
||||
match = ops[d_ind % actual_jobs]->matches;
|
||||
for (i = 0; i < nb_matches; i++) {
|
||||
printf("match %d, rule = %d, start = %d,len = %d\n",
|
||||
i, match->rule_id, match->start_offset,
|
||||
match->len);
|
||||
match++;
|
||||
}
|
||||
}
|
||||
printf("Total matches = %d\n", total_matches);
|
||||
printf("All Matches:\n");
|
||||
|
||||
/* Log absolute results. */
|
||||
for (d_ind = 0; d_ind < total_dequeue; d_ind++) {
|
||||
nb_matches = ops[d_ind % actual_jobs]->nb_matches;
|
||||
total_matches += nb_matches;
|
||||
match = ops[d_ind % actual_jobs]->matches;
|
||||
for (i = 0; i < nb_matches; i++) {
|
||||
printf("start = %ld, len = %d, rule = %d\n",
|
||||
match->start_offset + d_ind * job_len,
|
||||
match->len, match->rule_id);
|
||||
match++;
|
||||
}
|
||||
}
|
||||
}
|
||||
end:
|
||||
for (i = 0; i < actual_jobs; i++) {
|
||||
if (ops[i]) {
|
||||
if (ops[i]->mbuf)
|
||||
rte_pktmbuf_free(ops[i]->mbuf);
|
||||
rte_free(ops[i]);
|
||||
}
|
||||
}
|
||||
rte_free(ops);
|
||||
if (buf)
|
||||
rte_free(buf);
|
||||
return res;
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char **argv)
|
||||
{
|
||||
char rules_file[MAX_FILE_NAME];
|
||||
char data_file[MAX_FILE_NAME];
|
||||
struct rte_mempool *mbuf_mp = NULL;
|
||||
uint32_t nb_jobs = 0;
|
||||
uint16_t nb_max_payload = 0;
|
||||
bool perf_mode = 0;
|
||||
uint32_t nb_iterations = 0;
|
||||
uint8_t nb_max_matches = 0;
|
||||
int ret;
|
||||
|
||||
ret = rte_eal_init(argc, argv);
|
||||
if (ret < 0)
|
||||
rte_exit(EXIT_FAILURE, "EAL init failed\n");
|
||||
argc -= ret;
|
||||
argv += ret;
|
||||
if (argc > 1)
|
||||
args_parse(argc, argv, rules_file, data_file, &nb_jobs,
|
||||
&perf_mode, &nb_iterations);
|
||||
|
||||
ret = init_port(&mbuf_mp, nb_jobs, &nb_max_payload, rules_file,
|
||||
&nb_max_matches);
|
||||
if (ret < 0)
|
||||
rte_exit(EXIT_FAILURE, "init port failed\n");
|
||||
ret = run_regex(mbuf_mp, nb_jobs, nb_max_payload, perf_mode,
|
||||
nb_iterations, data_file, nb_max_matches);
|
||||
if (ret < 0) {
|
||||
rte_mempool_free(mbuf_mp);
|
||||
rte_exit(EXIT_FAILURE, "RegEx function failed\n");
|
||||
}
|
||||
rte_mempool_free(mbuf_mp);
|
||||
return EXIT_SUCCESS;
|
||||
}
|
5
app/test-regex/meson.build
Normal file
5
app/test-regex/meson.build
Normal file
@ -0,0 +1,5 @@
|
||||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
# Copyright 2020 Mellanox Technologies, Ltd
|
||||
|
||||
sources = files('main.c')
|
||||
deps = ['regexdev']
|
@ -17,3 +17,4 @@ DPDK Tools User Guides
|
||||
cryptoperf
|
||||
comp_perf
|
||||
testeventdev
|
||||
testregex
|
||||
|
73
doc/guides/tools/testregex.rst
Normal file
73
doc/guides/tools/testregex.rst
Normal file
@ -0,0 +1,73 @@
|
||||
.. SPDX-License-Identifier: BSD-3-Clause
|
||||
Copyright 2020 Mellanox Technologies, Ltd
|
||||
|
||||
dpdk-test-regex Tool
|
||||
====================
|
||||
|
||||
The ``dpdk-test-regex`` tool is a Data Plane Development Kit (DPDK)
|
||||
application that allows functional testing and performance measurement for
|
||||
the RegEx PMDs.
|
||||
The test supports only one core and one PMD.
|
||||
It is based on precompiled rule file, and an input file, both of them can
|
||||
be selected using command-line options.
|
||||
|
||||
In general case, each PMD has its own rule file.
|
||||
|
||||
The test outputs the following data:
|
||||
|
||||
* Performance, in gigabit per second.
|
||||
|
||||
* Matching results (rule id, position, length), for each job.
|
||||
|
||||
* Matching results in absolute location (rule id, position , length),
|
||||
relative to the start of the input data.
|
||||
|
||||
|
||||
Limitations
|
||||
~~~~~~~~~~~
|
||||
|
||||
* Only one queue is supported.
|
||||
|
||||
* Supports only precompiled rules.
|
||||
|
||||
|
||||
Application Options
|
||||
~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
``--rules NAME``
|
||||
precompiled rule file
|
||||
|
||||
``--data NAME``
|
||||
data file to use
|
||||
|
||||
``--nb_jobs N``
|
||||
number of jobs to use
|
||||
|
||||
``--perf N``
|
||||
only outputs the performance data
|
||||
|
||||
``--nb_iter N``
|
||||
number of iteration to run
|
||||
|
||||
``--help``
|
||||
print application options
|
||||
|
||||
|
||||
Running the Tool
|
||||
----------------
|
||||
|
||||
**Step 1: Compile a rule file**
|
||||
|
||||
In order for the RegEx to work it must have a precompiled rule file.
|
||||
to generate this file there is a need to use a RegEx compiler that matches the
|
||||
RegEx PMD.
|
||||
|
||||
**Step 2: Generate a data file**
|
||||
|
||||
The data file, will be used as a source data for the RegEx to work on.
|
||||
|
||||
**Step 3: Run the tool**
|
||||
|
||||
The tool has a number of command line options. Here is the sample command line::
|
||||
|
||||
./dpdk-test-regex -w 83:00.0 -- --rules rule_file.rof2 --data data_file.txt --job 100
|
Loading…
Reference in New Issue
Block a user