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:
Yuval Avnery 2020-07-29 18:09:57 +00:00 committed by Thomas Monjalon
parent 76e821a303
commit de06137cb2
8 changed files with 528 additions and 0 deletions

View File

@ -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

View File

@ -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

View File

@ -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
View 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
View 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;
}

View File

@ -0,0 +1,5 @@
# SPDX-License-Identifier: BSD-3-Clause
# Copyright 2020 Mellanox Technologies, Ltd
sources = files('main.c')
deps = ['regexdev']

View File

@ -17,3 +17,4 @@ DPDK Tools User Guides
cryptoperf
comp_perf
testeventdev
testregex

View 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