Many open source projects have moved to using SPDX identifiers to specify license information, reducing the amount of boilerplate code in every source file. This patch replaces the bulk of SPDK .c, .cpp and Makefiles with the BSD-3-Clause identifier. Almost all of these files share the exact same license text, and this patch only modifies the files that contain the most common license text. There can be slight variations because the third clause contains company names - most say "Intel Corporation", but there are instances for Nvidia, Samsung, Eideticom and even "the copyright holder". Used a bash script to automate replacement of the license text with SPDX identifier which is checked into scripts/spdx.sh. Signed-off-by: Jim Harris <james.r.harris@intel.com> Change-Id: Iaa88ab5e92ea471691dc298cfe41ebfb5d169780 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/12904 Community-CI: Broadcom CI <spdk-ci.pdl@broadcom.com> Community-CI: Mellanox Build Bot Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Aleksey Marchuk <alexeymar@nvidia.com> Reviewed-by: Changpeng Liu <changpeng.liu@intel.com> Reviewed-by: Dong Yi <dongx.yi@intel.com> Reviewed-by: Konrad Sztyber <konrad.sztyber@intel.com> Reviewed-by: Paul Luse <paul.e.luse@intel.com> Reviewed-by: <qun.wan@intel.com>
481 lines
11 KiB
C
481 lines
11 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (c) Intel Corporation.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
#include "spdk/stdinc.h"
|
|
|
|
#include "spdk_cunit.h"
|
|
#include "common/lib/test_env.c"
|
|
|
|
#include "ftl/ftl_reloc.c"
|
|
#include "../common/utils.c"
|
|
|
|
#define MAX_ACTIVE_RELOCS 5
|
|
#define MAX_RELOC_QDEPTH 31
|
|
|
|
struct base_bdev_geometry g_geo = {
|
|
.write_unit_size = 16,
|
|
.optimal_open_zones = 12,
|
|
.zone_size = 100,
|
|
.blockcnt = 1500 * 100 * 12,
|
|
};
|
|
|
|
DEFINE_STUB(ftl_dev_tail_md_disk_size, size_t, (const struct spdk_ftl_dev *dev), 1);
|
|
DEFINE_STUB(ftl_addr_is_written, bool, (struct ftl_band *band, struct ftl_addr addr), true);
|
|
DEFINE_STUB_V(ftl_band_set_state, (struct ftl_band *band, enum ftl_band_state state));
|
|
DEFINE_STUB_V(ftl_free_io, (struct ftl_io *io));
|
|
#if defined(DEBUG)
|
|
DEFINE_STUB_V(ftl_trace_lba_io_init, (struct spdk_ftl_dev *dev, const struct ftl_io *io));
|
|
#endif
|
|
|
|
int
|
|
ftl_band_alloc_lba_map(struct ftl_band *band)
|
|
{
|
|
struct spdk_ftl_dev *dev = band->dev;
|
|
|
|
ftl_band_acquire_lba_map(band);
|
|
band->lba_map.map = spdk_mempool_get(dev->lba_pool);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
ftl_band_release_lba_map(struct ftl_band *band)
|
|
{
|
|
struct spdk_ftl_dev *dev = band->dev;
|
|
|
|
band->lba_map.ref_cnt--;
|
|
spdk_mempool_put(dev->lba_pool, band->lba_map.map);
|
|
band->lba_map.map = NULL;
|
|
}
|
|
|
|
void
|
|
ftl_band_acquire_lba_map(struct ftl_band *band)
|
|
{
|
|
band->lba_map.ref_cnt++;
|
|
}
|
|
|
|
size_t
|
|
ftl_lba_map_num_blocks(const struct spdk_ftl_dev *dev)
|
|
{
|
|
return spdk_divide_round_up(ftl_get_num_blocks_in_band(dev) * sizeof(uint64_t), FTL_BLOCK_SIZE);
|
|
}
|
|
|
|
int
|
|
ftl_band_read_lba_map(struct ftl_band *band, size_t offset,
|
|
size_t num_blocks, ftl_io_fn fn, void *ctx)
|
|
{
|
|
fn(ctx, ctx, 0);
|
|
return 0;
|
|
}
|
|
|
|
uint64_t
|
|
ftl_band_block_offset_from_addr(struct ftl_band *band, struct ftl_addr addr)
|
|
{
|
|
return test_offset_from_addr(addr, band);
|
|
}
|
|
|
|
struct ftl_addr
|
|
ftl_band_addr_from_block_offset(struct ftl_band *band, uint64_t block_off)
|
|
{
|
|
struct ftl_addr addr = {};
|
|
|
|
addr.offset = block_off + band->id * ftl_get_num_blocks_in_band(band->dev);
|
|
return addr;
|
|
}
|
|
|
|
void
|
|
ftl_io_read(struct ftl_io *io)
|
|
{
|
|
io->cb_fn(io, io->cb_ctx, 0);
|
|
free(io);
|
|
}
|
|
|
|
void
|
|
ftl_io_write(struct ftl_io *io)
|
|
{
|
|
io->cb_fn(io, io->cb_ctx, 0);
|
|
free(io->lba.vector);
|
|
free(io);
|
|
}
|
|
|
|
struct ftl_io *
|
|
ftl_io_init_internal(const struct ftl_io_init_opts *opts)
|
|
{
|
|
struct ftl_io *io = opts->io;
|
|
|
|
if (!io) {
|
|
io = calloc(1, opts->size);
|
|
}
|
|
|
|
SPDK_CU_ASSERT_FATAL(io != NULL);
|
|
|
|
io->dev = opts->dev;
|
|
io->band = opts->band;
|
|
io->flags = opts->flags;
|
|
io->cb_fn = opts->cb_fn;
|
|
io->cb_ctx = io;
|
|
io->num_blocks = opts->num_blocks;
|
|
memcpy(&io->iov, &opts->iovs, sizeof(io->iov));
|
|
io->iov_cnt = opts->iovcnt;
|
|
|
|
if (opts->flags & FTL_IO_VECTOR_LBA) {
|
|
io->lba.vector = calloc(io->num_blocks, sizeof(uint64_t));
|
|
SPDK_CU_ASSERT_FATAL(io->lba.vector != NULL);
|
|
}
|
|
|
|
return io;
|
|
}
|
|
|
|
struct ftl_io *
|
|
ftl_io_alloc(struct spdk_io_channel *ch)
|
|
{
|
|
size_t io_size = sizeof(struct ftl_md_io);
|
|
|
|
return malloc(io_size);
|
|
}
|
|
|
|
void
|
|
ftl_io_reinit(struct ftl_io *io, ftl_io_fn fn, void *ctx, int flags, int type)
|
|
{
|
|
io->cb_fn = fn;
|
|
io->cb_ctx = ctx;
|
|
io->type = type;
|
|
}
|
|
|
|
static void
|
|
single_reloc_move(struct ftl_band_reloc *breloc)
|
|
{
|
|
/* Process read */
|
|
ftl_process_reloc(breloc);
|
|
/* Process lba map read */
|
|
ftl_process_reloc(breloc);
|
|
/* Process write */
|
|
ftl_process_reloc(breloc);
|
|
}
|
|
|
|
static void
|
|
add_to_active_queue(struct ftl_reloc *reloc, struct ftl_band_reloc *breloc)
|
|
{
|
|
TAILQ_REMOVE(&reloc->pending_queue, breloc, entry);
|
|
breloc->state = FTL_BAND_RELOC_STATE_ACTIVE;
|
|
TAILQ_INSERT_HEAD(&reloc->active_queue, breloc, entry);
|
|
}
|
|
|
|
static void
|
|
setup_reloc(struct spdk_ftl_dev **_dev, struct ftl_reloc **_reloc,
|
|
const struct base_bdev_geometry *geo)
|
|
{
|
|
size_t i;
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
|
|
dev = test_init_ftl_dev(geo);
|
|
|
|
dev->conf.max_active_relocs = MAX_ACTIVE_RELOCS;
|
|
dev->conf.max_reloc_qdepth = MAX_RELOC_QDEPTH;
|
|
|
|
SPDK_CU_ASSERT_FATAL(ftl_get_num_bands(dev) > 0);
|
|
|
|
for (i = 0; i < ftl_get_num_bands(dev); ++i) {
|
|
test_init_ftl_band(dev, i, geo->zone_size);
|
|
}
|
|
|
|
reloc = ftl_reloc_init(dev);
|
|
dev->reloc = reloc;
|
|
CU_ASSERT_PTR_NOT_NULL_FATAL(reloc);
|
|
ftl_reloc_resume(reloc);
|
|
|
|
*_dev = dev;
|
|
*_reloc = reloc;
|
|
}
|
|
|
|
static void
|
|
cleanup_reloc(struct spdk_ftl_dev *dev, struct ftl_reloc *reloc)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < ftl_get_num_bands(reloc->dev); ++i) {
|
|
SPDK_CU_ASSERT_FATAL(reloc->brelocs[i].state == FTL_BAND_RELOC_STATE_INACTIVE);
|
|
}
|
|
|
|
ftl_reloc_free(reloc);
|
|
|
|
for (i = 0; i < ftl_get_num_bands(dev); ++i) {
|
|
test_free_ftl_band(&dev->bands[i]);
|
|
}
|
|
test_free_ftl_dev(dev);
|
|
}
|
|
|
|
static void
|
|
set_band_valid_map(struct ftl_band *band, size_t offset, size_t num_blocks)
|
|
{
|
|
struct ftl_lba_map *lba_map = &band->lba_map;
|
|
size_t i;
|
|
|
|
SPDK_CU_ASSERT_FATAL(lba_map != NULL);
|
|
for (i = offset; i < offset + num_blocks; ++i) {
|
|
spdk_bit_array_set(lba_map->vld, i);
|
|
lba_map->num_vld++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_reloc_iter_full(void)
|
|
{
|
|
size_t num_blocks, num_iters, reminder, i;
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
struct ftl_addr addr;
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
g_geo.zone_size = 100;
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
|
|
set_band_valid_map(band, 0, ftl_get_num_blocks_in_band(dev));
|
|
|
|
ftl_reloc_add(reloc, band, 0, ftl_get_num_blocks_in_band(dev), 0, true);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, ftl_get_num_blocks_in_band(dev));
|
|
|
|
num_iters = ftl_get_num_punits(dev) *
|
|
(ftl_get_num_blocks_in_zone(dev) / reloc->xfer_size);
|
|
|
|
for (i = 0; i < num_iters; i++) {
|
|
num_blocks = ftl_reloc_next_blocks(breloc, &addr);
|
|
CU_ASSERT_EQUAL(num_blocks, reloc->xfer_size);
|
|
}
|
|
|
|
num_iters = ftl_get_num_punits(dev);
|
|
|
|
/* ftl_reloc_next_blocks is searching for maximum xfer_size */
|
|
/* contiguous valid logic blocks in zone, so we can end up */
|
|
/* with some reminder if number of logical blocks in zone */
|
|
/* is not divisible by xfer_size */
|
|
reminder = ftl_get_num_blocks_in_zone(dev) % reloc->xfer_size;
|
|
for (i = 0; i < num_iters; i++) {
|
|
num_blocks = ftl_reloc_next_blocks(breloc, &addr);
|
|
CU_ASSERT_EQUAL(reminder, num_blocks);
|
|
}
|
|
|
|
/* num_blocks should remain intact since all the blocks are valid */
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, ftl_get_num_blocks_in_band(dev));
|
|
breloc->state = FTL_BAND_RELOC_STATE_INACTIVE;
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
static void
|
|
test_reloc_empty_band(void)
|
|
{
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
|
|
ftl_reloc_add(reloc, band, 0, ftl_get_num_blocks_in_band(dev), 0, true);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 0);
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
static void
|
|
test_reloc_full_band(void)
|
|
{
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
size_t num_moves, num_iters, num_block, i;
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
num_moves = MAX_RELOC_QDEPTH * reloc->xfer_size;
|
|
num_iters = ftl_get_num_blocks_in_band(dev) / num_moves;
|
|
|
|
set_band_valid_map(band, 0, ftl_get_num_blocks_in_band(dev));
|
|
|
|
ftl_reloc_add(reloc, band, 0, ftl_get_num_blocks_in_band(dev), 0, true);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, ftl_get_num_blocks_in_band(dev));
|
|
|
|
ftl_reloc_prep(breloc);
|
|
add_to_active_queue(reloc, breloc);
|
|
|
|
for (i = 1; i <= num_iters; ++i) {
|
|
single_reloc_move(breloc);
|
|
num_block = ftl_get_num_blocks_in_band(dev) - (i * num_moves);
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, num_block);
|
|
|
|
}
|
|
|
|
/* Process reminder blocks */
|
|
single_reloc_move(breloc);
|
|
/* Drain move queue */
|
|
ftl_reloc_process_moves(breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 0);
|
|
CU_ASSERT_TRUE(ftl_reloc_done(breloc));
|
|
ftl_reloc_release(breloc);
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
static void
|
|
test_reloc_scatter_band(void)
|
|
{
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
size_t num_iters, i;
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
num_iters = spdk_divide_round_up(ftl_get_num_blocks_in_band(dev), MAX_RELOC_QDEPTH * 2);
|
|
|
|
for (i = 0; i < ftl_get_num_blocks_in_band(dev); ++i) {
|
|
if (i % 2) {
|
|
set_band_valid_map(band, i, 1);
|
|
}
|
|
}
|
|
|
|
ftl_reloc_add(reloc, band, 0, ftl_get_num_blocks_in_band(dev), 0, true);
|
|
ftl_reloc_prep(breloc);
|
|
add_to_active_queue(reloc, breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, ftl_get_num_blocks_in_band(dev));
|
|
|
|
for (i = 0; i < num_iters ; ++i) {
|
|
single_reloc_move(breloc);
|
|
}
|
|
|
|
ftl_process_reloc(breloc);
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 0);
|
|
CU_ASSERT_TRUE(ftl_reloc_done(breloc));
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
static void
|
|
test_reloc_zone(void)
|
|
{
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
size_t num_io, num_iters, num_block, i;
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
/* High priority band have allocated lba map */
|
|
band->high_prio = 1;
|
|
ftl_band_alloc_lba_map(band);
|
|
num_io = MAX_RELOC_QDEPTH * reloc->xfer_size;
|
|
num_iters = ftl_get_num_blocks_in_zone(dev) / num_io;
|
|
|
|
set_band_valid_map(band, 0, ftl_get_num_blocks_in_band(dev));
|
|
|
|
ftl_reloc_add(reloc, band, ftl_get_num_blocks_in_zone(dev) * 3,
|
|
ftl_get_num_blocks_in_zone(dev), 1, false);
|
|
add_to_active_queue(reloc, breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, ftl_get_num_blocks_in_zone(dev));
|
|
|
|
for (i = 1; i <= num_iters ; ++i) {
|
|
single_reloc_move(breloc);
|
|
num_block = ftl_get_num_blocks_in_zone(dev) - (i * num_io);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, num_block);
|
|
}
|
|
|
|
/* In case num_blocks_in_zone % num_io != 0 one extra iteration is needed */
|
|
single_reloc_move(breloc);
|
|
/* Drain move queue */
|
|
ftl_reloc_process_moves(breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 0);
|
|
CU_ASSERT_TRUE(ftl_reloc_done(breloc));
|
|
ftl_reloc_release(breloc);
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
static void
|
|
test_reloc_single_block(void)
|
|
{
|
|
struct spdk_ftl_dev *dev;
|
|
struct ftl_reloc *reloc;
|
|
struct ftl_band_reloc *breloc;
|
|
struct ftl_band *band;
|
|
#define TEST_RELOC_OFFSET 6
|
|
|
|
setup_reloc(&dev, &reloc, &g_geo);
|
|
|
|
breloc = &reloc->brelocs[0];
|
|
band = breloc->band;
|
|
|
|
set_band_valid_map(band, TEST_RELOC_OFFSET, 1);
|
|
|
|
ftl_reloc_add(reloc, band, TEST_RELOC_OFFSET, 1, 0, false);
|
|
SPDK_CU_ASSERT_FATAL(breloc == TAILQ_FIRST(&reloc->pending_queue));
|
|
ftl_reloc_prep(breloc);
|
|
add_to_active_queue(reloc, breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 1);
|
|
|
|
single_reloc_move(breloc);
|
|
/* Drain move queue */
|
|
ftl_reloc_process_moves(breloc);
|
|
|
|
CU_ASSERT_EQUAL(breloc->num_blocks, 0);
|
|
CU_ASSERT_TRUE(ftl_reloc_done(breloc));
|
|
ftl_reloc_release(breloc);
|
|
|
|
cleanup_reloc(dev, reloc);
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
CU_pSuite suite = NULL;
|
|
unsigned int num_failures;
|
|
|
|
CU_set_error_action(CUEA_ABORT);
|
|
CU_initialize_registry();
|
|
|
|
suite = CU_add_suite("ftl_band_suite", NULL, NULL);
|
|
|
|
|
|
CU_ADD_TEST(suite, test_reloc_iter_full);
|
|
CU_ADD_TEST(suite, test_reloc_empty_band);
|
|
CU_ADD_TEST(suite, test_reloc_full_band);
|
|
CU_ADD_TEST(suite, test_reloc_scatter_band);
|
|
CU_ADD_TEST(suite, test_reloc_zone);
|
|
CU_ADD_TEST(suite, test_reloc_single_block);
|
|
|
|
CU_basic_set_mode(CU_BRM_VERBOSE);
|
|
CU_basic_run_tests();
|
|
num_failures = CU_get_number_of_failures();
|
|
CU_cleanup_registry();
|
|
|
|
return num_failures;
|
|
}
|