This patch removes references to deprecated PMEM from accel library. The code that was executed when ACCEL_FLAG_PERSISTENT flag is set, is no longer needed and is removed. _sw_accel_copy() function is removed and replaced with memcpy(), as after PMEM removal its functionality is the same as memcpy(). _sw_accel_dualcast() is no longer needed, replaced with direct calls to memcpy() Removed 'flags' parameter - it is no longer needed accel_ut.c: removed references to PMDK deprecation.md updated ACCEL_FLAG_PERSISTENT flag will be removed in next patch. Change-Id: I86130466fe7a5f6ee547df1517b803035ff41a7a Signed-off-by: Marcin Spiewak <marcin.spiewak@intel.com> Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/16899 Reviewed-by: Tomasz Zawadzki <tomasz.zawadzki@intel.com> Reviewed-by: Aleksey Marchuk <alexeymar@nvidia.com> Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Konrad Sztyber <konrad.sztyber@intel.com> Reviewed-by: Ben Walker <benjamin.walker@intel.com>
3684 lines
114 KiB
C
3684 lines
114 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (C) 2021 Intel Corporation.
|
|
* All rights reserved.
|
|
* Copyright (c) 2022 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
*/
|
|
|
|
#include "spdk_cunit.h"
|
|
#include "spdk_internal/mock.h"
|
|
#include "spdk_internal/accel_module.h"
|
|
#include "thread/thread_internal.h"
|
|
#include "common/lib/ut_multithread.c"
|
|
#include "accel/accel.c"
|
|
#include "accel/accel_sw.c"
|
|
#include "unit/lib/json_mock.c"
|
|
|
|
DEFINE_STUB_V(spdk_memory_domain_destroy, (struct spdk_memory_domain *domain));
|
|
DEFINE_STUB(spdk_memory_domain_get_dma_device_id, const char *,
|
|
(struct spdk_memory_domain *domain), "UT_DMA");
|
|
|
|
int
|
|
spdk_memory_domain_create(struct spdk_memory_domain **domain, enum spdk_dma_device_type type,
|
|
struct spdk_memory_domain_ctx *ctx, const char *id)
|
|
{
|
|
*domain = (void *)0xdeadbeef;
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct ut_domain_ctx {
|
|
struct iovec iov;
|
|
struct iovec expected;
|
|
int pull_submit_status;
|
|
int push_submit_status;
|
|
int pull_complete_status;
|
|
int push_complete_status;
|
|
};
|
|
|
|
static struct spdk_memory_domain *g_ut_domain = (void *)0xa55e1;
|
|
|
|
int
|
|
spdk_memory_domain_pull_data(struct spdk_memory_domain *sd, void *sctx, struct iovec *siov,
|
|
uint32_t siovcnt, struct iovec *diov, uint32_t diovcnt,
|
|
spdk_memory_domain_data_cpl_cb cpl_cb, void *cpl_arg)
|
|
{
|
|
struct ut_domain_ctx *ctx = sctx;
|
|
|
|
CU_ASSERT_EQUAL(sd, g_ut_domain);
|
|
CU_ASSERT_EQUAL(siovcnt, 1);
|
|
CU_ASSERT_EQUAL(memcmp(siov, &ctx->expected, sizeof(*siov)), 0);
|
|
|
|
if (ctx->pull_submit_status != 0) {
|
|
return ctx->pull_submit_status;
|
|
}
|
|
|
|
if (ctx->pull_complete_status != 0) {
|
|
cpl_cb(cpl_arg, ctx->pull_complete_status);
|
|
return 0;
|
|
}
|
|
|
|
spdk_iovcpy(&ctx->iov, 1, diov, diovcnt);
|
|
cpl_cb(cpl_arg, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
spdk_memory_domain_push_data(struct spdk_memory_domain *dd, void *dctx, struct iovec *diov,
|
|
uint32_t diovcnt, struct iovec *siov, uint32_t siovcnt,
|
|
spdk_memory_domain_data_cpl_cb cpl_cb, void *cpl_arg)
|
|
{
|
|
struct ut_domain_ctx *ctx = dctx;
|
|
|
|
CU_ASSERT_EQUAL(dd, g_ut_domain);
|
|
CU_ASSERT_EQUAL(diovcnt, 1);
|
|
CU_ASSERT_EQUAL(memcmp(diov, &ctx->expected, sizeof(*diov)), 0);
|
|
|
|
if (ctx->push_submit_status != 0) {
|
|
return ctx->push_submit_status;
|
|
}
|
|
|
|
if (ctx->push_complete_status != 0) {
|
|
cpl_cb(cpl_arg, ctx->push_complete_status);
|
|
return 0;
|
|
}
|
|
|
|
spdk_iovcpy(siov, siovcnt, &ctx->iov, 1);
|
|
cpl_cb(cpl_arg, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* global vars and setup/cleanup functions used for all test functions */
|
|
struct spdk_accel_module_if g_module_if = {};
|
|
struct accel_module g_module = { .module = &g_module_if };
|
|
struct spdk_io_channel *g_ch = NULL;
|
|
struct accel_io_channel *g_accel_ch = NULL;
|
|
struct sw_accel_io_channel *g_sw_ch = NULL;
|
|
struct spdk_io_channel *g_module_ch = NULL;
|
|
|
|
static uint64_t g_opc_mask = 0;
|
|
|
|
static uint64_t
|
|
_accel_op_to_bit(enum accel_opcode opc)
|
|
{
|
|
return (1 << opc);
|
|
}
|
|
|
|
static bool
|
|
_supports_opcode(enum accel_opcode opc)
|
|
{
|
|
if (_accel_op_to_bit(opc) & g_opc_mask) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int
|
|
test_setup(void)
|
|
{
|
|
int i;
|
|
|
|
g_ch = calloc(1, sizeof(struct spdk_io_channel) + sizeof(struct accel_io_channel));
|
|
if (g_ch == NULL) {
|
|
/* for some reason the assert fatal macro doesn't work in the setup function. */
|
|
CU_ASSERT(false);
|
|
return -1;
|
|
}
|
|
g_accel_ch = (struct accel_io_channel *)((char *)g_ch + sizeof(struct spdk_io_channel));
|
|
g_module_ch = calloc(1, sizeof(struct spdk_io_channel) + sizeof(struct sw_accel_io_channel));
|
|
if (g_module_ch == NULL) {
|
|
CU_ASSERT(false);
|
|
return -1;
|
|
}
|
|
|
|
g_module_if.submit_tasks = sw_accel_submit_tasks;
|
|
g_module_if.name = "software";
|
|
for (i = 0; i < ACCEL_OPC_LAST; i++) {
|
|
g_accel_ch->module_ch[i] = g_module_ch;
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
g_sw_ch = (struct sw_accel_io_channel *)((char *)g_module_ch + sizeof(
|
|
struct spdk_io_channel));
|
|
TAILQ_INIT(&g_sw_ch->tasks_to_complete);
|
|
g_module_if.supports_opcode = _supports_opcode;
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
test_cleanup(void)
|
|
{
|
|
free(g_ch);
|
|
free(g_module_ch);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define DUMMY_ARG 0xDEADBEEF
|
|
static bool g_dummy_cb_called = false;
|
|
static void
|
|
dummy_cb_fn(void *cb_arg, int status)
|
|
{
|
|
CU_ASSERT(*(uint32_t *)cb_arg == DUMMY_ARG);
|
|
CU_ASSERT(status == 0);
|
|
g_dummy_cb_called = true;
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_task_complete(void)
|
|
{
|
|
struct spdk_accel_task accel_task = {};
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
uint32_t cb_arg = DUMMY_ARG;
|
|
int status = 0;
|
|
|
|
accel_task.accel_ch = g_accel_ch;
|
|
accel_task.cb_fn = dummy_cb_fn;
|
|
accel_task.cb_arg = &cb_arg;
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Confirm cb is called and task added to list. */
|
|
spdk_accel_task_complete(&accel_task, status);
|
|
CU_ASSERT(g_dummy_cb_called == true);
|
|
expected_accel_task = TAILQ_FIRST(&g_accel_ch->task_pool);
|
|
TAILQ_REMOVE(&g_accel_ch->task_pool, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &accel_task);
|
|
}
|
|
|
|
static void
|
|
test_get_task(void)
|
|
{
|
|
struct spdk_accel_task *task;
|
|
struct spdk_accel_task _task;
|
|
void *cb_arg = NULL;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* no tasks left, return NULL. */
|
|
task = _get_task(g_accel_ch, dummy_cb_fn, cb_arg);
|
|
CU_ASSERT(task == NULL);
|
|
|
|
_task.cb_fn = dummy_cb_fn;
|
|
_task.cb_arg = cb_arg;
|
|
_task.accel_ch = g_accel_ch;
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &_task, link);
|
|
|
|
/* Get a valid task. */
|
|
task = _get_task(g_accel_ch, dummy_cb_fn, cb_arg);
|
|
CU_ASSERT(task == &_task);
|
|
CU_ASSERT(_task.cb_fn == dummy_cb_fn);
|
|
CU_ASSERT(_task.cb_arg == cb_arg);
|
|
CU_ASSERT(_task.accel_ch == g_accel_ch);
|
|
}
|
|
|
|
#define TEST_SUBMIT_SIZE 64
|
|
static void
|
|
test_spdk_accel_submit_copy(void)
|
|
{
|
|
const uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
uint8_t dst[TEST_SUBMIT_SIZE] = {0};
|
|
uint8_t src[TEST_SUBMIT_SIZE] = {0};
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
int flags = 0;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_copy(g_ch, src, dst, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
task.accel_ch = g_accel_ch;
|
|
task.flags = 1;
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* submission OK. */
|
|
rc = spdk_accel_submit_copy(g_ch, dst, src, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_COPY);
|
|
CU_ASSERT(task.flags == 0);
|
|
CU_ASSERT(memcmp(dst, src, TEST_SUBMIT_SIZE) == 0);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_dualcast(void)
|
|
{
|
|
void *dst1;
|
|
void *dst2;
|
|
void *src;
|
|
uint32_t align = ALIGN_4K;
|
|
uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
int flags = 0;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Dualcast requires 4K alignment on dst addresses,
|
|
* hence using the hard coded address to test the buffer alignment
|
|
*/
|
|
dst1 = (void *)0x5000;
|
|
dst2 = (void *)0x60f0;
|
|
src = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(src != NULL);
|
|
memset(src, 0x5A, TEST_SUBMIT_SIZE);
|
|
|
|
/* This should fail since dst2 is not 4k aligned */
|
|
rc = spdk_accel_submit_dualcast(g_ch, dst1, dst2, src, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == -EINVAL);
|
|
|
|
dst1 = (void *)0x7010;
|
|
dst2 = (void *)0x6000;
|
|
/* This should fail since dst1 is not 4k aligned */
|
|
rc = spdk_accel_submit_dualcast(g_ch, dst1, dst2, src, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == -EINVAL);
|
|
|
|
/* Dualcast requires 4K alignment on dst addresses */
|
|
dst1 = (void *)0x7000;
|
|
dst2 = (void *)0x6000;
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_dualcast(g_ch, dst1, dst2, src, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK., since we test the SW path , need to use valid memory addresses
|
|
* cannot hardcode them anymore */
|
|
dst1 = spdk_dma_zmalloc(nbytes, align, NULL);
|
|
SPDK_CU_ASSERT_FATAL(dst1 != NULL);
|
|
dst2 = spdk_dma_zmalloc(nbytes, align, NULL);
|
|
SPDK_CU_ASSERT_FATAL(dst2 != NULL);
|
|
/* SW module does the dualcast. */
|
|
rc = spdk_accel_submit_dualcast(g_ch, dst1, dst2, src, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_DUALCAST);
|
|
CU_ASSERT(task.flags == 0);
|
|
CU_ASSERT(memcmp(dst1, src, TEST_SUBMIT_SIZE) == 0);
|
|
CU_ASSERT(memcmp(dst2, src, TEST_SUBMIT_SIZE) == 0);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
|
|
free(src);
|
|
spdk_free(dst1);
|
|
spdk_free(dst2);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_compare(void)
|
|
{
|
|
void *src1;
|
|
void *src2;
|
|
uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
src1 = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(src1 != NULL);
|
|
src2 = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(src2 != NULL);
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_compare(g_ch, src1, src2, nbytes, NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK. */
|
|
rc = spdk_accel_submit_compare(g_ch, src1, src2, nbytes, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_COMPARE);
|
|
CU_ASSERT(memcmp(src1, src2, TEST_SUBMIT_SIZE) == 0);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
|
|
free(src1);
|
|
free(src2);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_fill(void)
|
|
{
|
|
void *dst;
|
|
void *src;
|
|
uint8_t fill = 0xf;
|
|
uint64_t fill64;
|
|
uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
int flags = 0;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
dst = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(dst != NULL);
|
|
src = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(src != NULL);
|
|
memset(src, fill, TEST_SUBMIT_SIZE);
|
|
memset(&fill64, fill, sizeof(uint64_t));
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_fill(g_ch, dst, fill, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK. */
|
|
rc = spdk_accel_submit_fill(g_ch, dst, fill, nbytes, flags, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.fill_pattern == fill64);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_FILL);
|
|
CU_ASSERT(task.flags == 0);
|
|
|
|
CU_ASSERT(memcmp(dst, src, TEST_SUBMIT_SIZE) == 0);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
|
|
free(dst);
|
|
free(src);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_crc32c(void)
|
|
{
|
|
const uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
uint32_t crc_dst;
|
|
uint8_t src[TEST_SUBMIT_SIZE];
|
|
uint32_t seed = 1;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_crc32c(g_ch, &crc_dst, src, seed, nbytes, NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK. */
|
|
rc = spdk_accel_submit_crc32c(g_ch, &crc_dst, src, seed, nbytes, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.crc_dst == &crc_dst);
|
|
CU_ASSERT(task.seed == seed);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_CRC32C);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_crc32cv(void)
|
|
{
|
|
uint32_t crc_dst;
|
|
uint32_t seed = 0;
|
|
uint32_t iov_cnt = 32;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
uint32_t i = 0;
|
|
struct spdk_accel_task task;
|
|
struct iovec iov[32];
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
for (i = 0; i < iov_cnt; i++) {
|
|
iov[i].iov_base = calloc(1, TEST_SUBMIT_SIZE);
|
|
SPDK_CU_ASSERT_FATAL(iov[i].iov_base != NULL);
|
|
iov[i].iov_len = TEST_SUBMIT_SIZE;
|
|
}
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK. */
|
|
rc = spdk_accel_submit_crc32cv(g_ch, &crc_dst, iov, iov_cnt, seed, NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.s.iovs == iov);
|
|
CU_ASSERT(task.s.iovcnt == iov_cnt);
|
|
CU_ASSERT(task.crc_dst == &crc_dst);
|
|
CU_ASSERT(task.seed == seed);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_CRC32C);
|
|
CU_ASSERT(task.cb_arg == cb_arg);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
|
|
for (i = 0; i < iov_cnt; i++) {
|
|
free(iov[i].iov_base);
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_copy_crc32c(void)
|
|
{
|
|
const uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
uint32_t crc_dst;
|
|
uint8_t dst[TEST_SUBMIT_SIZE];
|
|
uint8_t src[TEST_SUBMIT_SIZE];
|
|
uint32_t seed = 0;
|
|
void *cb_arg = NULL;
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
int flags = 0;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_copy_crc32c(g_ch, dst, src, &crc_dst, seed, nbytes, flags,
|
|
NULL, cb_arg);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* accel submission OK. */
|
|
rc = spdk_accel_submit_copy_crc32c(g_ch, dst, src, &crc_dst, seed, nbytes, flags,
|
|
NULL, cb_arg);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.crc_dst == &crc_dst);
|
|
CU_ASSERT(task.seed == seed);
|
|
CU_ASSERT(task.flags == 0);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_COPY_CRC32C);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_submit_xor(void)
|
|
{
|
|
const uint64_t nbytes = TEST_SUBMIT_SIZE;
|
|
uint8_t dst[TEST_SUBMIT_SIZE] = {0};
|
|
uint8_t src1[TEST_SUBMIT_SIZE] = {0};
|
|
uint8_t src2[TEST_SUBMIT_SIZE] = {0};
|
|
void *sources[] = { src1, src2 };
|
|
uint32_t nsrcs = SPDK_COUNTOF(sources);
|
|
int rc;
|
|
struct spdk_accel_task task;
|
|
struct spdk_accel_task *expected_accel_task = NULL;
|
|
|
|
TAILQ_INIT(&g_accel_ch->task_pool);
|
|
|
|
/* Fail with no tasks on _get_task() */
|
|
rc = spdk_accel_submit_xor(g_ch, dst, sources, nsrcs, nbytes, NULL, NULL);
|
|
CU_ASSERT(rc == -ENOMEM);
|
|
|
|
TAILQ_INSERT_TAIL(&g_accel_ch->task_pool, &task, link);
|
|
|
|
/* submission OK. */
|
|
rc = spdk_accel_submit_xor(g_ch, dst, sources, nsrcs, nbytes, NULL, NULL);
|
|
CU_ASSERT(rc == 0);
|
|
CU_ASSERT(task.nsrcs.srcs == sources);
|
|
CU_ASSERT(task.nsrcs.cnt == nsrcs);
|
|
CU_ASSERT(task.d.iovcnt == 1);
|
|
CU_ASSERT(task.d.iovs[0].iov_base == dst);
|
|
CU_ASSERT(task.d.iovs[0].iov_len == nbytes);
|
|
CU_ASSERT(task.op_code == ACCEL_OPC_XOR);
|
|
expected_accel_task = TAILQ_FIRST(&g_sw_ch->tasks_to_complete);
|
|
TAILQ_REMOVE(&g_sw_ch->tasks_to_complete, expected_accel_task, link);
|
|
CU_ASSERT(expected_accel_task == &task);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_module_find_by_name(void)
|
|
{
|
|
struct spdk_accel_module_if mod1 = {};
|
|
struct spdk_accel_module_if mod2 = {};
|
|
struct spdk_accel_module_if mod3 = {};
|
|
struct spdk_accel_module_if *accel_module = NULL;
|
|
|
|
mod1.name = "ioat";
|
|
mod2.name = "idxd";
|
|
mod3.name = "software";
|
|
|
|
TAILQ_INIT(&spdk_accel_module_list);
|
|
TAILQ_INSERT_TAIL(&spdk_accel_module_list, &mod1, tailq);
|
|
TAILQ_INSERT_TAIL(&spdk_accel_module_list, &mod2, tailq);
|
|
TAILQ_INSERT_TAIL(&spdk_accel_module_list, &mod3, tailq);
|
|
|
|
/* Now let's find a valid engine */
|
|
accel_module = _module_find_by_name("ioat");
|
|
CU_ASSERT(accel_module != NULL);
|
|
|
|
/* Try to find one that doesn't exist */
|
|
accel_module = _module_find_by_name("XXX");
|
|
CU_ASSERT(accel_module == NULL);
|
|
}
|
|
|
|
static void
|
|
test_spdk_accel_module_register(void)
|
|
{
|
|
struct spdk_accel_module_if mod1 = {};
|
|
struct spdk_accel_module_if mod2 = {};
|
|
struct spdk_accel_module_if mod3 = {};
|
|
struct spdk_accel_module_if mod4 = {};
|
|
struct spdk_accel_module_if *accel_module = NULL;
|
|
int i = 0;
|
|
|
|
mod1.name = "ioat";
|
|
mod2.name = "idxd";
|
|
mod3.name = "software";
|
|
mod4.name = "nothing";
|
|
|
|
TAILQ_INIT(&spdk_accel_module_list);
|
|
|
|
spdk_accel_module_list_add(&mod1);
|
|
spdk_accel_module_list_add(&mod2);
|
|
spdk_accel_module_list_add(&mod3);
|
|
spdk_accel_module_list_add(&mod4);
|
|
|
|
/* Now confirm they're in the right order. */
|
|
TAILQ_FOREACH(accel_module, &spdk_accel_module_list, tailq) {
|
|
switch (i++) {
|
|
case 0:
|
|
CU_ASSERT(strcmp(accel_module->name, "software") == 0);
|
|
break;
|
|
case 1:
|
|
CU_ASSERT(strcmp(accel_module->name, "ioat") == 0);
|
|
break;
|
|
case 2:
|
|
CU_ASSERT(strcmp(accel_module->name, "idxd") == 0);
|
|
break;
|
|
case 3:
|
|
CU_ASSERT(strcmp(accel_module->name, "nothing") == 0);
|
|
break;
|
|
default:
|
|
CU_ASSERT(false);
|
|
break;
|
|
}
|
|
}
|
|
CU_ASSERT(i == 4);
|
|
}
|
|
|
|
struct ut_sequence {
|
|
bool complete;
|
|
int status;
|
|
};
|
|
|
|
static void
|
|
ut_sequence_step_cb(void *cb_arg)
|
|
{
|
|
int *completed = cb_arg;
|
|
|
|
(*completed)++;
|
|
}
|
|
|
|
static void
|
|
ut_sequence_complete_cb(void *cb_arg, int status)
|
|
{
|
|
struct ut_sequence *seq = cb_arg;
|
|
|
|
seq->complete = true;
|
|
seq->status = status;
|
|
}
|
|
|
|
static void
|
|
test_sequence_fill_copy(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct ut_sequence ut_seq;
|
|
char buf[4096], tmp[2][4096], expected[4096];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
int rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* First check the simplest case - single task in a sequence */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
completed = 0;
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, sizeof(buf), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_EQUAL(completed, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check a single copy operation */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0xa5, sizeof(tmp[0]));
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check multiple fill operations */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0xfe, 4096);
|
|
memset(expected, 0xde, 2048);
|
|
memset(expected, 0xa5, 1024);
|
|
seq = NULL;
|
|
completed = 0;
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 4096, NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 2048, NULL, NULL, 0xde, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 1024, NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check multiple copy operations */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[0], sizeof(tmp[0]), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check that adding a copy operation at the end will change destination buffer */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0, sizeof(tmp[0]));
|
|
memset(expected, 0xa5, sizeof(buf));
|
|
seq = NULL;
|
|
completed = 0;
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[0], sizeof(tmp[0]), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check that it's also possible to add copy operation at the beginning */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0xde, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], sizeof(tmp[1]), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static void
|
|
test_sequence_abort(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
char buf[4096], tmp[2][4096], expected[4096];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
int rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Check that aborting a sequence calls operation's callback, the operation is not executed
|
|
* and the sequence is freed
|
|
*/
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0, sizeof(buf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, sizeof(buf), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_abort(seq);
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check sequence with multiple operations */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0, sizeof(buf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 4096, NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 2048, NULL, NULL, 0xde, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_abort(seq);
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* This should be a no-op */
|
|
spdk_accel_sequence_abort(NULL);
|
|
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static void
|
|
test_sequence_append_error(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct accel_io_channel *accel_ch;
|
|
struct iovec src_iovs, dst_iovs;
|
|
char buf[4096];
|
|
TAILQ_HEAD(, spdk_accel_task) tasks = TAILQ_HEAD_INITIALIZER(tasks);
|
|
TAILQ_HEAD(, spdk_accel_sequence) seqs = TAILQ_HEAD_INITIALIZER(seqs);
|
|
int rc;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
accel_ch = spdk_io_channel_get_ctx(ioch);
|
|
|
|
/* Check that append fails and no sequence object is allocated when there are no more free
|
|
* tasks */
|
|
TAILQ_SWAP(&tasks, &accel_ch->task_pool, spdk_accel_task, link);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, sizeof(buf), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = 2048;
|
|
src_iovs.iov_base = &buf[2048];
|
|
src_iovs.iov_len = 2048;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0, ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = 2048;
|
|
src_iovs.iov_base = &buf[2048];
|
|
src_iovs.iov_len = 2048;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0, ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
/* Check that the same happens when the sequence queue is empty */
|
|
TAILQ_SWAP(&tasks, &accel_ch->task_pool, spdk_accel_task, link);
|
|
TAILQ_SWAP(&seqs, &accel_ch->seq_pool, spdk_accel_sequence, link);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, sizeof(buf), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = 2048;
|
|
src_iovs.iov_base = &buf[2048];
|
|
src_iovs.iov_len = 2048;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0, ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = 2048;
|
|
src_iovs.iov_base = &buf[2048];
|
|
src_iovs.iov_len = 2048;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0, ut_sequence_step_cb, NULL);
|
|
CU_ASSERT_EQUAL(rc, -ENOMEM);
|
|
CU_ASSERT_PTR_NULL(seq);
|
|
|
|
TAILQ_SWAP(&tasks, &accel_ch->task_pool, spdk_accel_task, link);
|
|
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
struct ut_sequence_operation {
|
|
int complete_status;
|
|
int submit_status;
|
|
int count;
|
|
struct iovec *src_iovs;
|
|
uint32_t src_iovcnt;
|
|
struct iovec *dst_iovs;
|
|
uint32_t dst_iovcnt;
|
|
int (*submit)(struct spdk_io_channel *ch, struct spdk_accel_task *t);
|
|
};
|
|
|
|
static struct ut_sequence_operation g_seq_operations[ACCEL_OPC_LAST];
|
|
|
|
static int
|
|
ut_sequnce_submit_tasks(struct spdk_io_channel *ch, struct spdk_accel_task *task)
|
|
{
|
|
struct ut_sequence_operation *op = &g_seq_operations[task->op_code];
|
|
|
|
op->count++;
|
|
if (op->submit != NULL) {
|
|
return op->submit(ch, task);
|
|
}
|
|
if (op->src_iovs != NULL) {
|
|
CU_ASSERT_EQUAL(task->s.iovcnt, op->src_iovcnt);
|
|
CU_ASSERT_EQUAL(memcmp(task->s.iovs, op->src_iovs,
|
|
sizeof(struct iovec) * op->src_iovcnt), 0);
|
|
}
|
|
if (op->dst_iovs != NULL) {
|
|
CU_ASSERT_EQUAL(task->d.iovcnt, op->dst_iovcnt);
|
|
CU_ASSERT_EQUAL(memcmp(task->d.iovs, op->dst_iovs,
|
|
sizeof(struct iovec) * op->dst_iovcnt), 0);
|
|
}
|
|
|
|
if (op->submit_status != 0) {
|
|
return op->submit_status;
|
|
}
|
|
|
|
spdk_accel_task_complete(task, op->complete_status);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
ut_clear_operations(void)
|
|
{
|
|
memset(&g_seq_operations, 0, sizeof(g_seq_operations));
|
|
}
|
|
|
|
static void
|
|
test_sequence_completion_error(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct ut_sequence ut_seq;
|
|
struct iovec src_iovs, dst_iovs;
|
|
char buf[4096], tmp[4096];
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
int i, rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp, 0, sizeof(tmp));
|
|
|
|
/* Check that if the first operation completes with an error, the whole sequence is
|
|
* completed with that error and that all operations' completion callbacks are executed
|
|
*/
|
|
g_seq_operations[ACCEL_OPC_FILL].complete_status = -E2BIG;
|
|
completed = 0;
|
|
seq = NULL;
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp, sizeof(tmp), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = sizeof(buf);
|
|
src_iovs.iov_base = tmp;
|
|
src_iovs.iov_len = sizeof(tmp);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -E2BIG);
|
|
|
|
/* Check the same with a second operation in the sequence */
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].complete_status = -EACCES;
|
|
g_seq_operations[ACCEL_OPC_FILL].complete_status = 0;
|
|
completed = 0;
|
|
seq = NULL;
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp, sizeof(tmp), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = sizeof(buf);
|
|
src_iovs.iov_base = tmp;
|
|
src_iovs.iov_len = sizeof(tmp);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EACCES);
|
|
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].complete_status = 0;
|
|
g_seq_operations[ACCEL_OPC_FILL].complete_status = 0;
|
|
|
|
/* Check submission failure of the first operation */
|
|
g_seq_operations[ACCEL_OPC_FILL].submit_status = -EADDRINUSE;
|
|
completed = 0;
|
|
seq = NULL;
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp, sizeof(tmp), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = sizeof(buf);
|
|
src_iovs.iov_base = tmp;
|
|
src_iovs.iov_len = sizeof(tmp);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EADDRINUSE);
|
|
|
|
/* Check the same with a second operation */
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].submit_status = -EADDRNOTAVAIL;
|
|
g_seq_operations[ACCEL_OPC_FILL].submit_status = 0;
|
|
completed = 0;
|
|
seq = NULL;
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp, sizeof(tmp), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs.iov_base = buf;
|
|
dst_iovs.iov_len = sizeof(buf);
|
|
src_iovs.iov_base = tmp;
|
|
src_iovs.iov_len = sizeof(tmp);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs, 1, NULL, NULL,
|
|
&src_iovs, 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EADDRNOTAVAIL);
|
|
|
|
/* Cleanup module pointers to make subsequent tests work correctly */
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
#ifdef SPDK_CONFIG_ISAL
|
|
static void
|
|
ut_compress_cb(void *cb_arg, int status)
|
|
{
|
|
int *completed = cb_arg;
|
|
|
|
CU_ASSERT_EQUAL(status, 0);
|
|
|
|
*completed = 1;
|
|
}
|
|
|
|
static void
|
|
test_sequence_decompress(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct ut_sequence ut_seq;
|
|
char buf[4096], tmp[2][4096], expected[4096];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
uint32_t compressed_size;
|
|
int rc, completed = 0;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
src_iovs[0].iov_base = expected;
|
|
src_iovs[0].iov_len = sizeof(expected);
|
|
rc = spdk_accel_submit_compress(ioch, tmp[0], sizeof(tmp[0]), &src_iovs[0], 1,
|
|
&compressed_size, 0, ut_compress_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
while (!completed) {
|
|
poll_threads();
|
|
}
|
|
|
|
/* Check a single decompress operation in a sequence */
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Put the decompress operation in the middle of a sequence with a copy operation at the
|
|
* beginning and a fill at the end modifying the first 2048B of the buffer.
|
|
*/
|
|
memset(expected, 0xfe, 2048);
|
|
memset(buf, 0, sizeof(buf));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = compressed_size;
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 2048, NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check sequence with decompress at the beginning: decompress -> copy */
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(buf, 0, sizeof(buf));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static void
|
|
test_sequence_reverse(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct ut_sequence ut_seq;
|
|
char buf[4096], tmp[2][4096], expected[4096];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
uint32_t compressed_size;
|
|
int rc, completed = 0;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
src_iovs[0].iov_base = expected;
|
|
src_iovs[0].iov_len = sizeof(expected);
|
|
rc = spdk_accel_submit_compress(ioch, tmp[0], sizeof(tmp[0]), &src_iovs[0], 1,
|
|
&compressed_size, 0, ut_compress_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
while (!completed) {
|
|
poll_threads();
|
|
}
|
|
|
|
/* First check that reversing a sequnce with a single operation is a no-op */
|
|
memset(buf, 0, sizeof(buf));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_reverse(seq);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Add a copy operation at the end with src set to the compressed data. After reverse(),
|
|
* that copy operation should be first, so decompress() should receive compressed data in
|
|
* its src buffer.
|
|
*/
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[1];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[1];
|
|
dst_iovs[1].iov_len = compressed_size;
|
|
src_iovs[1].iov_base = tmp[0];
|
|
src_iovs[1].iov_len = compressed_size;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_reverse(seq);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Check the same, but add an extra fill operation at the beginning that should execute last
|
|
* after reverse().
|
|
*/
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(expected, 0xfe, 2048);
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 2048, NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[1];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[1];
|
|
dst_iovs[1].iov_len = compressed_size;
|
|
src_iovs[1].iov_base = tmp[0];
|
|
src_iovs[1].iov_len = compressed_size;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_reverse(seq);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
/* Build the sequence in order and then reverse it twice */
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = compressed_size;
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = compressed_size;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = compressed_size;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 2048, NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
spdk_accel_sequence_reverse(seq);
|
|
spdk_accel_sequence_reverse(seq);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
test_sequence_copy_elision(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct ut_sequence ut_seq;
|
|
struct iovec src_iovs[4], dst_iovs[4], exp_iovs[2];
|
|
char buf[4096], tmp[4][4096];
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
struct spdk_accel_crypto_key key = {};
|
|
int i, rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_seq_operations[i].complete_status = 0;
|
|
g_seq_operations[i].submit_status = 0;
|
|
g_seq_operations[i].count = 0;
|
|
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
|
|
/* Check that a copy operation at the beginning is removed */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].dst_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].src_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = 2048;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = 2048;
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
|
|
/* Check that a copy operation at the end is removed too */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = 2048;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = 2048;
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = 2048;
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = 2048;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
|
|
/* Check a copy operation both at the beginning and the end */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = 2048;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = 2048;
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = 2048;
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = 2048;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
|
|
/* Check decompress + copy + decompress + copy */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].src_iovs = NULL;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].dst_iovs = NULL;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = 2048;
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = tmp[3];
|
|
dst_iovs[2].iov_len = 1024;
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = 2048;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[3].iov_base = buf;
|
|
dst_iovs[3].iov_len = 1024;
|
|
src_iovs[3].iov_base = tmp[3];
|
|
src_iovs[3].iov_len = 1024;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[3], 1, NULL, NULL,
|
|
&src_iovs[3], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 2);
|
|
|
|
/* Check two copy operations - one of them should be removed, while the other should be
|
|
* executed normally */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].dst_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_COPY].src_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_COPY].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_COPY].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = sizeof(buf);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 1);
|
|
|
|
/* Check fill + copy */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].src_iovs = NULL;
|
|
g_seq_operations[ACCEL_OPC_COPY].dst_iovs = NULL;
|
|
g_seq_operations[ACCEL_OPC_FILL].dst_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_FILL].dst_iovs = &exp_iovs[0];
|
|
exp_iovs[0].iov_base = buf;
|
|
exp_iovs[0].iov_len = sizeof(buf);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[0], sizeof(tmp[0]), NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = buf;
|
|
dst_iovs[0].iov_len = sizeof(buf);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 1);
|
|
|
|
/* Check copy + encrypt + copy */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].dst_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].src_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = sizeof(buf);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = sizeof(tmp[2]);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, sizeof(tmp[2]), 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
|
|
/* Check copy + decrypt + copy */
|
|
seq = NULL;
|
|
completed = 0;
|
|
g_seq_operations[ACCEL_OPC_COPY].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].dst_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].src_iovcnt = 1;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].src_iovs = &exp_iovs[0];
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].dst_iovs = &exp_iovs[1];
|
|
exp_iovs[0].iov_base = tmp[0];
|
|
exp_iovs[0].iov_len = sizeof(tmp[0]);
|
|
exp_iovs[1].iov_base = buf;
|
|
exp_iovs[1].iov_len = sizeof(buf);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = sizeof(tmp[2]);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, sizeof(tmp[2]), 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 1);
|
|
|
|
/* Cleanup module pointers to make subsequent tests work correctly */
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static int
|
|
ut_submit_decompress(struct spdk_io_channel *ch, struct spdk_accel_task *task)
|
|
{
|
|
spdk_iovmove(task->s.iovs, task->s.iovcnt, task->d.iovs, task->d.iovcnt);
|
|
|
|
spdk_accel_task_complete(task, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
test_sequence_accel_buffers(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct accel_io_channel *accel_ch;
|
|
struct ut_sequence ut_seq;
|
|
struct iovec src_iovs[3], dst_iovs[3];
|
|
char srcbuf[4096], dstbuf[4096], expected[4096];
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
void *buf[2], *domain_ctx[2], *iobuf_buf;
|
|
struct spdk_memory_domain *domain[2];
|
|
struct spdk_iobuf_buffer *cache_entry;
|
|
spdk_iobuf_buffer_stailq_t small_cache;
|
|
uint32_t small_cache_count;
|
|
int i, rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
/* Intercept decompress to make it simply copy the data, so that we can chain multiple
|
|
* decompress operations together in one sequence.
|
|
*/
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].submit = ut_submit_decompress;
|
|
g_seq_operations[ACCEL_OPC_COPY].submit = sw_accel_submit_tasks;
|
|
g_seq_operations[ACCEL_OPC_FILL].submit = sw_accel_submit_tasks;
|
|
|
|
/* Check the simplest case: one operation using accel buffer as destination + copy operation
|
|
* specifying the actual destination buffer
|
|
*/
|
|
memset(srcbuf, 0xa5, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
memset(expected, 0xa5, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
src_iovs[0].iov_base = srcbuf;
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = buf[0];
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, domain[0], domain_ctx[0],
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = buf[0];
|
|
src_iovs[1].iov_len = 4096;
|
|
dst_iovs[1].iov_base = dstbuf;
|
|
dst_iovs[1].iov_len = 4096;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
|
|
/* Start with a fill operation using accel buffer, followed by a decompress using another
|
|
* accel buffer as dst, followed by a copy operation specifying dst buffer of the whole
|
|
* sequence
|
|
*/
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
memset(expected, 0x5a, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 4096, domain[0], domain_ctx[0], 0x5a, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[1], &domain[1], &domain_ctx[1]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[1]);
|
|
|
|
src_iovs[0].iov_base = buf[0];
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = buf[1];
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, domain[1], domain_ctx[1],
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = buf[1];
|
|
src_iovs[1].iov_len = 4096;
|
|
dst_iovs[1].iov_base = dstbuf;
|
|
dst_iovs[1].iov_len = 4096;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, domain[1], domain_ctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
spdk_accel_put_buf(ioch, buf[1], domain[1], domain_ctx[1]);
|
|
|
|
/* Check the same, but with two decompress operations with the first one being in-place */
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
memset(expected, 0x5a, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 4096, domain[0], domain_ctx[0], 0x5a, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[0].iov_base = buf[0];
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = buf[0];
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, domain[0], domain_ctx[0],
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[1], &domain[1], &domain_ctx[1]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[1]);
|
|
|
|
src_iovs[1].iov_base = buf[0];
|
|
src_iovs[1].iov_len = 4096;
|
|
dst_iovs[1].iov_base = buf[1];
|
|
dst_iovs[1].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, domain[1], domain_ctx[1],
|
|
&src_iovs[1], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[2].iov_base = buf[1];
|
|
src_iovs[2].iov_len = 4096;
|
|
dst_iovs[2].iov_base = dstbuf;
|
|
dst_iovs[2].iov_len = 4096;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, domain[1], domain_ctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
spdk_accel_put_buf(ioch, buf[1], domain[1], domain_ctx[1]);
|
|
|
|
/* Check that specifying offsets within accel buffers works correctly */
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
/* Fill in each 1K of the buffer with different pattern */
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 1024, domain[0], domain_ctx[0], 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 1024, 1024, domain[0], domain_ctx[0],
|
|
0x5a, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 2048, 1024, domain[0], domain_ctx[0],
|
|
0xfe, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 3072, 1024, domain[0], domain_ctx[0],
|
|
0xed, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[0].iov_base = buf[0];
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = dstbuf;
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
memset(&expected[0], 0xa5, 1024);
|
|
memset(&expected[1024], 0x5a, 1024);
|
|
memset(&expected[2048], 0xfe, 1024);
|
|
memset(&expected[3072], 0xed, 1024);
|
|
CU_ASSERT_EQUAL(completed, 5);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
|
|
/* Check the same but this time with a decompress operation on part of the buffer (512B
|
|
* offset) */
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
/* Fill in each 1K of the buffer with different pattern */
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 1024, domain[0], domain_ctx[0], 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 1024, 1024, domain[0], domain_ctx[0],
|
|
0x5a, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 2048, 1024, domain[0], domain_ctx[0],
|
|
0xfe, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, (char *)buf[0] + 3072, 1024, domain[0], domain_ctx[0],
|
|
0xed, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_get_buf(ioch, 3072, &buf[1], &domain[1], &domain_ctx[1]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[1]);
|
|
|
|
src_iovs[0].iov_base = (char *)buf[0] + 512;
|
|
src_iovs[0].iov_len = 3072;
|
|
dst_iovs[0].iov_base = (char *)buf[1] + 256;
|
|
dst_iovs[0].iov_len = 3072;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, domain[1], domain_ctx[1],
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
src_iovs[1].iov_base = (char *)buf[1] + 256;
|
|
src_iovs[1].iov_len = 3072;
|
|
dst_iovs[1].iov_base = dstbuf;
|
|
dst_iovs[1].iov_len = 3072;
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, domain[1], domain_ctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
memset(&expected[0], 0xa5, 512);
|
|
memset(&expected[512], 0x5a, 1024);
|
|
memset(&expected[1536], 0xfe, 1024);
|
|
memset(&expected[2560], 0xed, 512);
|
|
CU_ASSERT_EQUAL(completed, 6);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 3072), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
spdk_accel_put_buf(ioch, buf[1], domain[1], domain_ctx[1]);
|
|
|
|
/* Check that if iobuf pool is empty, the sequence processing will wait until a buffer is
|
|
* available
|
|
*/
|
|
accel_ch = spdk_io_channel_get_ctx(ioch);
|
|
small_cache_count = accel_ch->iobuf.small.cache_count;
|
|
STAILQ_INIT(&small_cache);
|
|
STAILQ_SWAP(&accel_ch->iobuf.small.cache, &small_cache, spdk_iobuf_buffer);
|
|
accel_ch->iobuf.small.cache_count = 0;
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
|
|
/* First allocate a single buffer used by two operations */
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
memset(expected, 0xa5, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 4096, domain[0], domain_ctx[0], 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[0].iov_base = buf[0];
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = dstbuf;
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 0);
|
|
CU_ASSERT(!ut_seq.complete);
|
|
|
|
/* Get a buffer and return it to the pool to trigger the sequence to finish */
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
iobuf_buf = spdk_iobuf_get(&accel_ch->iobuf, 4096, NULL, NULL);
|
|
CU_ASSERT_PTR_NOT_NULL(iobuf_buf);
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
spdk_iobuf_put(&accel_ch->iobuf, iobuf_buf, 4096);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
|
|
/* Return the buffers back to the cache */
|
|
while (!STAILQ_EMPTY(&accel_ch->iobuf.small.cache)) {
|
|
cache_entry = STAILQ_FIRST(&accel_ch->iobuf.small.cache);
|
|
STAILQ_REMOVE_HEAD(&accel_ch->iobuf.small.cache, stailq);
|
|
STAILQ_INSERT_HEAD(&small_cache, cache_entry, stailq);
|
|
small_cache_count++;
|
|
}
|
|
accel_ch->iobuf.small.cache_count = 0;
|
|
|
|
/* Check a bit more complex scenario, with two buffers in the sequence */
|
|
memset(srcbuf, 0x0, 4096);
|
|
memset(dstbuf, 0x0, 4096);
|
|
memset(expected, 0x5a, 4096);
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[0], &domain[0], &domain_ctx[0]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[0]);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf[0], 4096, domain[0], domain_ctx[0], 0x5a, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf[1], &domain[1], &domain_ctx[1]);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
CU_ASSERT_PTR_NOT_NULL(buf[1]);
|
|
|
|
src_iovs[0].iov_base = buf[0];
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = buf[1];
|
|
dst_iovs[0].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, domain[1], domain_ctx[1],
|
|
&src_iovs[0], 1, domain[0], domain_ctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = buf[1];
|
|
src_iovs[1].iov_len = 4096;
|
|
dst_iovs[1].iov_base = dstbuf;
|
|
dst_iovs[1].iov_len = 4096;
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, domain[1], domain_ctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 0);
|
|
CU_ASSERT(!ut_seq.complete);
|
|
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
iobuf_buf = spdk_iobuf_get(&accel_ch->iobuf, 4096, NULL, NULL);
|
|
CU_ASSERT_PTR_NOT_NULL(iobuf_buf);
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
spdk_iobuf_put(&accel_ch->iobuf, iobuf_buf, 4096);
|
|
|
|
/* One buffer is not enough to finish the whole sequence */
|
|
poll_threads();
|
|
CU_ASSERT(!ut_seq.complete);
|
|
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
iobuf_buf = spdk_iobuf_get(&accel_ch->iobuf, 4096, NULL, NULL);
|
|
CU_ASSERT_PTR_NOT_NULL(iobuf_buf);
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
spdk_iobuf_put(&accel_ch->iobuf, iobuf_buf, 4096);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
spdk_accel_put_buf(ioch, buf[0], domain[0], domain_ctx[0]);
|
|
spdk_accel_put_buf(ioch, buf[1], domain[1], domain_ctx[1]);
|
|
|
|
/* Return the buffers back to the cache */
|
|
while (!STAILQ_EMPTY(&accel_ch->iobuf.small.cache)) {
|
|
cache_entry = STAILQ_FIRST(&accel_ch->iobuf.small.cache);
|
|
STAILQ_REMOVE_HEAD(&accel_ch->iobuf.small.cache, stailq);
|
|
STAILQ_INSERT_HEAD(&small_cache, cache_entry, stailq);
|
|
small_cache_count++;
|
|
}
|
|
accel_ch->iobuf.small.cache_count = 0;
|
|
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
STAILQ_SWAP(&accel_ch->iobuf.small.cache, &small_cache, spdk_iobuf_buffer);
|
|
accel_ch->iobuf.small.cache_count = small_cache_count;
|
|
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static void
|
|
ut_domain_ctx_init(struct ut_domain_ctx *ctx, void *base, size_t len, struct iovec *expected)
|
|
{
|
|
ctx->iov.iov_base = base;
|
|
ctx->iov.iov_len = len;
|
|
ctx->expected = *expected;
|
|
ctx->pull_submit_status = 0;
|
|
ctx->push_submit_status = 0;
|
|
ctx->pull_complete_status = 0;
|
|
ctx->push_complete_status = 0;
|
|
}
|
|
|
|
static void
|
|
test_sequence_memory_domain(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct accel_io_channel *accel_ch;
|
|
struct ut_sequence ut_seq;
|
|
struct ut_domain_ctx domctx[4];
|
|
struct spdk_iobuf_buffer *cache_entry;
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
struct spdk_memory_domain *accel_domain;
|
|
spdk_iobuf_buffer_stailq_t small_cache;
|
|
char srcbuf[4096], dstbuf[4096], expected[4096], tmp[4096];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
uint32_t small_cache_count;
|
|
void *accel_buf, *accel_domain_ctx, *iobuf_buf;
|
|
int i, rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
g_module.supports_memory_domains = false;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
/* Intercept decompress to make it simply copy the data, so that we can chain multiple
|
|
* decompress operations together in one sequence.
|
|
*/
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].submit = ut_submit_decompress;
|
|
g_seq_operations[ACCEL_OPC_COPY].submit = sw_accel_submit_tasks;
|
|
g_seq_operations[ACCEL_OPC_FILL].submit = sw_accel_submit_tasks;
|
|
|
|
/* First check the simplest case - single fill operation with dstbuf in domain */
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(dstbuf, 0x0, sizeof(dstbuf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
/* Use some garbage pointer as dst and use domain ctx to get the actual dstbuf */
|
|
dst_iovs[0].iov_base = (void *)0xcafebabe;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, dst_iovs[0].iov_base, dst_iovs[0].iov_len,
|
|
g_ut_domain, &domctx[0], 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(expected, dstbuf, sizeof(dstbuf)), 0);
|
|
|
|
/* Check operation with both srcbuf and dstbuf in remote memory domain */
|
|
memset(expected, 0x5a, sizeof(expected));
|
|
memset(dstbuf, 0x0, sizeof(dstbuf));
|
|
memset(srcbuf, 0x5a, sizeof(dstbuf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xcafebabe;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xbeefdead;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(expected, dstbuf, sizeof(dstbuf)), 0);
|
|
|
|
/* Two operations using a buffer in remote domain */
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(srcbuf, 0xa5, sizeof(srcbuf));
|
|
memset(tmp, 0x0, sizeof(tmp));
|
|
memset(dstbuf, 0x0, sizeof(dstbuf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xcafebabe;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xbeefdead;
|
|
dst_iovs[0].iov_len = sizeof(tmp);
|
|
ut_domain_ctx_init(&domctx[0], tmp, sizeof(tmp), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = (void *)0xbeefdead;
|
|
src_iovs[1].iov_len = sizeof(tmp);
|
|
dst_iovs[1].iov_base = (void *)0xa5a5a5a5;
|
|
dst_iovs[1].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[2], dstbuf, sizeof(dstbuf), &dst_iovs[1]);
|
|
ut_domain_ctx_init(&domctx[3], tmp, sizeof(tmp), &src_iovs[1]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, g_ut_domain, &domctx[2],
|
|
&src_iovs[1], 1, g_ut_domain, &domctx[3], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(expected, dstbuf, sizeof(dstbuf)), 0);
|
|
|
|
/* Use buffer in remote memory domain and accel buffer at the same time */
|
|
memset(expected, 0x5a, sizeof(expected));
|
|
memset(srcbuf, 0x5a, sizeof(expected));
|
|
memset(dstbuf, 0x0, sizeof(dstbuf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
rc = spdk_accel_get_buf(ioch, sizeof(dstbuf), &accel_buf, &accel_domain, &accel_domain_ctx);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = accel_buf;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, accel_domain, accel_domain_ctx,
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[0], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = accel_buf;
|
|
src_iovs[1].iov_len = sizeof(dstbuf);
|
|
dst_iovs[1].iov_base = (void *)0xbeeffeed;
|
|
dst_iovs[1].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[1], dstbuf, sizeof(dstbuf), &dst_iovs[1]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, g_ut_domain, &domctx[1],
|
|
&src_iovs[1], 1, accel_domain, accel_domain_ctx, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(expected, dstbuf, sizeof(dstbuf)), 0);
|
|
spdk_accel_put_buf(ioch, accel_buf, accel_domain, accel_domain_ctx);
|
|
|
|
/* Check that a sequence with memory domains is correctly executed if buffers are not
|
|
* immediately available */
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(srcbuf, 0xa5, sizeof(srcbuf));
|
|
memset(dstbuf, 0x0, sizeof(dstbuf));
|
|
completed = 0;
|
|
seq = NULL;
|
|
/* Make sure the buffer pool is empty */
|
|
accel_ch = spdk_io_channel_get_ctx(ioch);
|
|
small_cache_count = accel_ch->iobuf.small.cache_count;
|
|
STAILQ_INIT(&small_cache);
|
|
STAILQ_SWAP(&accel_ch->iobuf.small.cache, &small_cache, spdk_iobuf_buffer);
|
|
accel_ch->iobuf.small.cache_count = 0;
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
|
|
src_iovs[0].iov_base = (void *)0xdeadbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 0);
|
|
CU_ASSERT(!ut_seq.complete);
|
|
|
|
/* Get a buffer and return it to the pool to trigger the sequence to resume. It shouldn't
|
|
* be able to complete, as it needs two buffers */
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
iobuf_buf = spdk_iobuf_get(&accel_ch->iobuf, sizeof(dstbuf), NULL, NULL);
|
|
CU_ASSERT_PTR_NOT_NULL(iobuf_buf);
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
spdk_iobuf_put(&accel_ch->iobuf, iobuf_buf, sizeof(dstbuf));
|
|
|
|
CU_ASSERT_EQUAL(completed, 0);
|
|
CU_ASSERT(!ut_seq.complete);
|
|
|
|
/* Return another buffer, this time the sequence should finish */
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
iobuf_buf = spdk_iobuf_get(&accel_ch->iobuf, sizeof(dstbuf), NULL, NULL);
|
|
CU_ASSERT_PTR_NOT_NULL(iobuf_buf);
|
|
MOCK_SET(spdk_mempool_get, NULL);
|
|
spdk_iobuf_put(&accel_ch->iobuf, iobuf_buf, sizeof(dstbuf));
|
|
|
|
poll_threads();
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(expected, dstbuf, sizeof(dstbuf)), 0);
|
|
|
|
/* Return the buffers back to the cache */
|
|
while (!STAILQ_EMPTY(&accel_ch->iobuf.small.cache)) {
|
|
cache_entry = STAILQ_FIRST(&accel_ch->iobuf.small.cache);
|
|
STAILQ_REMOVE_HEAD(&accel_ch->iobuf.small.cache, stailq);
|
|
STAILQ_INSERT_HEAD(&small_cache, cache_entry, stailq);
|
|
small_cache_count++;
|
|
}
|
|
accel_ch->iobuf.small.cache_count = 0;
|
|
|
|
MOCK_CLEAR(spdk_mempool_get);
|
|
STAILQ_SWAP(&accel_ch->iobuf.small.cache, &small_cache, spdk_iobuf_buffer);
|
|
accel_ch->iobuf.small.cache_count = small_cache_count;
|
|
|
|
/* Check error cases, starting with an error from spdk_memory_domain_pull_data() */
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xdeadbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
domctx[1].pull_submit_status = -E2BIG;
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -E2BIG);
|
|
|
|
/* Check completion error from spdk_memory_domain_pull_data() */
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xdeadbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
domctx[1].pull_complete_status = -EACCES;
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EACCES);
|
|
|
|
/* Check submission error from spdk_memory_domain_push_data() */
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xdeadbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
domctx[0].push_submit_status = -EADDRINUSE;
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EADDRINUSE);
|
|
|
|
/* Check completion error from spdk_memory_domain_push_data() */
|
|
completed = 0;
|
|
seq = NULL;
|
|
|
|
src_iovs[0].iov_base = (void *)0xdeadbeef;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
domctx[0].push_complete_status = -EADDRNOTAVAIL;
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EADDRNOTAVAIL);
|
|
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static int
|
|
ut_submit_decompress_memory_domain(struct spdk_io_channel *ch, struct spdk_accel_task *task)
|
|
{
|
|
struct ut_domain_ctx *ctx;
|
|
struct iovec *src_iovs, *dst_iovs;
|
|
uint32_t src_iovcnt, dst_iovcnt;
|
|
|
|
src_iovs = task->s.iovs;
|
|
dst_iovs = task->d.iovs;
|
|
src_iovcnt = task->s.iovcnt;
|
|
dst_iovcnt = task->d.iovcnt;
|
|
|
|
if (task->src_domain != NULL) {
|
|
ctx = task->src_domain_ctx;
|
|
CU_ASSERT_EQUAL(memcmp(task->s.iovs, &ctx->expected, sizeof(struct iovec)), 0);
|
|
src_iovs = &ctx->iov;
|
|
src_iovcnt = 1;
|
|
}
|
|
|
|
if (task->dst_domain != NULL) {
|
|
ctx = task->dst_domain_ctx;
|
|
CU_ASSERT_EQUAL(memcmp(task->d.iovs, &ctx->expected, sizeof(struct iovec)), 0);
|
|
dst_iovs = &ctx->iov;
|
|
dst_iovcnt = 1;
|
|
}
|
|
|
|
spdk_iovcpy(src_iovs, src_iovcnt, dst_iovs, dst_iovcnt);
|
|
spdk_accel_task_complete(task, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
test_sequence_module_memory_domain(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
struct spdk_memory_domain *accel_domain;
|
|
struct ut_sequence ut_seq;
|
|
struct ut_domain_ctx domctx[2];
|
|
struct iovec src_iovs[2], dst_iovs[2];
|
|
void *buf, *accel_domain_ctx;
|
|
char srcbuf[4096], dstbuf[4096], tmp[4096], expected[4096];
|
|
int i, rc, completed;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
g_module.supports_memory_domains = true;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].submit = ut_submit_decompress_memory_domain;
|
|
g_seq_operations[ACCEL_OPC_FILL].submit = sw_accel_submit_tasks;
|
|
|
|
/* Check a sequence with both buffers in memory domains */
|
|
memset(srcbuf, 0xa5, sizeof(srcbuf));
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(dstbuf, 0, sizeof(dstbuf));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
src_iovs[0].iov_base = (void *)0xcafebabe;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
ut_domain_ctx_init(&domctx[1], srcbuf, sizeof(srcbuf), &src_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 1);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
|
|
/* Check two operations each with a single buffer in memory domain */
|
|
memset(srcbuf, 0x5a, sizeof(srcbuf));
|
|
memset(expected, 0x5a, sizeof(expected));
|
|
memset(dstbuf, 0, sizeof(dstbuf));
|
|
memset(tmp, 0, sizeof(tmp));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
src_iovs[0].iov_base = srcbuf;
|
|
src_iovs[0].iov_len = sizeof(srcbuf);
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(tmp);
|
|
ut_domain_ctx_init(&domctx[0], tmp, sizeof(tmp), &dst_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[1].iov_base = (void *)0xfeedbeef;
|
|
src_iovs[1].iov_len = sizeof(tmp);
|
|
dst_iovs[1].iov_base = dstbuf;
|
|
dst_iovs[1].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[1], tmp, sizeof(tmp), &src_iovs[1]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, g_ut_domain, &domctx[1], 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
|
|
/* Check a sequence with an accel buffer and a buffer in a regular memory domain */
|
|
memset(expected, 0xa5, sizeof(expected));
|
|
memset(dstbuf, 0, sizeof(dstbuf));
|
|
memset(tmp, 0, sizeof(tmp));
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
rc = spdk_accel_get_buf(ioch, 4096, &buf, &accel_domain, &accel_domain_ctx);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, buf, 4096, accel_domain, accel_domain_ctx,
|
|
0xa5, 0, ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
src_iovs[0].iov_base = buf;
|
|
src_iovs[0].iov_len = 4096;
|
|
dst_iovs[0].iov_base = (void *)0xfeedbeef;
|
|
dst_iovs[0].iov_len = sizeof(dstbuf);
|
|
ut_domain_ctx_init(&domctx[0], dstbuf, sizeof(dstbuf), &dst_iovs[0]);
|
|
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, g_ut_domain, &domctx[0],
|
|
&src_iovs[0], 1, accel_domain, accel_domain_ctx, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 2);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(dstbuf, expected, 4096), 0);
|
|
|
|
spdk_accel_put_buf(ioch, buf, accel_domain, accel_domain_ctx);
|
|
|
|
g_module.supports_memory_domains = false;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
#ifdef SPDK_CONFIG_ISAL_CRYPTO
|
|
static void
|
|
ut_encrypt_cb(void *cb_arg, int status)
|
|
{
|
|
int *completed = cb_arg;
|
|
|
|
CU_ASSERT_EQUAL(status, 0);
|
|
|
|
*completed = 1;
|
|
}
|
|
|
|
static void
|
|
test_sequence_crypto(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct spdk_accel_crypto_key *key;
|
|
struct spdk_accel_crypto_key_create_param key_params = {
|
|
.cipher = "AES_XTS",
|
|
.hex_key = "00112233445566778899aabbccddeeff",
|
|
.hex_key2 = "ffeeddccbbaa99887766554433221100",
|
|
.key_name = "ut_key",
|
|
};
|
|
struct ut_sequence ut_seq;
|
|
unsigned char buf[4096], encrypted[4096] = {}, data[4096], tmp[3][4096];
|
|
struct iovec src_iovs[4], dst_iovs[4];
|
|
int rc, completed = 0;
|
|
size_t i;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
|
|
rc = spdk_accel_crypto_key_create(&key_params);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
key = spdk_accel_crypto_key_get(key_params.key_name);
|
|
SPDK_CU_ASSERT_FATAL(key != NULL);
|
|
|
|
for (i = 0; i < sizeof(data); ++i) {
|
|
data[i] = (uint8_t)i & 0xff;
|
|
}
|
|
|
|
dst_iovs[0].iov_base = encrypted;
|
|
dst_iovs[0].iov_len = sizeof(encrypted);
|
|
src_iovs[0].iov_base = data;
|
|
src_iovs[0].iov_len = sizeof(data);
|
|
rc = spdk_accel_submit_encrypt(ioch, key, &dst_iovs[0], 1, &src_iovs[0], 1, 0, 4096, 0,
|
|
ut_encrypt_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
while (!completed) {
|
|
poll_threads();
|
|
}
|
|
|
|
/* Verify that encryption operation in a sequence produces the same result */
|
|
seq = NULL;
|
|
completed = 0;
|
|
|
|
dst_iovs[0].iov_base = tmp[0];
|
|
dst_iovs[0].iov_len = sizeof(tmp[0]);
|
|
src_iovs[0].iov_base = data;
|
|
src_iovs[0].iov_len = sizeof(data);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[1];
|
|
dst_iovs[1].iov_len = sizeof(tmp[1]);
|
|
src_iovs[1].iov_base = tmp[0];
|
|
src_iovs[1].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[1];
|
|
src_iovs[2].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, encrypted, sizeof(buf)), 0);
|
|
|
|
/* Check that decryption produces the original buffer */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
dst_iovs[0].iov_base = tmp[0];
|
|
dst_iovs[0].iov_len = sizeof(tmp[0]);
|
|
src_iovs[0].iov_base = encrypted;
|
|
src_iovs[0].iov_len = sizeof(encrypted);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[1];
|
|
dst_iovs[1].iov_len = sizeof(tmp[1]);
|
|
src_iovs[1].iov_base = tmp[0];
|
|
src_iovs[1].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[1];
|
|
src_iovs[2].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, data, sizeof(buf)), 0);
|
|
|
|
/* Check encrypt + decrypt in a single sequence */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
dst_iovs[0].iov_base = tmp[0];
|
|
dst_iovs[0].iov_len = sizeof(tmp[0]);
|
|
src_iovs[0].iov_base = data;
|
|
src_iovs[0].iov_len = sizeof(data);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[1];
|
|
dst_iovs[1].iov_len = sizeof(tmp[1]);
|
|
src_iovs[1].iov_base = tmp[0];
|
|
src_iovs[1].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
|
|
dst_iovs[2].iov_base = tmp[2];
|
|
dst_iovs[2].iov_len = sizeof(tmp[2]);
|
|
src_iovs[2].iov_base = tmp[1];
|
|
src_iovs[2].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, key, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[3].iov_base = buf;
|
|
dst_iovs[3].iov_len = sizeof(buf);
|
|
src_iovs[3].iov_base = tmp[2];
|
|
src_iovs[3].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[3], 1, NULL, NULL,
|
|
&src_iovs[3], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(memcmp(buf, data, sizeof(buf)), 0);
|
|
|
|
rc = spdk_accel_crypto_key_destroy(key);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
#endif /* SPDK_CONFIG_ISAL_CRYPTO */
|
|
|
|
static int
|
|
ut_submit_crypto(struct spdk_io_channel *ch, struct spdk_accel_task *task)
|
|
{
|
|
spdk_iovmove(task->s.iovs, task->s.iovcnt, task->d.iovs, task->d.iovcnt);
|
|
|
|
spdk_accel_task_complete(task, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct ut_driver_operation {
|
|
int complete_status;
|
|
int submit_status;
|
|
int count;
|
|
bool supported;
|
|
};
|
|
|
|
static struct ut_driver_operation g_drv_operations[ACCEL_OPC_LAST];
|
|
|
|
static int
|
|
ut_driver_execute_sequence(struct spdk_accel_sequence *seq)
|
|
{
|
|
struct spdk_accel_task *task;
|
|
struct ut_driver_operation *drv_ops;
|
|
|
|
while ((task = spdk_accel_sequence_first_task(seq)) != NULL) {
|
|
drv_ops = &g_drv_operations[task->op_code];
|
|
if (!drv_ops->supported) {
|
|
break;
|
|
}
|
|
|
|
drv_ops->count++;
|
|
if (drv_ops->submit_status != 0) {
|
|
return drv_ops->submit_status;
|
|
}
|
|
|
|
if (drv_ops->complete_status != 0) {
|
|
spdk_accel_task_complete(task, drv_ops->complete_status);
|
|
break;
|
|
}
|
|
|
|
switch (task->op_code) {
|
|
case ACCEL_OPC_DECOMPRESS:
|
|
spdk_iovmove(task->s.iovs, task->s.iovcnt, task->d.iovs, task->d.iovcnt);
|
|
break;
|
|
case ACCEL_OPC_FILL:
|
|
spdk_iov_memset(task->d.iovs, task->d.iovcnt,
|
|
(int)(task->fill_pattern & 0xff));
|
|
break;
|
|
default:
|
|
CU_ASSERT(0 && "unexpected opcode");
|
|
break;
|
|
}
|
|
|
|
spdk_accel_task_complete(task, 0);
|
|
}
|
|
|
|
spdk_accel_sequence_continue(seq);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct spdk_accel_driver g_ut_driver = {
|
|
.name = "ut",
|
|
.execute_sequence = ut_driver_execute_sequence,
|
|
};
|
|
|
|
SPDK_ACCEL_DRIVER_REGISTER(ut, &g_ut_driver);
|
|
|
|
static void
|
|
test_sequence_driver(void)
|
|
{
|
|
struct spdk_accel_sequence *seq = NULL;
|
|
struct spdk_io_channel *ioch;
|
|
struct spdk_accel_crypto_key key = {};
|
|
struct ut_sequence ut_seq;
|
|
struct accel_module modules[ACCEL_OPC_LAST];
|
|
char buf[4096], tmp[3][4096], expected[4096];
|
|
struct iovec src_iovs[3], dst_iovs[3];
|
|
int i, rc, completed = 0;
|
|
|
|
ioch = spdk_accel_get_io_channel();
|
|
SPDK_CU_ASSERT_FATAL(ioch != NULL);
|
|
rc = spdk_accel_set_driver("ut");
|
|
SPDK_CU_ASSERT_FATAL(rc == 0);
|
|
|
|
/* Override the submit_tasks function */
|
|
g_module_if.submit_tasks = ut_sequnce_submit_tasks;
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
modules[i] = g_modules_opc[i];
|
|
g_modules_opc[i] = g_module;
|
|
}
|
|
/* Intercept crypto operations, as they should be executed by an accel module */
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].submit = ut_submit_crypto;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].submit = ut_submit_crypto;
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
|
|
g_drv_operations[ACCEL_OPC_FILL].supported = true;
|
|
g_drv_operations[ACCEL_OPC_DECOMPRESS].supported = true;
|
|
|
|
/* First check a sequence that is fully executed using a driver, with the copy at the end
|
|
* being removed */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(&expected[0], 0xa5, 2048);
|
|
memset(&expected[2048], 0xbe, 2048);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[0], 2048, NULL, NULL, 0xa5, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
rc = spdk_accel_append_fill(&seq, ioch, &tmp[0][2048], 2048, NULL, NULL, 0xbe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_copy(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_COPY].count, 0);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 2);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
g_drv_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_drv_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
|
|
/* Check a sequence when the first two operations are executed by a driver, while the rest
|
|
* is executed via modules */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(tmp[2], 0, sizeof(tmp[2]));
|
|
memset(&expected[0], 0xfe, 4096);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[0], sizeof(tmp[0]), NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = sizeof(tmp[2]);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].count = 0;
|
|
g_drv_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_drv_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
|
|
/* Check sequence when the first and last operations are executed through modules, while the
|
|
* ones in the middle are executed by the driver */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0xa5, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(tmp[2], 0, sizeof(tmp[2]));
|
|
memset(&expected[0], 0xfe, 2048);
|
|
memset(&expected[2048], 0xa5, 2048);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 2048, NULL, NULL, 0xfe, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = sizeof(tmp[2]);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].count = 0;
|
|
g_drv_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_drv_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
|
|
/* Check a sequence with operations executed by: module, driver, module, driver */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(tmp[0], 0x5a, sizeof(tmp[0]));
|
|
memset(tmp[1], 0, sizeof(tmp[1]));
|
|
memset(tmp[2], 0, sizeof(tmp[2]));
|
|
memset(&expected[0], 0xef, 2048);
|
|
memset(&expected[2048], 0x5a, 2048);
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 2048, NULL, NULL, 0xef, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = tmp[2];
|
|
dst_iovs[1].iov_len = sizeof(tmp[2]);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[2].iov_base = buf;
|
|
dst_iovs[2].iov_len = sizeof(buf);
|
|
src_iovs[2].iov_base = tmp[2];
|
|
src_iovs[2].iov_len = sizeof(tmp[2]);
|
|
rc = spdk_accel_append_decompress(&seq, ioch, &dst_iovs[2], 1, NULL, NULL,
|
|
&src_iovs[2], 1, NULL, NULL, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 4);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECOMPRESS].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 1);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_DECOMPRESS].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_seq_operations[ACCEL_OPC_DECRYPT].count = 0;
|
|
g_drv_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_drv_operations[ACCEL_OPC_DECOMPRESS].count = 0;
|
|
|
|
/* Check that an error returned from driver's execute_sequence() will fail the whole
|
|
* sequence and any subsequent operations won't be processed */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0, sizeof(expected));
|
|
memset(tmp[0], 0xa5, sizeof(tmp[0]));
|
|
g_drv_operations[ACCEL_OPC_FILL].submit_status = -EPERM;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 2048, NULL, NULL, 0xef, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -EPERM);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 0);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, 4096), 0);
|
|
|
|
g_seq_operations[ACCEL_OPC_ENCRYPT].count = 0;
|
|
g_drv_operations[ACCEL_OPC_FILL].count = 0;
|
|
g_drv_operations[ACCEL_OPC_FILL].submit_status = 0;
|
|
|
|
/* Check that a failed task completed by a driver will cause the whole sequence to be failed
|
|
* and any subsequent operations won't be processed */
|
|
seq = NULL;
|
|
completed = 0;
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(expected, 0, sizeof(expected));
|
|
memset(tmp[0], 0xa5, sizeof(tmp[0]));
|
|
g_drv_operations[ACCEL_OPC_FILL].complete_status = -ENOENT;
|
|
|
|
dst_iovs[0].iov_base = tmp[1];
|
|
dst_iovs[0].iov_len = sizeof(tmp[1]);
|
|
src_iovs[0].iov_base = tmp[0];
|
|
src_iovs[0].iov_len = sizeof(tmp[0]);
|
|
rc = spdk_accel_append_encrypt(&seq, ioch, &key, &dst_iovs[0], 1, NULL, NULL,
|
|
&src_iovs[0], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
rc = spdk_accel_append_fill(&seq, ioch, tmp[1], 2048, NULL, NULL, 0xef, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
dst_iovs[1].iov_base = buf;
|
|
dst_iovs[1].iov_len = sizeof(buf);
|
|
src_iovs[1].iov_base = tmp[1];
|
|
src_iovs[1].iov_len = sizeof(tmp[1]);
|
|
rc = spdk_accel_append_decrypt(&seq, ioch, &key, &dst_iovs[1], 1, NULL, NULL,
|
|
&src_iovs[1], 1, NULL, NULL, 0, 4096, 0,
|
|
ut_sequence_step_cb, &completed);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
ut_seq.complete = false;
|
|
rc = spdk_accel_sequence_finish(seq, ut_sequence_complete_cb, &ut_seq);
|
|
CU_ASSERT_EQUAL(rc, 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT_EQUAL(completed, 3);
|
|
CU_ASSERT(ut_seq.complete);
|
|
CU_ASSERT_EQUAL(ut_seq.status, -ENOENT);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_FILL].count, 0);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_ENCRYPT].count, 1);
|
|
CU_ASSERT_EQUAL(g_seq_operations[ACCEL_OPC_DECRYPT].count, 0);
|
|
CU_ASSERT_EQUAL(g_drv_operations[ACCEL_OPC_FILL].count, 1);
|
|
CU_ASSERT_EQUAL(memcmp(buf, expected, sizeof(buf)), 0);
|
|
|
|
for (i = 0; i < ACCEL_OPC_LAST; ++i) {
|
|
g_modules_opc[i] = modules[i];
|
|
}
|
|
|
|
/* Clear the driver so that other tests won't use it */
|
|
g_accel_driver = NULL;
|
|
memset(&g_drv_operations, 0, sizeof(g_drv_operations));
|
|
|
|
ut_clear_operations();
|
|
spdk_put_io_channel(ioch);
|
|
poll_threads();
|
|
}
|
|
|
|
static int
|
|
test_sequence_setup(void)
|
|
{
|
|
int rc;
|
|
|
|
allocate_cores(1);
|
|
allocate_threads(1);
|
|
set_thread(0);
|
|
|
|
rc = spdk_iobuf_initialize();
|
|
if (rc != 0) {
|
|
CU_ASSERT(false);
|
|
return -1;
|
|
}
|
|
|
|
rc = spdk_accel_initialize();
|
|
if (rc != 0) {
|
|
CU_ASSERT(false);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
finish_cb(void *cb_arg)
|
|
{
|
|
bool *done = cb_arg;
|
|
|
|
*done = true;
|
|
}
|
|
|
|
static int
|
|
test_sequence_cleanup(void)
|
|
{
|
|
bool done = false;
|
|
|
|
spdk_accel_finish(finish_cb, &done);
|
|
|
|
while (!done) {
|
|
poll_threads();
|
|
}
|
|
|
|
done = false;
|
|
spdk_iobuf_finish(finish_cb, &done);
|
|
while (!done) {
|
|
poll_threads();
|
|
}
|
|
|
|
free_threads();
|
|
free_cores();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
CU_pSuite suite = NULL, seq_suite;
|
|
unsigned int num_failures;
|
|
|
|
CU_set_error_action(CUEA_ABORT);
|
|
CU_initialize_registry();
|
|
|
|
/* Sequence tests require accel to be initialized normally, so run them before the other
|
|
* tests which register accel modules which aren't fully implemented, causing accel
|
|
* initialization to fail.
|
|
*/
|
|
seq_suite = CU_add_suite("accel_sequence", test_sequence_setup, test_sequence_cleanup);
|
|
CU_ADD_TEST(seq_suite, test_sequence_fill_copy);
|
|
CU_ADD_TEST(seq_suite, test_sequence_abort);
|
|
CU_ADD_TEST(seq_suite, test_sequence_append_error);
|
|
CU_ADD_TEST(seq_suite, test_sequence_completion_error);
|
|
#ifdef SPDK_CONFIG_ISAL /* accel_sw requires isa-l for compression */
|
|
CU_ADD_TEST(seq_suite, test_sequence_decompress);
|
|
CU_ADD_TEST(seq_suite, test_sequence_reverse);
|
|
#endif
|
|
CU_ADD_TEST(seq_suite, test_sequence_copy_elision);
|
|
CU_ADD_TEST(seq_suite, test_sequence_accel_buffers);
|
|
CU_ADD_TEST(seq_suite, test_sequence_memory_domain);
|
|
CU_ADD_TEST(seq_suite, test_sequence_module_memory_domain);
|
|
#ifdef SPDK_CONFIG_ISAL_CRYPTO /* accel_sw requires isa-l-crypto for crypto operations */
|
|
CU_ADD_TEST(seq_suite, test_sequence_crypto);
|
|
#endif
|
|
CU_ADD_TEST(seq_suite, test_sequence_driver);
|
|
|
|
suite = CU_add_suite("accel", test_setup, test_cleanup);
|
|
CU_ADD_TEST(suite, test_spdk_accel_task_complete);
|
|
CU_ADD_TEST(suite, test_get_task);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_copy);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_dualcast);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_compare);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_fill);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_crc32c);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_crc32cv);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_copy_crc32c);
|
|
CU_ADD_TEST(suite, test_spdk_accel_submit_xor);
|
|
CU_ADD_TEST(suite, test_spdk_accel_module_find_by_name);
|
|
CU_ADD_TEST(suite, test_spdk_accel_module_register);
|
|
|
|
CU_basic_set_mode(CU_BRM_VERBOSE);
|
|
CU_basic_run_tests();
|
|
num_failures = CU_get_number_of_failures();
|
|
CU_cleanup_registry();
|
|
|
|
return num_failures;
|
|
}
|