2022-06-03 19:15:11 +00:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
2022-11-01 20:26:26 +00:00
|
|
|
* Copyright (C) 2018 Intel Corporation.
|
2018-02-26 20:22:38 +00:00
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "spdk_cunit.h"
|
|
|
|
|
2018-08-15 21:19:49 +00:00
|
|
|
#include "common/lib/ut_multithread.c"
|
2018-02-22 12:48:13 +00:00
|
|
|
#include "unit/lib/json_mock.c"
|
2018-02-26 20:22:38 +00:00
|
|
|
|
2018-09-27 19:38:15 +00:00
|
|
|
#include "spdk/config.h"
|
2018-02-28 01:53:42 +00:00
|
|
|
/* HACK: disable VTune integration so the unit test doesn't need VTune headers and libs to build */
|
|
|
|
#undef SPDK_CONFIG_VTUNE
|
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
#include "bdev/bdev.c"
|
|
|
|
#include "bdev/part.c"
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
struct ut_expected_io {
|
|
|
|
};
|
|
|
|
|
|
|
|
struct bdev_ut_channel {
|
|
|
|
TAILQ_HEAD(, spdk_bdev_io) outstanding_io;
|
|
|
|
uint32_t outstanding_io_count;
|
|
|
|
TAILQ_HEAD(, ut_expected_io) expected_io;
|
|
|
|
};
|
|
|
|
|
|
|
|
static uint32_t g_part_ut_io_device;
|
|
|
|
static struct bdev_ut_channel *g_bdev_ut_channel;
|
|
|
|
|
2019-03-29 15:04:48 +00:00
|
|
|
DEFINE_STUB(spdk_notify_send, uint64_t, (const char *type, const char *ctx), 0);
|
|
|
|
DEFINE_STUB(spdk_notify_type_register, struct spdk_notify_type *, (const char *type), NULL);
|
2021-10-20 06:35:28 +00:00
|
|
|
DEFINE_STUB(spdk_memory_domain_get_dma_device_id, const char *, (struct spdk_memory_domain *domain),
|
|
|
|
"test_domain");
|
|
|
|
DEFINE_STUB(spdk_memory_domain_get_dma_device_type, enum spdk_dma_device_type,
|
|
|
|
(struct spdk_memory_domain *domain), 0);
|
|
|
|
|
|
|
|
DEFINE_RETURN_MOCK(spdk_memory_domain_pull_data, int);
|
|
|
|
int
|
|
|
|
spdk_memory_domain_pull_data(struct spdk_memory_domain *src_domain, void *src_domain_ctx,
|
|
|
|
struct iovec *src_iov, uint32_t src_iov_cnt, struct iovec *dst_iov, uint32_t dst_iov_cnt,
|
|
|
|
spdk_memory_domain_data_cpl_cb cpl_cb, void *cpl_cb_arg)
|
|
|
|
{
|
|
|
|
HANDLE_RETURN_MOCK(spdk_memory_domain_pull_data);
|
|
|
|
|
|
|
|
cpl_cb(cpl_cb_arg, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEFINE_RETURN_MOCK(spdk_memory_domain_push_data, int);
|
|
|
|
int
|
|
|
|
spdk_memory_domain_push_data(struct spdk_memory_domain *dst_domain, void *dst_domain_ctx,
|
|
|
|
struct iovec *dst_iov, uint32_t dst_iovcnt, struct iovec *src_iov, uint32_t src_iovcnt,
|
|
|
|
spdk_memory_domain_data_cpl_cb cpl_cb, void *cpl_cb_arg)
|
|
|
|
{
|
|
|
|
HANDLE_RETURN_MOCK(spdk_memory_domain_push_data);
|
|
|
|
|
|
|
|
cpl_cb(cpl_cb_arg, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
2018-08-31 20:13:32 +00:00
|
|
|
|
2018-08-30 14:59:10 +00:00
|
|
|
static void
|
|
|
|
_part_cleanup(struct spdk_bdev_part *part)
|
|
|
|
{
|
2022-08-16 14:17:05 +00:00
|
|
|
spdk_io_device_unregister(part, NULL);
|
2018-08-30 14:59:10 +00:00
|
|
|
free(part->internal.bdev.name);
|
|
|
|
free(part->internal.bdev.product_name);
|
|
|
|
}
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
static struct spdk_io_channel *
|
|
|
|
part_ut_get_io_channel(void *ctx)
|
|
|
|
{
|
|
|
|
return spdk_get_io_channel(&g_part_ut_io_device);
|
|
|
|
}
|
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
void
|
|
|
|
spdk_scsi_nvme_translate(const struct spdk_bdev_io *bdev_io,
|
|
|
|
int *sc, int *sk, int *asc, int *ascq)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
static int
|
|
|
|
bdev_ut_create_ch(void *io_device, void *ctx_buf)
|
|
|
|
{
|
|
|
|
struct bdev_ut_channel *ch = ctx_buf;
|
|
|
|
|
|
|
|
CU_ASSERT(g_bdev_ut_channel == NULL);
|
|
|
|
g_bdev_ut_channel = ch;
|
|
|
|
g_part_ut_io_device++;
|
|
|
|
|
|
|
|
TAILQ_INIT(&ch->outstanding_io);
|
|
|
|
ch->outstanding_io_count = 0;
|
|
|
|
TAILQ_INIT(&ch->expected_io);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bdev_ut_destroy_ch(void *io_device, void *ctx_buf)
|
|
|
|
{
|
|
|
|
CU_ASSERT(g_bdev_ut_channel != NULL);
|
|
|
|
g_bdev_ut_channel = NULL;
|
|
|
|
g_part_ut_io_device--;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct spdk_bdev_module bdev_ut_if;
|
|
|
|
|
|
|
|
static int
|
|
|
|
bdev_ut_module_init(void)
|
|
|
|
{
|
|
|
|
spdk_io_device_register(&g_part_ut_io_device, bdev_ut_create_ch, bdev_ut_destroy_ch,
|
|
|
|
sizeof(struct bdev_ut_channel), NULL);
|
|
|
|
spdk_bdev_module_init_done(&bdev_ut_if);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bdev_ut_module_fini(void)
|
|
|
|
{
|
|
|
|
spdk_io_device_unregister(&g_part_ut_io_device, NULL);
|
|
|
|
}
|
|
|
|
|
2018-03-09 22:20:21 +00:00
|
|
|
struct spdk_bdev_module bdev_ut_if = {
|
2018-03-06 18:52:46 +00:00
|
|
|
.name = "bdev_ut",
|
2022-09-23 06:05:38 +00:00
|
|
|
.module_init = bdev_ut_module_init,
|
|
|
|
.module_fini = bdev_ut_module_fini,
|
|
|
|
.async_init = true,
|
2018-03-06 18:52:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void vbdev_ut_examine(struct spdk_bdev *bdev);
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
static int
|
|
|
|
vbdev_ut_module_init(void)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vbdev_ut_module_fini(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-03-09 22:20:21 +00:00
|
|
|
struct spdk_bdev_module vbdev_ut_if = {
|
2018-03-06 18:52:46 +00:00
|
|
|
.name = "vbdev_ut",
|
2022-09-23 06:05:38 +00:00
|
|
|
.module_init = vbdev_ut_module_init,
|
|
|
|
.module_fini = vbdev_ut_module_fini,
|
2018-06-26 12:07:11 +00:00
|
|
|
.examine_config = vbdev_ut_examine,
|
2018-03-06 18:52:46 +00:00
|
|
|
};
|
|
|
|
|
2019-02-05 10:46:48 +00:00
|
|
|
SPDK_BDEV_MODULE_REGISTER(bdev_ut, &bdev_ut_if)
|
|
|
|
SPDK_BDEV_MODULE_REGISTER(vbdev_ut, &vbdev_ut_if)
|
2018-02-26 20:22:38 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
vbdev_ut_examine(struct spdk_bdev *bdev)
|
|
|
|
{
|
2018-03-06 18:52:46 +00:00
|
|
|
spdk_bdev_module_examine_done(&vbdev_ut_if);
|
2018-02-26 20:22:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
__destruct(void *ctx)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct spdk_bdev_fn_table base_fn_table = {
|
|
|
|
.destruct = __destruct,
|
2022-09-23 06:05:38 +00:00
|
|
|
.get_io_channel = part_ut_get_io_channel,
|
2018-02-26 20:22:38 +00:00
|
|
|
};
|
|
|
|
static struct spdk_bdev_fn_table part_fn_table = {
|
|
|
|
.destruct = __destruct,
|
|
|
|
};
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
static void
|
|
|
|
bdev_init_cb(void *arg, int rc)
|
|
|
|
{
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bdev_fini_cb(void *arg)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bdev_ut_event_cb(enum spdk_bdev_event_type type, struct spdk_bdev *bdev, void *event_ctx)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
static void
|
|
|
|
part_test(void)
|
|
|
|
{
|
|
|
|
struct spdk_bdev_part_base *base;
|
2018-03-23 19:35:21 +00:00
|
|
|
struct spdk_bdev_part part1 = {};
|
|
|
|
struct spdk_bdev_part part2 = {};
|
2018-02-26 20:22:38 +00:00
|
|
|
struct spdk_bdev bdev_base = {};
|
|
|
|
SPDK_BDEV_PART_TAILQ tailq = TAILQ_HEAD_INITIALIZER(tailq);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
bdev_base.name = "base";
|
|
|
|
bdev_base.fn_table = &base_fn_table;
|
2018-03-06 18:52:46 +00:00
|
|
|
bdev_base.module = &bdev_ut_if;
|
2018-02-26 20:22:38 +00:00
|
|
|
rc = spdk_bdev_register(&bdev_base);
|
|
|
|
CU_ASSERT(rc == 0);
|
2021-03-02 12:09:54 +00:00
|
|
|
rc = spdk_bdev_part_base_construct_ext("base", NULL, &vbdev_ut_if,
|
|
|
|
&part_fn_table, &tailq, NULL,
|
|
|
|
NULL, 0, NULL, NULL, &base);
|
2018-06-21 22:23:42 +00:00
|
|
|
|
2021-03-02 12:09:54 +00:00
|
|
|
CU_ASSERT(rc == 0);
|
2018-06-21 22:23:42 +00:00
|
|
|
SPDK_CU_ASSERT_FATAL(base != NULL);
|
2018-02-26 20:22:38 +00:00
|
|
|
|
2018-08-29 16:13:32 +00:00
|
|
|
rc = spdk_bdev_part_construct(&part1, base, "test1", 0, 100, "test");
|
|
|
|
SPDK_CU_ASSERT_FATAL(rc == 0);
|
|
|
|
rc = spdk_bdev_part_construct(&part2, base, "test2", 100, 100, "test");
|
|
|
|
SPDK_CU_ASSERT_FATAL(rc == 0);
|
2018-02-26 20:22:38 +00:00
|
|
|
|
2018-11-26 13:07:28 +00:00
|
|
|
spdk_bdev_part_base_hotremove(base, &tailq);
|
2018-02-26 20:22:38 +00:00
|
|
|
|
|
|
|
spdk_bdev_part_base_free(base);
|
2018-08-30 14:59:10 +00:00
|
|
|
_part_cleanup(&part1);
|
|
|
|
_part_cleanup(&part2);
|
2018-02-26 20:22:38 +00:00
|
|
|
spdk_bdev_unregister(&bdev_base, NULL, NULL);
|
2018-08-15 21:19:49 +00:00
|
|
|
|
|
|
|
poll_threads();
|
2018-02-26 20:22:38 +00:00
|
|
|
}
|
|
|
|
|
2022-08-17 09:13:20 +00:00
|
|
|
static void
|
|
|
|
part_free_test(void)
|
|
|
|
{
|
|
|
|
struct spdk_bdev_part_base *base = NULL;
|
|
|
|
struct spdk_bdev_part *part;
|
|
|
|
struct spdk_bdev bdev_base = {};
|
|
|
|
SPDK_BDEV_PART_TAILQ tailq = TAILQ_HEAD_INITIALIZER(tailq);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
bdev_base.name = "base";
|
|
|
|
bdev_base.fn_table = &base_fn_table;
|
|
|
|
bdev_base.module = &bdev_ut_if;
|
|
|
|
rc = spdk_bdev_register(&bdev_base);
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
poll_threads();
|
|
|
|
|
|
|
|
rc = spdk_bdev_part_base_construct_ext("base", NULL, &vbdev_ut_if,
|
|
|
|
&part_fn_table, &tailq, NULL,
|
|
|
|
NULL, 0, NULL, NULL, &base);
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
CU_ASSERT(TAILQ_EMPTY(&tailq));
|
|
|
|
SPDK_CU_ASSERT_FATAL(base != NULL);
|
|
|
|
|
|
|
|
part = calloc(1, sizeof(*part));
|
|
|
|
SPDK_CU_ASSERT_FATAL(part != NULL);
|
|
|
|
rc = spdk_bdev_part_construct(part, base, "test", 0, 100, "test");
|
|
|
|
SPDK_CU_ASSERT_FATAL(rc == 0);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT(!TAILQ_EMPTY(&tailq));
|
|
|
|
|
|
|
|
spdk_bdev_unregister(&part->internal.bdev, NULL, NULL);
|
|
|
|
poll_threads();
|
|
|
|
|
|
|
|
rc = spdk_bdev_part_free(part);
|
|
|
|
CU_ASSERT(rc == 1);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT(TAILQ_EMPTY(&tailq));
|
|
|
|
|
|
|
|
spdk_bdev_unregister(&bdev_base, NULL, NULL);
|
|
|
|
poll_threads();
|
|
|
|
}
|
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
static void
|
|
|
|
part_get_io_channel_test(void)
|
|
|
|
{
|
|
|
|
struct spdk_bdev_part_base *base = NULL;
|
|
|
|
struct spdk_bdev_desc *desc = NULL;
|
|
|
|
struct spdk_io_channel *io_ch;
|
|
|
|
struct spdk_bdev_part *part;
|
|
|
|
struct spdk_bdev bdev_base = {};
|
|
|
|
SPDK_BDEV_PART_TAILQ tailq = TAILQ_HEAD_INITIALIZER(tailq);
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
spdk_bdev_initialize(bdev_init_cb, NULL);
|
|
|
|
|
|
|
|
bdev_base.name = "base";
|
|
|
|
bdev_base.blocklen = 512;
|
|
|
|
bdev_base.blockcnt = 1024;
|
|
|
|
bdev_base.fn_table = &base_fn_table;
|
|
|
|
bdev_base.module = &bdev_ut_if;
|
|
|
|
rc = spdk_bdev_register(&bdev_base);
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
|
|
|
|
rc = spdk_bdev_part_base_construct_ext("base", NULL, &vbdev_ut_if,
|
|
|
|
&part_fn_table, &tailq, NULL,
|
|
|
|
NULL, 100, NULL, NULL, &base);
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
CU_ASSERT(TAILQ_EMPTY(&tailq));
|
|
|
|
SPDK_CU_ASSERT_FATAL(base != NULL);
|
|
|
|
|
|
|
|
part = calloc(1, sizeof(*part));
|
|
|
|
SPDK_CU_ASSERT_FATAL(part != NULL);
|
|
|
|
rc = spdk_bdev_part_construct(part, base, "test", 0, 100, "test");
|
|
|
|
SPDK_CU_ASSERT_FATAL(rc == 0);
|
|
|
|
CU_ASSERT(!TAILQ_EMPTY(&tailq));
|
|
|
|
|
|
|
|
rc = spdk_bdev_open_ext("test", true, bdev_ut_event_cb, NULL, &desc);
|
|
|
|
CU_ASSERT(rc == 0);
|
|
|
|
SPDK_CU_ASSERT_FATAL(desc != NULL);
|
|
|
|
CU_ASSERT(&part->internal.bdev == spdk_bdev_desc_get_bdev(desc));
|
|
|
|
|
|
|
|
io_ch = spdk_bdev_get_io_channel(desc);
|
|
|
|
CU_ASSERT(io_ch != NULL);
|
|
|
|
CU_ASSERT(g_part_ut_io_device == 1);
|
|
|
|
|
|
|
|
spdk_put_io_channel(io_ch);
|
|
|
|
spdk_bdev_close(desc);
|
|
|
|
spdk_bdev_unregister(&part->internal.bdev, NULL, NULL);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT(g_part_ut_io_device == 0);
|
|
|
|
|
|
|
|
rc = spdk_bdev_part_free(part);
|
|
|
|
CU_ASSERT(rc == 1);
|
|
|
|
poll_threads();
|
|
|
|
CU_ASSERT(TAILQ_EMPTY(&tailq));
|
|
|
|
|
|
|
|
spdk_bdev_unregister(&bdev_base, NULL, NULL);
|
|
|
|
spdk_bdev_finish(bdev_fini_cb, NULL);
|
|
|
|
poll_threads();
|
|
|
|
}
|
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
2018-09-17 23:18:53 +00:00
|
|
|
CU_pSuite suite = NULL;
|
|
|
|
unsigned int num_failures;
|
2018-02-26 20:22:38 +00:00
|
|
|
|
2020-03-11 17:59:24 +00:00
|
|
|
CU_set_error_action(CUEA_ABORT);
|
|
|
|
CU_initialize_registry();
|
2018-02-26 20:22:38 +00:00
|
|
|
|
|
|
|
suite = CU_add_suite("bdev_part", NULL, NULL);
|
2020-03-11 17:59:24 +00:00
|
|
|
|
2020-03-11 19:15:39 +00:00
|
|
|
CU_ADD_TEST(suite, part_test);
|
2022-08-17 09:13:20 +00:00
|
|
|
CU_ADD_TEST(suite, part_free_test);
|
2022-09-23 06:05:38 +00:00
|
|
|
CU_ADD_TEST(suite, part_get_io_channel_test);
|
2018-02-26 20:22:38 +00:00
|
|
|
|
2022-09-23 06:05:38 +00:00
|
|
|
allocate_cores(1);
|
2018-08-15 21:19:49 +00:00
|
|
|
allocate_threads(1);
|
|
|
|
set_thread(0);
|
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
CU_basic_set_mode(CU_BRM_VERBOSE);
|
|
|
|
CU_basic_run_tests();
|
|
|
|
num_failures = CU_get_number_of_failures();
|
|
|
|
CU_cleanup_registry();
|
2018-08-15 21:19:49 +00:00
|
|
|
|
|
|
|
free_threads();
|
2022-09-23 06:05:38 +00:00
|
|
|
free_cores();
|
2018-08-15 21:19:49 +00:00
|
|
|
|
2018-02-26 20:22:38 +00:00
|
|
|
return num_failures;
|
|
|
|
}
|