/* SPDX-License-Identifier: BSD-3-Clause * Copyright(c) 2018 - 2019 Intel Corporation */ #include #include #include #include #include #include #include #include #include #include #include #include "test_compressdev_test_buffer.h" #include "test.h" #define DIV_CEIL(a, b) ((a) / (b) + ((a) % (b) != 0)) #define DEFAULT_WINDOW_SIZE 15 #define DEFAULT_MEM_LEVEL 8 #define MAX_DEQD_RETRIES 10 #define DEQUEUE_WAIT_TIME 10000 /* * 30% extra size for compressed data compared to original data, * in case data size cannot be reduced and it is actually bigger * due to the compress block headers */ #define COMPRESS_BUF_SIZE_RATIO 1.3 #define COMPRESS_BUF_SIZE_RATIO_OVERFLOW 0.2 #define NUM_LARGE_MBUFS 16 #define SMALL_SEG_SIZE 256 #define MAX_SEGS 16 #define NUM_OPS 16 #define NUM_MAX_XFORMS 16 #define NUM_MAX_INFLIGHT_OPS 128 #define CACHE_SIZE 0 #define ZLIB_CRC_CHECKSUM_WINDOW_BITS 31 #define ZLIB_HEADER_SIZE 2 #define ZLIB_TRAILER_SIZE 4 #define GZIP_HEADER_SIZE 10 #define GZIP_TRAILER_SIZE 8 #define OUT_OF_SPACE_BUF 1 #define MAX_MBUF_SEGMENT_SIZE 65535 #define MAX_DATA_MBUF_SIZE (MAX_MBUF_SEGMENT_SIZE - RTE_PKTMBUF_HEADROOM) #define NUM_BIG_MBUFS 4 #define BIG_DATA_TEST_SIZE (MAX_DATA_MBUF_SIZE * NUM_BIG_MBUFS / 2) const char * huffman_type_strings[] = { [RTE_COMP_HUFFMAN_DEFAULT] = "PMD default", [RTE_COMP_HUFFMAN_FIXED] = "Fixed", [RTE_COMP_HUFFMAN_DYNAMIC] = "Dynamic" }; enum zlib_direction { ZLIB_NONE, ZLIB_COMPRESS, ZLIB_DECOMPRESS, ZLIB_ALL }; enum varied_buff { LB_BOTH = 0, /* both input and output are linear*/ SGL_BOTH, /* both input and output are chained */ SGL_TO_LB, /* input buffer is chained */ LB_TO_SGL /* output buffer is chained */ }; enum overflow_test { OVERFLOW_DISABLED, OVERFLOW_ENABLED }; enum operation_type { OPERATION_COMPRESSION, OPERATION_DECOMPRESSION }; struct priv_op_data { uint16_t orig_idx; }; struct comp_testsuite_params { struct rte_mempool *large_mbuf_pool; struct rte_mempool *small_mbuf_pool; struct rte_mempool *big_mbuf_pool; struct rte_mempool *op_pool; struct rte_comp_xform *def_comp_xform; struct rte_comp_xform *def_decomp_xform; }; struct interim_data_params { const char * const *test_bufs; unsigned int num_bufs; uint16_t *buf_idx; struct rte_comp_xform **compress_xforms; struct rte_comp_xform **decompress_xforms; unsigned int num_xforms; }; struct test_data_params { enum rte_comp_op_type compress_state; enum rte_comp_op_type decompress_state; enum varied_buff buff_type; enum zlib_direction zlib_dir; unsigned int out_of_space; unsigned int big_data; /* stateful decompression specific parameters */ unsigned int decompress_output_block_size; unsigned int decompress_steps_max; /* external mbufs specific parameters */ unsigned int use_external_mbufs; unsigned int inbuf_data_size; const struct rte_memzone *inbuf_memzone; const struct rte_memzone *compbuf_memzone; const struct rte_memzone *uncompbuf_memzone; /* overflow test activation */ enum overflow_test overflow; }; struct test_private_arrays { struct rte_mbuf **uncomp_bufs; struct rte_mbuf **comp_bufs; struct rte_comp_op **ops; struct rte_comp_op **ops_processed; void **priv_xforms; uint64_t *compress_checksum; uint32_t *compressed_data_size; void **stream; char **all_decomp_data; unsigned int *decomp_produced_data_size; uint16_t num_priv_xforms; }; static struct comp_testsuite_params testsuite_params = { 0 }; static void testsuite_teardown(void) { struct comp_testsuite_params *ts_params = &testsuite_params; if (rte_mempool_in_use_count(ts_params->large_mbuf_pool)) RTE_LOG(ERR, USER1, "Large mbuf pool still has unfreed bufs\n"); if (rte_mempool_in_use_count(ts_params->small_mbuf_pool)) RTE_LOG(ERR, USER1, "Small mbuf pool still has unfreed bufs\n"); if (rte_mempool_in_use_count(ts_params->big_mbuf_pool)) RTE_LOG(ERR, USER1, "Big mbuf pool still has unfreed bufs\n"); if (rte_mempool_in_use_count(ts_params->op_pool)) RTE_LOG(ERR, USER1, "op pool still has unfreed ops\n"); rte_mempool_free(ts_params->large_mbuf_pool); rte_mempool_free(ts_params->small_mbuf_pool); rte_mempool_free(ts_params->big_mbuf_pool); rte_mempool_free(ts_params->op_pool); rte_free(ts_params->def_comp_xform); rte_free(ts_params->def_decomp_xform); } static int testsuite_setup(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint32_t max_buf_size = 0; unsigned int i; if (rte_compressdev_count() == 0) { RTE_LOG(WARNING, USER1, "Need at least one compress device\n"); return TEST_SKIPPED; } RTE_LOG(NOTICE, USER1, "Running tests on device %s\n", rte_compressdev_name_get(0)); for (i = 0; i < RTE_DIM(compress_test_bufs); i++) max_buf_size = RTE_MAX(max_buf_size, strlen(compress_test_bufs[i]) + 1); /* * Buffers to be used in compression and decompression. * Since decompressed data might be larger than * compressed data (due to block header), * buffers should be big enough for both cases. */ max_buf_size *= COMPRESS_BUF_SIZE_RATIO; ts_params->large_mbuf_pool = rte_pktmbuf_pool_create("large_mbuf_pool", NUM_LARGE_MBUFS, CACHE_SIZE, 0, max_buf_size + RTE_PKTMBUF_HEADROOM, rte_socket_id()); if (ts_params->large_mbuf_pool == NULL) { RTE_LOG(ERR, USER1, "Large mbuf pool could not be created\n"); return TEST_FAILED; } /* Create mempool with smaller buffers for SGL testing */ ts_params->small_mbuf_pool = rte_pktmbuf_pool_create("small_mbuf_pool", NUM_LARGE_MBUFS * MAX_SEGS, CACHE_SIZE, 0, SMALL_SEG_SIZE + RTE_PKTMBUF_HEADROOM, rte_socket_id()); if (ts_params->small_mbuf_pool == NULL) { RTE_LOG(ERR, USER1, "Small mbuf pool could not be created\n"); goto exit; } /* Create mempool with big buffers for SGL testing */ ts_params->big_mbuf_pool = rte_pktmbuf_pool_create("big_mbuf_pool", NUM_BIG_MBUFS + 1, CACHE_SIZE, 0, MAX_MBUF_SEGMENT_SIZE, rte_socket_id()); if (ts_params->big_mbuf_pool == NULL) { RTE_LOG(ERR, USER1, "Big mbuf pool could not be created\n"); goto exit; } ts_params->op_pool = rte_comp_op_pool_create("op_pool", NUM_OPS, 0, sizeof(struct priv_op_data), rte_socket_id()); if (ts_params->op_pool == NULL) { RTE_LOG(ERR, USER1, "Operation pool could not be created\n"); goto exit; } ts_params->def_comp_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (ts_params->def_comp_xform == NULL) { RTE_LOG(ERR, USER1, "Default compress xform could not be created\n"); goto exit; } ts_params->def_decomp_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (ts_params->def_decomp_xform == NULL) { RTE_LOG(ERR, USER1, "Default decompress xform could not be created\n"); goto exit; } /* Initializes default values for compress/decompress xforms */ ts_params->def_comp_xform->type = RTE_COMP_COMPRESS; ts_params->def_comp_xform->compress.algo = RTE_COMP_ALGO_DEFLATE, ts_params->def_comp_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_DEFAULT; ts_params->def_comp_xform->compress.level = RTE_COMP_LEVEL_PMD_DEFAULT; ts_params->def_comp_xform->compress.chksum = RTE_COMP_CHECKSUM_NONE; ts_params->def_comp_xform->compress.window_size = DEFAULT_WINDOW_SIZE; ts_params->def_decomp_xform->type = RTE_COMP_DECOMPRESS; ts_params->def_decomp_xform->decompress.algo = RTE_COMP_ALGO_DEFLATE, ts_params->def_decomp_xform->decompress.chksum = RTE_COMP_CHECKSUM_NONE; ts_params->def_decomp_xform->decompress.window_size = DEFAULT_WINDOW_SIZE; return TEST_SUCCESS; exit: testsuite_teardown(); return TEST_FAILED; } static int generic_ut_setup(void) { /* Configure compressdev (one device, one queue pair) */ struct rte_compressdev_config config = { .socket_id = rte_socket_id(), .nb_queue_pairs = 1, .max_nb_priv_xforms = NUM_MAX_XFORMS, .max_nb_streams = 1 }; if (rte_compressdev_configure(0, &config) < 0) { RTE_LOG(ERR, USER1, "Device configuration failed\n"); return -1; } if (rte_compressdev_queue_pair_setup(0, 0, NUM_MAX_INFLIGHT_OPS, rte_socket_id()) < 0) { RTE_LOG(ERR, USER1, "Queue pair setup failed\n"); return -1; } if (rte_compressdev_start(0) < 0) { RTE_LOG(ERR, USER1, "Device could not be started\n"); return -1; } return 0; } static void generic_ut_teardown(void) { rte_compressdev_stop(0); if (rte_compressdev_close(0) < 0) RTE_LOG(ERR, USER1, "Device could not be closed\n"); } static int test_compressdev_invalid_configuration(void) { struct rte_compressdev_config invalid_config; struct rte_compressdev_config valid_config = { .socket_id = rte_socket_id(), .nb_queue_pairs = 1, .max_nb_priv_xforms = NUM_MAX_XFORMS, .max_nb_streams = 1 }; struct rte_compressdev_info dev_info; /* Invalid configuration with 0 queue pairs */ memcpy(&invalid_config, &valid_config, sizeof(struct rte_compressdev_config)); invalid_config.nb_queue_pairs = 0; TEST_ASSERT_FAIL(rte_compressdev_configure(0, &invalid_config), "Device configuration was successful " "with no queue pairs (invalid)\n"); /* * Invalid configuration with too many queue pairs * (if there is an actual maximum number of queue pairs) */ rte_compressdev_info_get(0, &dev_info); if (dev_info.max_nb_queue_pairs != 0) { memcpy(&invalid_config, &valid_config, sizeof(struct rte_compressdev_config)); invalid_config.nb_queue_pairs = dev_info.max_nb_queue_pairs + 1; TEST_ASSERT_FAIL(rte_compressdev_configure(0, &invalid_config), "Device configuration was successful " "with too many queue pairs (invalid)\n"); } /* Invalid queue pair setup, with no number of queue pairs set */ TEST_ASSERT_FAIL(rte_compressdev_queue_pair_setup(0, 0, NUM_MAX_INFLIGHT_OPS, rte_socket_id()), "Queue pair setup was successful " "with no queue pairs set (invalid)\n"); return TEST_SUCCESS; } static int compare_buffers(const char *buffer1, uint32_t buffer1_len, const char *buffer2, uint32_t buffer2_len) { if (buffer1_len != buffer2_len) { RTE_LOG(ERR, USER1, "Buffer lengths are different\n"); return -1; } if (memcmp(buffer1, buffer2, buffer1_len) != 0) { RTE_LOG(ERR, USER1, "Buffers are different\n"); return -1; } return 0; } /* * Maps compressdev and Zlib flush flags */ static int map_zlib_flush_flag(enum rte_comp_flush_flag flag) { switch (flag) { case RTE_COMP_FLUSH_NONE: return Z_NO_FLUSH; case RTE_COMP_FLUSH_SYNC: return Z_SYNC_FLUSH; case RTE_COMP_FLUSH_FULL: return Z_FULL_FLUSH; case RTE_COMP_FLUSH_FINAL: return Z_FINISH; /* * There should be only the values above, * so this should never happen */ default: return -1; } } static int compress_zlib(struct rte_comp_op *op, const struct rte_comp_xform *xform, int mem_level) { z_stream stream; int zlib_flush; int strategy, window_bits, comp_level; int ret = TEST_FAILED; uint8_t *single_src_buf = NULL; uint8_t *single_dst_buf = NULL; /* initialize zlib stream */ stream.zalloc = Z_NULL; stream.zfree = Z_NULL; stream.opaque = Z_NULL; if (xform->compress.deflate.huffman == RTE_COMP_HUFFMAN_FIXED) strategy = Z_FIXED; else strategy = Z_DEFAULT_STRATEGY; /* * Window bits is the base two logarithm of the window size (in bytes). * When doing raw DEFLATE, this number will be negative. */ window_bits = -(xform->compress.window_size); if (xform->compress.chksum == RTE_COMP_CHECKSUM_ADLER32) window_bits *= -1; else if (xform->compress.chksum == RTE_COMP_CHECKSUM_CRC32) window_bits = ZLIB_CRC_CHECKSUM_WINDOW_BITS; comp_level = xform->compress.level; if (comp_level != RTE_COMP_LEVEL_NONE) ret = deflateInit2(&stream, comp_level, Z_DEFLATED, window_bits, mem_level, strategy); else ret = deflateInit(&stream, Z_NO_COMPRESSION); if (ret != Z_OK) { printf("Zlib deflate could not be initialized\n"); goto exit; } /* Assuming stateless operation */ /* SGL Input */ if (op->m_src->nb_segs > 1) { single_src_buf = rte_malloc(NULL, rte_pktmbuf_pkt_len(op->m_src), 0); if (single_src_buf == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be allocated\n"); goto exit; } if (rte_pktmbuf_read(op->m_src, op->src.offset, rte_pktmbuf_pkt_len(op->m_src) - op->src.offset, single_src_buf) == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be read entirely\n"); goto exit; } stream.avail_in = op->src.length; stream.next_in = single_src_buf; } else { stream.avail_in = op->src.length; stream.next_in = rte_pktmbuf_mtod_offset(op->m_src, uint8_t *, op->src.offset); } /* SGL output */ if (op->m_dst->nb_segs > 1) { single_dst_buf = rte_malloc(NULL, rte_pktmbuf_pkt_len(op->m_dst), 0); if (single_dst_buf == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be allocated\n"); goto exit; } stream.avail_out = op->m_dst->pkt_len; stream.next_out = single_dst_buf; } else {/* linear output */ stream.avail_out = op->m_dst->data_len; stream.next_out = rte_pktmbuf_mtod_offset(op->m_dst, uint8_t *, op->dst.offset); } /* Stateless operation, all buffer will be compressed in one go */ zlib_flush = map_zlib_flush_flag(op->flush_flag); ret = deflate(&stream, zlib_flush); if (stream.avail_in != 0) { RTE_LOG(ERR, USER1, "Buffer could not be read entirely\n"); goto exit; } if (ret != Z_STREAM_END) goto exit; /* Copy data to destination SGL */ if (op->m_dst->nb_segs > 1) { uint32_t remaining_data = stream.total_out; uint8_t *src_data = single_dst_buf; struct rte_mbuf *dst_buf = op->m_dst; while (remaining_data > 0) { uint8_t *dst_data = rte_pktmbuf_mtod_offset(dst_buf, uint8_t *, op->dst.offset); /* Last segment */ if (remaining_data < dst_buf->data_len) { memcpy(dst_data, src_data, remaining_data); remaining_data = 0; } else { memcpy(dst_data, src_data, dst_buf->data_len); remaining_data -= dst_buf->data_len; src_data += dst_buf->data_len; dst_buf = dst_buf->next; } } } op->consumed = stream.total_in; if (xform->compress.chksum == RTE_COMP_CHECKSUM_ADLER32) { rte_pktmbuf_adj(op->m_dst, ZLIB_HEADER_SIZE); rte_pktmbuf_trim(op->m_dst, ZLIB_TRAILER_SIZE); op->produced = stream.total_out - (ZLIB_HEADER_SIZE + ZLIB_TRAILER_SIZE); } else if (xform->compress.chksum == RTE_COMP_CHECKSUM_CRC32) { rte_pktmbuf_adj(op->m_dst, GZIP_HEADER_SIZE); rte_pktmbuf_trim(op->m_dst, GZIP_TRAILER_SIZE); op->produced = stream.total_out - (GZIP_HEADER_SIZE + GZIP_TRAILER_SIZE); } else op->produced = stream.total_out; op->status = RTE_COMP_OP_STATUS_SUCCESS; op->output_chksum = stream.adler; deflateReset(&stream); ret = 0; exit: deflateEnd(&stream); rte_free(single_src_buf); rte_free(single_dst_buf); return ret; } static int decompress_zlib(struct rte_comp_op *op, const struct rte_comp_xform *xform) { z_stream stream; int window_bits; int zlib_flush; int ret = TEST_FAILED; uint8_t *single_src_buf = NULL; uint8_t *single_dst_buf = NULL; /* initialize zlib stream */ stream.zalloc = Z_NULL; stream.zfree = Z_NULL; stream.opaque = Z_NULL; /* * Window bits is the base two logarithm of the window size (in bytes). * When doing raw DEFLATE, this number will be negative. */ window_bits = -(xform->decompress.window_size); ret = inflateInit2(&stream, window_bits); if (ret != Z_OK) { printf("Zlib deflate could not be initialized\n"); goto exit; } /* Assuming stateless operation */ /* SGL */ if (op->m_src->nb_segs > 1) { single_src_buf = rte_malloc(NULL, rte_pktmbuf_pkt_len(op->m_src), 0); if (single_src_buf == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be allocated\n"); goto exit; } single_dst_buf = rte_malloc(NULL, rte_pktmbuf_pkt_len(op->m_dst), 0); if (single_dst_buf == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be allocated\n"); goto exit; } if (rte_pktmbuf_read(op->m_src, 0, rte_pktmbuf_pkt_len(op->m_src), single_src_buf) == NULL) { RTE_LOG(ERR, USER1, "Buffer could not be read entirely\n"); goto exit; } stream.avail_in = op->src.length; stream.next_in = single_src_buf; stream.avail_out = rte_pktmbuf_pkt_len(op->m_dst); stream.next_out = single_dst_buf; } else { stream.avail_in = op->src.length; stream.next_in = rte_pktmbuf_mtod(op->m_src, uint8_t *); stream.avail_out = op->m_dst->data_len; stream.next_out = rte_pktmbuf_mtod(op->m_dst, uint8_t *); } /* Stateless operation, all buffer will be compressed in one go */ zlib_flush = map_zlib_flush_flag(op->flush_flag); ret = inflate(&stream, zlib_flush); if (stream.avail_in != 0) { RTE_LOG(ERR, USER1, "Buffer could not be read entirely\n"); goto exit; } if (ret != Z_STREAM_END) goto exit; if (op->m_src->nb_segs > 1) { uint32_t remaining_data = stream.total_out; uint8_t *src_data = single_dst_buf; struct rte_mbuf *dst_buf = op->m_dst; while (remaining_data > 0) { uint8_t *dst_data = rte_pktmbuf_mtod(dst_buf, uint8_t *); /* Last segment */ if (remaining_data < dst_buf->data_len) { memcpy(dst_data, src_data, remaining_data); remaining_data = 0; } else { memcpy(dst_data, src_data, dst_buf->data_len); remaining_data -= dst_buf->data_len; src_data += dst_buf->data_len; dst_buf = dst_buf->next; } } } op->consumed = stream.total_in; op->produced = stream.total_out; op->status = RTE_COMP_OP_STATUS_SUCCESS; inflateReset(&stream); ret = 0; exit: inflateEnd(&stream); return ret; } static int prepare_sgl_bufs(const char *test_buf, struct rte_mbuf *head_buf, uint32_t total_data_size, struct rte_mempool *small_mbuf_pool, struct rte_mempool *large_mbuf_pool, uint8_t limit_segs_in_sgl, uint16_t seg_size) { uint32_t remaining_data = total_data_size; uint16_t num_remaining_segs = DIV_CEIL(remaining_data, seg_size); struct rte_mempool *pool; struct rte_mbuf *next_seg; uint32_t data_size; char *buf_ptr; const char *data_ptr = test_buf; uint16_t i; int ret; if (limit_segs_in_sgl != 0 && num_remaining_segs > limit_segs_in_sgl) num_remaining_segs = limit_segs_in_sgl - 1; /* * Allocate data in the first segment (header) and * copy data if test buffer is provided */ if (remaining_data < seg_size) data_size = remaining_data; else data_size = seg_size; buf_ptr = rte_pktmbuf_append(head_buf, data_size); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Not enough space in the 1st buffer\n"); return -1; } if (data_ptr != NULL) { /* Copy characters without NULL terminator */ strncpy(buf_ptr, data_ptr, data_size); data_ptr += data_size; } remaining_data -= data_size; num_remaining_segs--; /* * Allocate the rest of the segments, * copy the rest of the data and chain the segments. */ for (i = 0; i < num_remaining_segs; i++) { if (i == (num_remaining_segs - 1)) { /* last segment */ if (remaining_data > seg_size) pool = large_mbuf_pool; else pool = small_mbuf_pool; data_size = remaining_data; } else { data_size = seg_size; pool = small_mbuf_pool; } next_seg = rte_pktmbuf_alloc(pool); if (next_seg == NULL) { RTE_LOG(ERR, USER1, "New segment could not be allocated " "from the mempool\n"); return -1; } buf_ptr = rte_pktmbuf_append(next_seg, data_size); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Not enough space in the buffer\n"); rte_pktmbuf_free(next_seg); return -1; } if (data_ptr != NULL) { /* Copy characters without NULL terminator */ strncpy(buf_ptr, data_ptr, data_size); data_ptr += data_size; } remaining_data -= data_size; ret = rte_pktmbuf_chain(head_buf, next_seg); if (ret != 0) { rte_pktmbuf_free(next_seg); RTE_LOG(ERR, USER1, "Segment could not chained\n"); return -1; } } return 0; } static void extbuf_free_callback(void *addr __rte_unused, void *opaque __rte_unused) { } static int test_run_enqueue_dequeue(struct rte_comp_op **ops, struct rte_comp_op **ops_processed, unsigned int num_bufs) { uint16_t num_enqd, num_deqd, num_total_deqd; unsigned int deqd_retries = 0; /* Enqueue and dequeue all operations */ num_enqd = rte_compressdev_enqueue_burst(0, 0, ops, num_bufs); if (num_enqd < num_bufs) { RTE_LOG(ERR, USER1, "Some operations could not be enqueued\n"); return -1; } num_total_deqd = 0; do { /* * If retrying a dequeue call, wait for 10 ms to allow * enough time to the driver to process the operations */ if (deqd_retries != 0) { /* * Avoid infinite loop if not all the * operations get out of the device */ if (deqd_retries == MAX_DEQD_RETRIES) { RTE_LOG(ERR, USER1, "Not all operations could be dequeued\n"); return -1; } usleep(DEQUEUE_WAIT_TIME); } num_deqd = rte_compressdev_dequeue_burst(0, 0, &ops_processed[num_total_deqd], num_bufs); num_total_deqd += num_deqd; deqd_retries++; } while (num_total_deqd < num_enqd); return 0; } /** * Arrays initialization. Input buffers preparation for compression. * * API that initializes all the private arrays to NULL * and allocates input buffers to perform compression operations. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 0: On success. * - -1: On error. */ static int test_setup_com_bufs(const struct interim_data_params *int_data, const struct test_data_params *test_data, const struct test_private_arrays *test_priv_data) { /* local variables: */ unsigned int i; uint32_t data_size; char *buf_ptr; int ret; char **all_decomp_data = test_priv_data->all_decomp_data; struct comp_testsuite_params *ts_params = &testsuite_params; /* from int_data: */ const char * const *test_bufs = int_data->test_bufs; unsigned int num_bufs = int_data->num_bufs; /* from test_data: */ unsigned int buff_type = test_data->buff_type; unsigned int big_data = test_data->big_data; /* from test_priv_data: */ struct rte_mbuf **uncomp_bufs = test_priv_data->uncomp_bufs; struct rte_mempool *buf_pool; static struct rte_mbuf_ext_shared_info inbuf_info; size_t array_size = sizeof(void *) * num_bufs; /* Initialize all arrays to NULL */ memset(test_priv_data->uncomp_bufs, 0, array_size); memset(test_priv_data->comp_bufs, 0, array_size); memset(test_priv_data->ops, 0, array_size); memset(test_priv_data->ops_processed, 0, array_size); memset(test_priv_data->priv_xforms, 0, array_size); memset(test_priv_data->compressed_data_size, 0, sizeof(uint32_t) * num_bufs); if (test_data->decompress_state == RTE_COMP_OP_STATEFUL) { data_size = strlen(test_bufs[0]) + 1; *all_decomp_data = rte_malloc(NULL, data_size, RTE_CACHE_LINE_SIZE); } if (big_data) buf_pool = ts_params->big_mbuf_pool; else if (buff_type == SGL_BOTH) buf_pool = ts_params->small_mbuf_pool; else buf_pool = ts_params->large_mbuf_pool; /* for compression uncomp_bufs is used as a source buffer */ /* allocation from buf_pool (mempool type) */ ret = rte_pktmbuf_alloc_bulk(buf_pool, uncomp_bufs, num_bufs); if (ret < 0) { RTE_LOG(ERR, USER1, "Source mbufs could not be allocated " "from the mempool\n"); return -1; } if (test_data->use_external_mbufs) { inbuf_info.free_cb = extbuf_free_callback; inbuf_info.fcb_opaque = NULL; rte_mbuf_ext_refcnt_set(&inbuf_info, 1); for (i = 0; i < num_bufs; i++) { rte_pktmbuf_attach_extbuf(uncomp_bufs[i], test_data->inbuf_memzone->addr, test_data->inbuf_memzone->iova, test_data->inbuf_data_size, &inbuf_info); buf_ptr = rte_pktmbuf_append(uncomp_bufs[i], test_data->inbuf_data_size); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Append extra bytes to the source mbuf failed\n"); return -1; } } } else if (buff_type == SGL_BOTH || buff_type == SGL_TO_LB) { for (i = 0; i < num_bufs; i++) { data_size = strlen(test_bufs[i]) + 1; if (prepare_sgl_bufs(test_bufs[i], uncomp_bufs[i], data_size, big_data ? buf_pool : ts_params->small_mbuf_pool, big_data ? buf_pool : ts_params->large_mbuf_pool, big_data ? 0 : MAX_SEGS, big_data ? MAX_DATA_MBUF_SIZE : SMALL_SEG_SIZE) < 0) return -1; } } else { for (i = 0; i < num_bufs; i++) { data_size = strlen(test_bufs[i]) + 1; buf_ptr = rte_pktmbuf_append(uncomp_bufs[i], data_size); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Append extra bytes to the source mbuf failed\n"); return -1; } strlcpy(buf_ptr, test_bufs[i], data_size); } } return 0; } /** * Data size calculation (for both compression and decompression). * * Calculate size of anticipated output buffer required for both * compression and decompression operations based on input int_data. * * @param op_type * Operation type: compress or decompress * @param out_of_space_and_zlib * Boolean value to switch into "out of space" buffer if set. * To test "out-of-space" data size, zlib_decompress must be set as well. * @param test_priv_data * A container used for aggregation all the private test arrays. * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param i * current buffer index * @return * data size */ static inline uint32_t test_mbufs_calculate_data_size( enum operation_type op_type, unsigned int out_of_space_and_zlib, const struct test_private_arrays *test_priv_data, const struct interim_data_params *int_data, const struct test_data_params *test_data, unsigned int i) { /* local variables: */ uint32_t data_size; struct priv_op_data *priv_data; float ratio; uint8_t not_zlib_compr; /* true if zlib isn't current compression dev */ enum overflow_test overflow = test_data->overflow; /* from test_priv_data: */ struct rte_comp_op **ops_processed = test_priv_data->ops_processed; /* from int_data: */ const char * const *test_bufs = int_data->test_bufs; if (out_of_space_and_zlib) data_size = OUT_OF_SPACE_BUF; else { if (op_type == OPERATION_COMPRESSION) { not_zlib_compr = (test_data->zlib_dir == ZLIB_DECOMPRESS || test_data->zlib_dir == ZLIB_NONE); ratio = (not_zlib_compr && (overflow == OVERFLOW_ENABLED)) ? COMPRESS_BUF_SIZE_RATIO_OVERFLOW : COMPRESS_BUF_SIZE_RATIO; data_size = strlen(test_bufs[i]) * ratio; } else { priv_data = (struct priv_op_data *) (ops_processed[i] + 1); data_size = strlen(test_bufs[priv_data->orig_idx]) + 1; } } return data_size; } /** * Memory buffers preparation (for both compression and decompression). * * Function allocates output buffers to perform compression * or decompression operations depending on value of op_type. * * @param op_type * Operation type: compress or decompress * @param out_of_space_and_zlib * Boolean value to switch into "out of space" buffer if set. * To test "out-of-space" data size, zlib_decompress must be set as well. * @param test_priv_data * A container used for aggregation all the private test arrays. * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param current_extbuf_info, * The structure containing all the information related to external mbufs * @return * - 0: On success. * - -1: On error. */ static int test_setup_output_bufs( enum operation_type op_type, unsigned int out_of_space_and_zlib, const struct test_private_arrays *test_priv_data, const struct interim_data_params *int_data, const struct test_data_params *test_data, struct rte_mbuf_ext_shared_info *current_extbuf_info) { /* local variables: */ unsigned int i; uint32_t data_size; int ret; char *buf_ptr; /* from test_priv_data: */ struct rte_mbuf **current_bufs; /* from int_data: */ unsigned int num_bufs = int_data->num_bufs; /* from test_data: */ unsigned int buff_type = test_data->buff_type; unsigned int big_data = test_data->big_data; const struct rte_memzone *current_memzone; struct comp_testsuite_params *ts_params = &testsuite_params; struct rte_mempool *buf_pool; if (big_data) buf_pool = ts_params->big_mbuf_pool; else if (buff_type == SGL_BOTH) buf_pool = ts_params->small_mbuf_pool; else buf_pool = ts_params->large_mbuf_pool; if (op_type == OPERATION_COMPRESSION) current_bufs = test_priv_data->comp_bufs; else current_bufs = test_priv_data->uncomp_bufs; /* the mbufs allocation*/ ret = rte_pktmbuf_alloc_bulk(buf_pool, current_bufs, num_bufs); if (ret < 0) { RTE_LOG(ERR, USER1, "Destination mbufs could not be allocated " "from the mempool\n"); return -1; } if (test_data->use_external_mbufs) { current_extbuf_info->free_cb = extbuf_free_callback; current_extbuf_info->fcb_opaque = NULL; rte_mbuf_ext_refcnt_set(current_extbuf_info, 1); if (op_type == OPERATION_COMPRESSION) current_memzone = test_data->compbuf_memzone; else current_memzone = test_data->uncompbuf_memzone; for (i = 0; i < num_bufs; i++) { rte_pktmbuf_attach_extbuf(current_bufs[i], current_memzone->addr, current_memzone->iova, current_memzone->len, current_extbuf_info); rte_pktmbuf_append(current_bufs[i], current_memzone->len); } } else { for (i = 0; i < num_bufs; i++) { /* data size calculation */ data_size = test_mbufs_calculate_data_size( op_type, out_of_space_and_zlib, test_priv_data, int_data, test_data, i); /* data allocation */ if (buff_type == SGL_BOTH || buff_type == LB_TO_SGL) { ret = prepare_sgl_bufs(NULL, current_bufs[i], data_size, big_data ? buf_pool : ts_params->small_mbuf_pool, big_data ? buf_pool : ts_params->large_mbuf_pool, big_data ? 0 : MAX_SEGS, big_data ? MAX_DATA_MBUF_SIZE : SMALL_SEG_SIZE); if (ret < 0) return -1; } else { buf_ptr = rte_pktmbuf_append(current_bufs[i], data_size); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Append extra bytes to the destination mbuf failed\n"); return -1; } } } } return 0; } /** * The main compression function. * * Function performs compression operation. * Operation(s) configuration, depending on CLI parameters. * Operation(s) processing. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 0: On success. * - -1: On error. */ static int test_deflate_comp_run(const struct interim_data_params *int_data, const struct test_data_params *test_data, const struct test_private_arrays *test_priv_data) { /* local variables: */ struct priv_op_data *priv_data; unsigned int i; uint16_t num_priv_xforms = 0; int ret; int ret_status = 0; char *buf_ptr; struct comp_testsuite_params *ts_params = &testsuite_params; /* from test_data: */ enum rte_comp_op_type operation_type = test_data->compress_state; unsigned int zlib_compress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_COMPRESS); /* from int_data: */ struct rte_comp_xform **compress_xforms = int_data->compress_xforms; unsigned int num_xforms = int_data->num_xforms; unsigned int num_bufs = int_data->num_bufs; /* from test_priv_data: */ struct rte_mbuf **comp_bufs = test_priv_data->comp_bufs; struct rte_mbuf **uncomp_bufs = test_priv_data->uncomp_bufs; struct rte_comp_op **ops = test_priv_data->ops; struct rte_comp_op **ops_processed = test_priv_data->ops_processed; void **priv_xforms = test_priv_data->priv_xforms; const struct rte_compressdev_capabilities *capa = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); /* Build the compression operations */ ret = rte_comp_op_bulk_alloc(ts_params->op_pool, ops, num_bufs); if (ret < 0) { RTE_LOG(ERR, USER1, "Compress operations could not be allocated " "from the mempool\n"); ret_status = -1; goto exit; } for (i = 0; i < num_bufs; i++) { ops[i]->m_src = uncomp_bufs[i]; ops[i]->m_dst = comp_bufs[i]; ops[i]->src.offset = 0; ops[i]->src.length = rte_pktmbuf_pkt_len(uncomp_bufs[i]); ops[i]->dst.offset = 0; if (operation_type == RTE_COMP_OP_STATELESS) { ops[i]->flush_flag = RTE_COMP_FLUSH_FINAL; } else { RTE_LOG(ERR, USER1, "Compression: stateful operations are not " "supported in these tests yet\n"); ret_status = -1; goto exit; } ops[i]->input_chksum = 0; /* * Store original operation index in private data, * since ordering does not have to be maintained, * when dequeueing from compressdev, so a comparison * at the end of the test can be done. */ priv_data = (struct priv_op_data *) (ops[i] + 1); priv_data->orig_idx = i; } /* Compress data (either with Zlib API or compressdev API */ if (zlib_compress) { for (i = 0; i < num_bufs; i++) { const struct rte_comp_xform *compress_xform = compress_xforms[i % num_xforms]; ret = compress_zlib(ops[i], compress_xform, DEFAULT_MEM_LEVEL); if (ret < 0) { ret_status = -1; goto exit; } ops_processed[i] = ops[i]; } } else { /* Create compress private xform data */ for (i = 0; i < num_xforms; i++) { ret = rte_compressdev_private_xform_create(0, (const struct rte_comp_xform *) compress_xforms[i], &priv_xforms[i]); if (ret < 0) { RTE_LOG(ERR, USER1, "Compression private xform " "could not be created\n"); ret_status = -1; goto exit; } num_priv_xforms++; } if (capa->comp_feature_flags & RTE_COMP_FF_SHAREABLE_PRIV_XFORM) { /* Attach shareable private xform data to ops */ for (i = 0; i < num_bufs; i++) ops[i]->private_xform = priv_xforms[i % num_xforms]; } else { /* Create rest of the private xforms for the other ops */ for (i = num_xforms; i < num_bufs; i++) { ret = rte_compressdev_private_xform_create(0, compress_xforms[i % num_xforms], &priv_xforms[i]); if (ret < 0) { RTE_LOG(ERR, USER1, "Compression private xform " "could not be created\n"); ret_status = -1; goto exit; } num_priv_xforms++; } /* Attach non shareable private xform data to ops */ for (i = 0; i < num_bufs; i++) ops[i]->private_xform = priv_xforms[i]; } recovery_lb: ret = test_run_enqueue_dequeue(ops, ops_processed, num_bufs); if (ret < 0) { RTE_LOG(ERR, USER1, "Compression: enqueue/dequeue operation failed\n"); ret_status = -1; goto exit; } for (i = 0; i < num_bufs; i++) { test_priv_data->compressed_data_size[i] += ops_processed[i]->produced; if (ops_processed[i]->status == RTE_COMP_OP_STATUS_OUT_OF_SPACE_RECOVERABLE) { ops[i]->status = RTE_COMP_OP_STATUS_NOT_PROCESSED; ops[i]->src.offset += ops_processed[i]->consumed; ops[i]->src.length -= ops_processed[i]->consumed; ops[i]->dst.offset += ops_processed[i]->produced; buf_ptr = rte_pktmbuf_append( ops[i]->m_dst, ops_processed[i]->produced); if (buf_ptr == NULL) { RTE_LOG(ERR, USER1, "Data recovery: append extra bytes to the current mbuf failed\n"); ret_status = -1; goto exit; } goto recovery_lb; } } } exit: /* Free resources */ if (ret_status < 0) for (i = 0; i < num_bufs; i++) { rte_comp_op_free(ops[i]); ops_processed[i] = NULL; } /* Free compress private xforms */ for (i = 0; i < num_priv_xforms; i++) { if (priv_xforms[i] != NULL) { rte_compressdev_private_xform_free(0, priv_xforms[i]); priv_xforms[i] = NULL; } } return ret_status; } /** * Prints out the test report. Memory freeing. * * Called after successful compression. * Operation(s) status validation and decompression buffers freeing. * -1 returned if function fail. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 2: Some operation is not supported * - 1: Decompression should be skipped * - 0: On success. * - -1: On error. */ static int test_deflate_comp_finalize(const struct interim_data_params *int_data, const struct test_data_params *test_data, const struct test_private_arrays *test_priv_data) { /* local variables: */ unsigned int i; struct priv_op_data *priv_data; /* from int_data: */ unsigned int num_xforms = int_data->num_xforms; struct rte_comp_xform **compress_xforms = int_data->compress_xforms; uint16_t *buf_idx = int_data->buf_idx; unsigned int num_bufs = int_data->num_bufs; /* from test_priv_data: */ struct rte_comp_op **ops_processed = test_priv_data->ops_processed; uint64_t *compress_checksum = test_priv_data->compress_checksum; struct rte_mbuf **uncomp_bufs = test_priv_data->uncomp_bufs; struct rte_comp_op **ops = test_priv_data->ops; /* from test_data: */ unsigned int out_of_space = test_data->out_of_space; unsigned int zlib_compress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_COMPRESS); unsigned int zlib_decompress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_DECOMPRESS); for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *)(ops_processed[i] + 1); uint16_t xform_idx = priv_data->orig_idx % num_xforms; const struct rte_comp_compress_xform *compress_xform = &compress_xforms[xform_idx]->compress; enum rte_comp_huffman huffman_type = compress_xform->deflate.huffman; char engine[] = "zlib (directly, not PMD)"; if (zlib_decompress) strlcpy(engine, "PMD", sizeof(engine)); RTE_LOG(DEBUG, USER1, "Buffer %u compressed by %s from %u to" " %u bytes (level = %d, huffman = %s)\n", buf_idx[priv_data->orig_idx], engine, ops_processed[i]->consumed, ops_processed[i]->produced, compress_xform->level, huffman_type_strings[huffman_type]); RTE_LOG(DEBUG, USER1, "Compression ratio = %.2f\n", ops_processed[i]->consumed == 0 ? 0 : (float)ops_processed[i]->produced / ops_processed[i]->consumed * 100); if (compress_xform->chksum != RTE_COMP_CHECKSUM_NONE) compress_checksum[i] = ops_processed[i]->output_chksum; ops[i] = NULL; } /* * Check operation status and free source mbufs (destination mbuf and * compress operation information is needed for the decompression stage) */ for (i = 0; i < num_bufs; i++) { if (out_of_space && !zlib_compress) { if (ops_processed[i]->status != RTE_COMP_OP_STATUS_OUT_OF_SPACE_TERMINATED) { RTE_LOG(ERR, USER1, "Operation without expected out of " "space status error\n"); return -1; } else continue; } if (ops_processed[i]->status != RTE_COMP_OP_STATUS_SUCCESS) { if (test_data->overflow == OVERFLOW_ENABLED) { if (ops_processed[i]->status == RTE_COMP_OP_STATUS_OUT_OF_SPACE_TERMINATED) { RTE_LOG(INFO, USER1, "Out-of-space-recoverable functionality" " is not supported on this device\n"); return 2; } } RTE_LOG(ERR, USER1, "Some operations were not successful\n"); return -1; } priv_data = (struct priv_op_data *)(ops_processed[i] + 1); rte_pktmbuf_free(uncomp_bufs[priv_data->orig_idx]); uncomp_bufs[priv_data->orig_idx] = NULL; } if (out_of_space && !zlib_compress) return 1; return 0; } /** * The main decompression function. * * Function performs decompression operation. * Operation(s) configuration, depending on CLI parameters. * Operation(s) processing. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 0: On success. * - -1: On error. */ static int test_deflate_decomp_run(const struct interim_data_params *int_data, const struct test_data_params *test_data, struct test_private_arrays *test_priv_data) { /* local variables: */ struct priv_op_data *priv_data; unsigned int i; uint16_t num_priv_xforms = 0; int ret; int ret_status = 0; struct comp_testsuite_params *ts_params = &testsuite_params; /* from test_data: */ enum rte_comp_op_type operation_type = test_data->decompress_state; unsigned int zlib_decompress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_DECOMPRESS); /* from int_data: */ struct rte_comp_xform **decompress_xforms = int_data->decompress_xforms; unsigned int num_xforms = int_data->num_xforms; unsigned int num_bufs = int_data->num_bufs; /* from test_priv_data: */ struct rte_mbuf **uncomp_bufs = test_priv_data->uncomp_bufs; struct rte_comp_op **ops = test_priv_data->ops; struct rte_comp_op **ops_processed = test_priv_data->ops_processed; void **priv_xforms = test_priv_data->priv_xforms; uint32_t *compressed_data_size = test_priv_data->compressed_data_size; void **stream = test_priv_data->stream; const struct rte_compressdev_capabilities *capa = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); ret = rte_comp_op_bulk_alloc(ts_params->op_pool, ops, num_bufs); if (ret < 0) { RTE_LOG(ERR, USER1, "Decompress operations could not be allocated " "from the mempool\n"); ret_status = -1; goto exit; } /* Source buffer is the compressed data from the previous operations */ for (i = 0; i < num_bufs; i++) { ops[i]->m_src = ops_processed[i]->m_dst; ops[i]->m_dst = uncomp_bufs[i]; ops[i]->src.offset = 0; /* * Set the length of the compressed data to the * number of bytes that were produced in the previous stage */ if (compressed_data_size[i]) ops[i]->src.length = compressed_data_size[i]; else ops[i]->src.length = ops_processed[i]->produced; ops[i]->dst.offset = 0; if (operation_type == RTE_COMP_OP_STATELESS) { ops[i]->flush_flag = RTE_COMP_FLUSH_FINAL; ops[i]->op_type = RTE_COMP_OP_STATELESS; } else if (!zlib_decompress) { ops[i]->flush_flag = RTE_COMP_FLUSH_SYNC; ops[i]->op_type = RTE_COMP_OP_STATEFUL; } else { RTE_LOG(ERR, USER1, "Decompression: stateful operations are" " not supported in these tests yet\n"); ret_status = -1; goto exit; } ops[i]->input_chksum = 0; /* * Copy private data from previous operations, * to keep the pointer to the original buffer */ memcpy(ops[i] + 1, ops_processed[i] + 1, sizeof(struct priv_op_data)); } /* * Free the previous compress operations, * as they are not needed anymore */ rte_comp_op_bulk_free(ops_processed, num_bufs); /* Decompress data (either with Zlib API or compressdev API */ if (zlib_decompress) { for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *)(ops[i] + 1); uint16_t xform_idx = priv_data->orig_idx % num_xforms; const struct rte_comp_xform *decompress_xform = decompress_xforms[xform_idx]; ret = decompress_zlib(ops[i], decompress_xform); if (ret < 0) { ret_status = -1; goto exit; } ops_processed[i] = ops[i]; } } else { if (operation_type == RTE_COMP_OP_STATELESS) { /* Create decompress private xform data */ for (i = 0; i < num_xforms; i++) { ret = rte_compressdev_private_xform_create(0, (const struct rte_comp_xform *) decompress_xforms[i], &priv_xforms[i]); if (ret < 0) { RTE_LOG(ERR, USER1, "Decompression private xform " "could not be created\n"); ret_status = -1; goto exit; } num_priv_xforms++; } if (capa->comp_feature_flags & RTE_COMP_FF_SHAREABLE_PRIV_XFORM) { /* Attach shareable private xform data to ops */ for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *) (ops[i] + 1); uint16_t xform_idx = priv_data->orig_idx % num_xforms; ops[i]->private_xform = priv_xforms[xform_idx]; } } else { /* Create rest of the private xforms */ /* for the other ops */ for (i = num_xforms; i < num_bufs; i++) { ret = rte_compressdev_private_xform_create(0, decompress_xforms[i % num_xforms], &priv_xforms[i]); if (ret < 0) { RTE_LOG(ERR, USER1, "Decompression private xform" " could not be created\n"); ret_status = -1; goto exit; } num_priv_xforms++; } /* Attach non shareable private xform data */ /* to ops */ for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *) (ops[i] + 1); uint16_t xform_idx = priv_data->orig_idx; ops[i]->private_xform = priv_xforms[xform_idx]; } } } else { /* Create a stream object for stateful decompression */ ret = rte_compressdev_stream_create(0, decompress_xforms[0], stream); if (ret < 0) { RTE_LOG(ERR, USER1, "Decompression stream could not be created, error %d\n", ret); ret_status = -1; goto exit; } /* Attach stream to ops */ for (i = 0; i < num_bufs; i++) ops[i]->stream = *stream; } test_priv_data->num_priv_xforms = num_priv_xforms; } exit: return ret_status; } /** * Prints out the test report. Memory freeing. * * Called after successful decompression. * Operation(s) status validation and compression buffers freeing. * -1 returned if function fail. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 2: Next step must be executed by the caller (stateful decompression only) * - 1: On success (caller should stop and exit) * - 0: On success. * - -1: On error. */ static int test_deflate_decomp_finalize(const struct interim_data_params *int_data, const struct test_data_params *test_data, const struct test_private_arrays *test_priv_data) { /* local variables: */ unsigned int i; struct priv_op_data *priv_data; static unsigned int step; /* from int_data: */ uint16_t *buf_idx = int_data->buf_idx; unsigned int num_bufs = int_data->num_bufs; const char * const *test_bufs = int_data->test_bufs; struct rte_comp_xform **compress_xforms = int_data->compress_xforms; /* from test_priv_data: */ struct rte_comp_op **ops_processed = test_priv_data->ops_processed; struct rte_mbuf **comp_bufs = test_priv_data->comp_bufs; struct rte_comp_op **ops = test_priv_data->ops; uint64_t *compress_checksum = test_priv_data->compress_checksum; unsigned int *decomp_produced_data_size = test_priv_data->decomp_produced_data_size; char **all_decomp_data = test_priv_data->all_decomp_data; /* from test_data: */ unsigned int out_of_space = test_data->out_of_space; enum rte_comp_op_type operation_type = test_data->decompress_state; unsigned int zlib_compress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_COMPRESS); unsigned int zlib_decompress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_DECOMPRESS); for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *)(ops_processed[i] + 1); char engine[] = "zlib, (directly, no PMD)"; if (zlib_compress) strlcpy(engine, "pmd", sizeof(engine)); RTE_LOG(DEBUG, USER1, "Buffer %u decompressed by %s from %u to %u bytes\n", buf_idx[priv_data->orig_idx], engine, ops_processed[i]->consumed, ops_processed[i]->produced); ops[i] = NULL; } /* * Check operation status and free source mbuf (destination mbuf and * compress operation information is still needed) */ for (i = 0; i < num_bufs; i++) { if (out_of_space && !zlib_decompress) { if (ops_processed[i]->status != RTE_COMP_OP_STATUS_OUT_OF_SPACE_TERMINATED) { RTE_LOG(ERR, USER1, "Operation without expected out of " "space status error\n"); return -1; } else continue; } if (operation_type == RTE_COMP_OP_STATEFUL && (ops_processed[i]->status == RTE_COMP_OP_STATUS_OUT_OF_SPACE_RECOVERABLE || ops_processed[i]->status == RTE_COMP_OP_STATUS_SUCCESS)) { /* collect the output into all_decomp_data */ const void *ptr = rte_pktmbuf_read( ops_processed[i]->m_dst, ops_processed[i]->dst.offset, ops_processed[i]->produced, *all_decomp_data + *decomp_produced_data_size); if (ptr != *all_decomp_data + *decomp_produced_data_size) rte_memcpy(*all_decomp_data + *decomp_produced_data_size, ptr, ops_processed[i]->produced); *decomp_produced_data_size += ops_processed[i]->produced; if (ops_processed[i]->src.length > ops_processed[i]->consumed) { if (ops_processed[i]->status == RTE_COMP_OP_STATUS_SUCCESS) { RTE_LOG(ERR, USER1, "Operation finished too early\n"); return -1; } step++; if (step >= test_data->decompress_steps_max) { RTE_LOG(ERR, USER1, "Operation exceeded maximum steps\n"); return -1; } ops[i] = ops_processed[i]; ops[i]->status = RTE_COMP_OP_STATUS_NOT_PROCESSED; ops[i]->src.offset += ops_processed[i]->consumed; ops[i]->src.length -= ops_processed[i]->consumed; /* repeat the operation */ //goto next_step; return 2; } else { /* Compare the original stream with the */ /* decompressed stream (in size and the data) */ priv_data = (struct priv_op_data *) (ops_processed[i] + 1); const char *buf1 = test_bufs[priv_data->orig_idx]; const char *buf2 = *all_decomp_data; if (compare_buffers(buf1, strlen(buf1) + 1, buf2, *decomp_produced_data_size) < 0) return -1; /* Test checksums */ if (compress_xforms[0]->compress.chksum != RTE_COMP_CHECKSUM_NONE) { if (ops_processed[i]->output_chksum != compress_checksum[i]) { RTE_LOG(ERR, USER1, "The checksums differ\n" "Compression Checksum: %" PRIu64 "\tDecompression " "Checksum: %" PRIu64 "\n", compress_checksum[i], ops_processed[i]->output_chksum); return -1; } } } } else if (ops_processed[i]->status != RTE_COMP_OP_STATUS_SUCCESS) { RTE_LOG(ERR, USER1, "Some operations were not successful\n"); return -1; } priv_data = (struct priv_op_data *)(ops_processed[i] + 1); rte_pktmbuf_free(comp_bufs[priv_data->orig_idx]); comp_bufs[priv_data->orig_idx] = NULL; } if (out_of_space && !zlib_decompress) return 1; return 0; } /** * Validation of the output (compression/decompression) data. * * The function compares the source stream with the output stream, * after decompression, to check if compression/decompression * was correct. * -1 returned if function fail. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @param test_priv_data * A container used for aggregation all the private test arrays. * @return * - 0: On success. * - -1: On error. */ static int test_results_validation(const struct interim_data_params *int_data, const struct test_data_params *test_data, const struct test_private_arrays *test_priv_data) { /* local variables: */ unsigned int i; struct priv_op_data *priv_data; const char *buf1; const char *buf2; char *contig_buf = NULL; uint32_t data_size; /* from int_data: */ struct rte_comp_xform **compress_xforms = int_data->compress_xforms; unsigned int num_bufs = int_data->num_bufs; const char * const *test_bufs = int_data->test_bufs; /* from test_priv_data: */ uint64_t *compress_checksum = test_priv_data->compress_checksum; struct rte_comp_op **ops_processed = test_priv_data->ops_processed; /* * Compare the original stream with the decompressed stream * (in size and the data) */ for (i = 0; i < num_bufs; i++) { priv_data = (struct priv_op_data *)(ops_processed[i] + 1); buf1 = test_data->use_external_mbufs ? test_data->inbuf_memzone->addr : test_bufs[priv_data->orig_idx]; data_size = test_data->use_external_mbufs ? test_data->inbuf_data_size : strlen(buf1) + 1; contig_buf = rte_malloc(NULL, ops_processed[i]->produced, 0); if (contig_buf == NULL) { RTE_LOG(ERR, USER1, "Contiguous buffer could not " "be allocated\n"); goto exit; } buf2 = rte_pktmbuf_read(ops_processed[i]->m_dst, 0, ops_processed[i]->produced, contig_buf); if (compare_buffers(buf1, data_size, buf2, ops_processed[i]->produced) < 0) goto exit; /* Test checksums */ if (compress_xforms[0]->compress.chksum != RTE_COMP_CHECKSUM_NONE) { if (ops_processed[i]->output_chksum != compress_checksum[i]) { RTE_LOG(ERR, USER1, "The checksums differ\n" "Compression Checksum: %" PRIu64 "\tDecompression " "Checksum: %" PRIu64 "\n", compress_checksum[i], ops_processed[i]->output_chksum); goto exit; } } rte_free(contig_buf); contig_buf = NULL; } return 0; exit: rte_free(contig_buf); return -1; } /** * Compresses and decompresses input stream with compressdev API and Zlib API * * Basic test function. Common for all the functional tests. * -1 returned if function fail. * * @param int_data * Interim data containing session/transformation objects. * @param test_data * The test parameters set by users (command line parameters). * @return * - 1: Some operation not supported * - 0: On success. * - -1: On error. */ static int test_deflate_comp_decomp(const struct interim_data_params *int_data, const struct test_data_params *test_data) { unsigned int num_bufs = int_data->num_bufs; unsigned int out_of_space = test_data->out_of_space; void *stream = NULL; char *all_decomp_data = NULL; unsigned int decomp_produced_data_size = 0; int ret_status = -1; int ret; struct rte_mbuf *uncomp_bufs[num_bufs]; struct rte_mbuf *comp_bufs[num_bufs]; struct rte_comp_op *ops[num_bufs]; struct rte_comp_op *ops_processed[num_bufs]; void *priv_xforms[num_bufs]; unsigned int i; uint64_t compress_checksum[num_bufs]; uint32_t compressed_data_size[num_bufs]; char *contig_buf = NULL; struct rte_mbuf_ext_shared_info compbuf_info; struct rte_mbuf_ext_shared_info decompbuf_info; const struct rte_compressdev_capabilities *capa; /* Compressing with CompressDev */ unsigned int zlib_compress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_COMPRESS); unsigned int zlib_decompress = (test_data->zlib_dir == ZLIB_ALL || test_data->zlib_dir == ZLIB_DECOMPRESS); struct test_private_arrays test_priv_data; test_priv_data.uncomp_bufs = uncomp_bufs; test_priv_data.comp_bufs = comp_bufs; test_priv_data.ops = ops; test_priv_data.ops_processed = ops_processed; test_priv_data.priv_xforms = priv_xforms; test_priv_data.compress_checksum = compress_checksum; test_priv_data.compressed_data_size = compressed_data_size; test_priv_data.stream = &stream; test_priv_data.all_decomp_data = &all_decomp_data; test_priv_data.decomp_produced_data_size = &decomp_produced_data_size; test_priv_data.num_priv_xforms = 0; /* it's used for deompression only */ capa = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); if (capa == NULL) { RTE_LOG(ERR, USER1, "Compress device does not support DEFLATE\n"); return -1; } //test_objects_init(&test_priv_data, num_bufs); /* Prepare the source mbufs with the data */ ret = test_setup_com_bufs(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } /* COMPRESSION */ /* Prepare output (destination) mbufs for compressed data */ ret = test_setup_output_bufs( OPERATION_COMPRESSION, out_of_space == 1 && !zlib_compress, &test_priv_data, int_data, test_data, &compbuf_info); if (ret < 0) { ret_status = -1; goto exit; } /* Run compression */ ret = test_deflate_comp_run(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } ret = test_deflate_comp_finalize(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } else if (ret == 1) { ret_status = 0; goto exit; } else if (ret == 2) { ret_status = 1; /* some operation not supported */ goto exit; } /* DECOMPRESSION */ /* Prepare output (destination) mbufs for decompressed data */ ret = test_setup_output_bufs( OPERATION_DECOMPRESSION, out_of_space == 1 && !zlib_decompress, &test_priv_data, int_data, test_data, &decompbuf_info); if (ret < 0) { ret_status = -1; goto exit; } /* Run decompression */ ret = test_deflate_decomp_run(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } if (!zlib_decompress) { next_step: /* next step for stateful decompression only */ ret = test_run_enqueue_dequeue(ops, ops_processed, num_bufs); if (ret < 0) { ret_status = -1; RTE_LOG(ERR, USER1, "Decompression: enqueue/dequeue operation failed\n"); } } ret = test_deflate_decomp_finalize(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } else if (ret == 1) { ret_status = 0; goto exit; } else if (ret == 2) { goto next_step; } /* FINAL PROCESSING */ ret = test_results_validation(int_data, test_data, &test_priv_data); if (ret < 0) { ret_status = -1; goto exit; } ret_status = 0; exit: /* Free resources */ if (stream != NULL) rte_compressdev_stream_free(0, stream); if (all_decomp_data != NULL) rte_free(all_decomp_data); /* Free compress private xforms */ for (i = 0; i < test_priv_data.num_priv_xforms; i++) { if (priv_xforms[i] != NULL) { rte_compressdev_private_xform_free(0, priv_xforms[i]); priv_xforms[i] = NULL; } } for (i = 0; i < num_bufs; i++) { rte_pktmbuf_free(uncomp_bufs[i]); rte_pktmbuf_free(comp_bufs[i]); rte_comp_op_free(ops[i]); rte_comp_op_free(ops_processed[i]); } rte_free(contig_buf); return ret_status; } static int test_compressdev_deflate_stateless_fixed(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i; int ret; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_HUFFMAN_FIXED) == 0) return -ENOTSUP; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); compress_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_FIXED; struct interim_data_params int_data = { NULL, 1, NULL, &compress_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } ret = TEST_SUCCESS; exit: rte_free(compress_xform); return ret; } static int test_compressdev_deflate_stateless_dynamic(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i; int ret; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_HUFFMAN_DYNAMIC) == 0) return -ENOTSUP; if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); compress_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_DYNAMIC; struct interim_data_params int_data = { NULL, 1, NULL, &compress_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } ret = TEST_SUCCESS; exit: rte_free(compress_xform); return ret; } static int test_compressdev_deflate_stateless_multi_op(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t num_bufs = RTE_DIM(compress_test_bufs); uint16_t buf_idx[num_bufs]; uint16_t i; int ret; for (i = 0; i < num_bufs; i++) buf_idx[i] = i; struct interim_data_params int_data = { compress_test_bufs, num_bufs, buf_idx, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; return TEST_SUCCESS; } static int test_compressdev_deflate_stateless_multi_level(void) { struct comp_testsuite_params *ts_params = &testsuite_params; unsigned int level; uint16_t i; int ret; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); struct interim_data_params int_data = { NULL, 1, NULL, &compress_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; for (level = RTE_COMP_LEVEL_MIN; level <= RTE_COMP_LEVEL_MAX; level++) { compress_xform->compress.level = level; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } } ret = TEST_SUCCESS; exit: rte_free(compress_xform); return ret; } #define NUM_XFORMS 3 static int test_compressdev_deflate_stateless_multi_xform(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t num_bufs = NUM_XFORMS; struct rte_comp_xform *compress_xforms[NUM_XFORMS] = {NULL}; struct rte_comp_xform *decompress_xforms[NUM_XFORMS] = {NULL}; const char *test_buffers[NUM_XFORMS]; uint16_t i; unsigned int level = RTE_COMP_LEVEL_MIN; uint16_t buf_idx[num_bufs]; int ret; /* Create multiple xforms with various levels */ for (i = 0; i < NUM_XFORMS; i++) { compress_xforms[i] = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xforms[i] == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(compress_xforms[i], ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); compress_xforms[i]->compress.level = level; level++; decompress_xforms[i] = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (decompress_xforms[i] == NULL) { RTE_LOG(ERR, USER1, "Decompress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(decompress_xforms[i], ts_params->def_decomp_xform, sizeof(struct rte_comp_xform)); } for (i = 0; i < NUM_XFORMS; i++) { buf_idx[i] = 0; /* Use the same buffer in all sessions */ test_buffers[i] = compress_test_bufs[0]; } struct interim_data_params int_data = { test_buffers, num_bufs, buf_idx, compress_xforms, decompress_xforms, NUM_XFORMS }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; /* Compress with compressdev, decompress with Zlib */ ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; ret = TEST_SUCCESS; exit: for (i = 0; i < NUM_XFORMS; i++) { rte_free(compress_xforms[i]); rte_free(decompress_xforms[i]); } return ret; } static int test_compressdev_deflate_stateless_sgl(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i; int ret; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) == 0) return -ENOTSUP; struct interim_data_params int_data = { NULL, 1, NULL, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = SGL_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_LB_OUT) { /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; test_data.buff_type = SGL_TO_LB; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; test_data.buff_type = SGL_TO_LB; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; } if (capab->comp_feature_flags & RTE_COMP_FF_OOP_LB_IN_SGL_OUT) { /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; test_data.buff_type = LB_TO_SGL; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; test_data.buff_type = LB_TO_SGL; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) return ret; } } return TEST_SUCCESS; } static int test_compressdev_deflate_stateless_checksum(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i; int ret; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); /* Check if driver supports any checksum */ if ((capab->comp_feature_flags & RTE_COMP_FF_CRC32_CHECKSUM) == 0 && (capab->comp_feature_flags & RTE_COMP_FF_ADLER32_CHECKSUM) == 0 && (capab->comp_feature_flags & RTE_COMP_FF_CRC32_ADLER32_CHECKSUM) == 0) return -ENOTSUP; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); return TEST_FAILED; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); struct rte_comp_xform *decompress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (decompress_xform == NULL) { RTE_LOG(ERR, USER1, "Decompress xform could not be created\n"); rte_free(compress_xform); return TEST_FAILED; } memcpy(decompress_xform, ts_params->def_decomp_xform, sizeof(struct rte_comp_xform)); struct interim_data_params int_data = { NULL, 1, NULL, &compress_xform, &decompress_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_DISABLED }; /* Check if driver supports crc32 checksum and test */ if ((capab->comp_feature_flags & RTE_COMP_FF_CRC32_CHECKSUM)) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_CRC32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_CRC32; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { /* Compress with compressdev, decompress with Zlib */ int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Generate zlib checksum and test against selected * drivers decompression checksum */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Generate compression and decompression * checksum of selected driver */ test_data.zlib_dir = ZLIB_NONE; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } } /* Check if driver supports adler32 checksum and test */ if ((capab->comp_feature_flags & RTE_COMP_FF_ADLER32_CHECKSUM)) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_ADLER32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_ADLER32; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Generate zlib checksum and test against selected * drivers decompression checksum */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Generate compression and decompression * checksum of selected driver */ test_data.zlib_dir = ZLIB_NONE; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } } /* Check if driver supports combined crc and adler checksum and test */ if ((capab->comp_feature_flags & RTE_COMP_FF_CRC32_ADLER32_CHECKSUM)) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_CRC32_ADLER32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_CRC32_ADLER32; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Generate compression and decompression * checksum of selected driver */ test_data.zlib_dir = ZLIB_NONE; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } } ret = TEST_SUCCESS; exit: rte_free(compress_xform); rte_free(decompress_xform); return ret; } static int test_compressdev_out_of_space_buffer(void) { struct comp_testsuite_params *ts_params = &testsuite_params; int ret; uint16_t i; const struct rte_compressdev_capabilities *capab; RTE_LOG(INFO, USER1, "This is a negative test, errors are expected\n"); capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_HUFFMAN_FIXED) == 0) return -ENOTSUP; struct interim_data_params int_data = { &compress_test_bufs[0], 1, &i, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 1, /* run out-of-space test */ .big_data = 0, .overflow = OVERFLOW_DISABLED }; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) { /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; test_data.buff_type = SGL_BOTH; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; test_data.buff_type = SGL_BOTH; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; } ret = TEST_SUCCESS; exit: return ret; } static int test_compressdev_deflate_stateless_dynamic_big(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i = 0; int ret; int j; const struct rte_compressdev_capabilities *capab; char *test_buffer = NULL; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_HUFFMAN_DYNAMIC) == 0) return -ENOTSUP; if ((capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) == 0) return -ENOTSUP; test_buffer = rte_malloc(NULL, BIG_DATA_TEST_SIZE, 0); if (test_buffer == NULL) { RTE_LOG(ERR, USER1, "Can't allocate buffer for big-data\n"); return TEST_FAILED; } struct interim_data_params int_data = { (const char * const *)&test_buffer, 1, &i, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = SGL_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 1, .overflow = OVERFLOW_DISABLED }; ts_params->def_comp_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_DYNAMIC; /* fill the buffer with data based on rand. data */ srand(BIG_DATA_TEST_SIZE); for (j = 0; j < BIG_DATA_TEST_SIZE - 1; ++j) test_buffer[j] = (uint8_t)(rand() % ((uint8_t)-1)) | 1; test_buffer[BIG_DATA_TEST_SIZE-1] = 0; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; ret = test_deflate_comp_decomp(&int_data, &test_data); if (ret < 0) goto exit; ret = TEST_SUCCESS; exit: ts_params->def_comp_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_DEFAULT; rte_free(test_buffer); return ret; } static int test_compressdev_deflate_stateful_decomp(void) { struct comp_testsuite_params *ts_params = &testsuite_params; int ret; uint16_t i; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if (!(capab->comp_feature_flags & RTE_COMP_FF_STATEFUL_DECOMPRESSION)) return -ENOTSUP; struct interim_data_params int_data = { &compress_test_bufs[0], 1, &i, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATEFUL, .buff_type = LB_BOTH, .zlib_dir = ZLIB_COMPRESS, .out_of_space = 0, .big_data = 0, .decompress_output_block_size = 2000, .decompress_steps_max = 4, .overflow = OVERFLOW_DISABLED }; /* Compress with Zlib, decompress with compressdev */ if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) { /* Now test with SGL buffers */ test_data.buff_type = SGL_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } } ret = TEST_SUCCESS; exit: return ret; } static int test_compressdev_deflate_stateful_decomp_checksum(void) { struct comp_testsuite_params *ts_params = &testsuite_params; int ret; uint16_t i; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if (!(capab->comp_feature_flags & RTE_COMP_FF_STATEFUL_DECOMPRESSION)) return -ENOTSUP; /* Check if driver supports any checksum */ if (!(capab->comp_feature_flags & (RTE_COMP_FF_CRC32_CHECKSUM | RTE_COMP_FF_ADLER32_CHECKSUM | RTE_COMP_FF_CRC32_ADLER32_CHECKSUM))) return -ENOTSUP; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); return TEST_FAILED; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); struct rte_comp_xform *decompress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (decompress_xform == NULL) { RTE_LOG(ERR, USER1, "Decompress xform could not be created\n"); rte_free(compress_xform); return TEST_FAILED; } memcpy(decompress_xform, ts_params->def_decomp_xform, sizeof(struct rte_comp_xform)); struct interim_data_params int_data = { &compress_test_bufs[0], 1, &i, &compress_xform, &decompress_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATEFUL, .buff_type = LB_BOTH, .zlib_dir = ZLIB_COMPRESS, .out_of_space = 0, .big_data = 0, .decompress_output_block_size = 2000, .decompress_steps_max = 4, .overflow = OVERFLOW_DISABLED }; /* Check if driver supports crc32 checksum and test */ if (capab->comp_feature_flags & RTE_COMP_FF_CRC32_CHECKSUM) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_CRC32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_CRC32; /* Compress with Zlib, decompress with compressdev */ test_data.buff_type = LB_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) { /* Now test with SGL buffers */ test_data.buff_type = SGL_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } } } /* Check if driver supports adler32 checksum and test */ if (capab->comp_feature_flags & RTE_COMP_FF_ADLER32_CHECKSUM) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_ADLER32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_ADLER32; /* Compress with Zlib, decompress with compressdev */ test_data.buff_type = LB_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) { /* Now test with SGL buffers */ test_data.buff_type = SGL_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } } } /* Check if driver supports combined crc and adler checksum and test */ if (capab->comp_feature_flags & RTE_COMP_FF_CRC32_ADLER32_CHECKSUM) { compress_xform->compress.chksum = RTE_COMP_CHECKSUM_CRC32_ADLER32; decompress_xform->decompress.chksum = RTE_COMP_CHECKSUM_CRC32_ADLER32; /* Zlib doesn't support combined checksum */ test_data.zlib_dir = ZLIB_NONE; /* Compress stateless, decompress stateful with compressdev */ test_data.buff_type = LB_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } if (capab->comp_feature_flags & RTE_COMP_FF_OOP_SGL_IN_SGL_OUT) { /* Now test with SGL buffers */ test_data.buff_type = SGL_BOTH; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) { ret = TEST_FAILED; goto exit; } } } ret = TEST_SUCCESS; exit: rte_free(compress_xform); rte_free(decompress_xform); return ret; } static const struct rte_memzone * make_memzone(const char *name, size_t size) { unsigned int socket_id = rte_socket_id(); char mz_name[RTE_MEMZONE_NAMESIZE]; const struct rte_memzone *memzone; snprintf(mz_name, RTE_MEMZONE_NAMESIZE, "%s_%u", name, socket_id); memzone = rte_memzone_lookup(mz_name); if (memzone != NULL && memzone->len != size) { rte_memzone_free(memzone); memzone = NULL; } if (memzone == NULL) { memzone = rte_memzone_reserve_aligned(mz_name, size, socket_id, RTE_MEMZONE_IOVA_CONTIG, RTE_CACHE_LINE_SIZE); if (memzone == NULL) RTE_LOG(ERR, USER1, "Can't allocate memory zone %s", mz_name); } return memzone; } static int test_compressdev_external_mbufs(void) { struct comp_testsuite_params *ts_params = &testsuite_params; size_t data_len = 0; uint16_t i; int ret = TEST_FAILED; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) data_len = RTE_MAX(data_len, strlen(compress_test_bufs[i]) + 1); struct interim_data_params int_data = { NULL, 1, NULL, &ts_params->def_comp_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .use_external_mbufs = 1, .inbuf_data_size = data_len, .inbuf_memzone = make_memzone("inbuf", data_len), .compbuf_memzone = make_memzone("compbuf", data_len * COMPRESS_BUF_SIZE_RATIO), .uncompbuf_memzone = make_memzone("decompbuf", data_len), .overflow = OVERFLOW_DISABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { /* prepare input data */ data_len = strlen(compress_test_bufs[i]) + 1; rte_memcpy(test_data.inbuf_memzone->addr, compress_test_bufs[i], data_len); test_data.inbuf_data_size = data_len; int_data.buf_idx = &i; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) goto exit; /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; if (test_deflate_comp_decomp(&int_data, &test_data) < 0) goto exit; } ret = TEST_SUCCESS; exit: rte_memzone_free(test_data.inbuf_memzone); rte_memzone_free(test_data.compbuf_memzone); rte_memzone_free(test_data.uncompbuf_memzone); return ret; } static int test_compressdev_deflate_stateless_fixed_oos_recoverable(void) { struct comp_testsuite_params *ts_params = &testsuite_params; uint16_t i; int ret; int comp_result; const struct rte_compressdev_capabilities *capab; capab = rte_compressdev_capability_get(0, RTE_COMP_ALGO_DEFLATE); TEST_ASSERT(capab != NULL, "Failed to retrieve device capabilities"); if ((capab->comp_feature_flags & RTE_COMP_FF_HUFFMAN_FIXED) == 0) return -ENOTSUP; struct rte_comp_xform *compress_xform = rte_malloc(NULL, sizeof(struct rte_comp_xform), 0); if (compress_xform == NULL) { RTE_LOG(ERR, USER1, "Compress xform could not be created\n"); ret = TEST_FAILED; goto exit; } memcpy(compress_xform, ts_params->def_comp_xform, sizeof(struct rte_comp_xform)); compress_xform->compress.deflate.huffman = RTE_COMP_HUFFMAN_FIXED; struct interim_data_params int_data = { NULL, 1, NULL, &compress_xform, &ts_params->def_decomp_xform, 1 }; struct test_data_params test_data = { .compress_state = RTE_COMP_OP_STATELESS, .decompress_state = RTE_COMP_OP_STATELESS, .buff_type = LB_BOTH, .zlib_dir = ZLIB_DECOMPRESS, .out_of_space = 0, .big_data = 0, .overflow = OVERFLOW_ENABLED }; for (i = 0; i < RTE_DIM(compress_test_bufs); i++) { int_data.test_bufs = &compress_test_bufs[i]; int_data.buf_idx = &i; /* Compress with compressdev, decompress with Zlib */ test_data.zlib_dir = ZLIB_DECOMPRESS; comp_result = test_deflate_comp_decomp(&int_data, &test_data); if (comp_result < 0) { ret = TEST_FAILED; goto exit; } else if (comp_result > 0) { ret = -ENOTSUP; goto exit; } /* Compress with Zlib, decompress with compressdev */ test_data.zlib_dir = ZLIB_COMPRESS; comp_result = test_deflate_comp_decomp(&int_data, &test_data); if (comp_result < 0) { ret = TEST_FAILED; goto exit; } else if (comp_result > 0) { ret = -ENOTSUP; goto exit; } } ret = TEST_SUCCESS; exit: rte_free(compress_xform); return ret; } static struct unit_test_suite compressdev_testsuite = { .suite_name = "compressdev unit test suite", .setup = testsuite_setup, .teardown = testsuite_teardown, .unit_test_cases = { TEST_CASE_ST(NULL, NULL, test_compressdev_invalid_configuration), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_fixed), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_dynamic), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_dynamic_big), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_multi_op), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_multi_level), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_multi_xform), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_sgl), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_checksum), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_out_of_space_buffer), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateful_decomp), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateful_decomp_checksum), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_external_mbufs), TEST_CASE_ST(generic_ut_setup, generic_ut_teardown, test_compressdev_deflate_stateless_fixed_oos_recoverable), TEST_CASES_END() /**< NULL terminate unit test array */ } }; static int test_compressdev(void) { return unit_test_suite_runner(&compressdev_testsuite); } REGISTER_TEST_COMMAND(compressdev_autotest, test_compressdev);