/* SPDX-License-Identifier: BSD-3-Clause * Copyright (c) Intel Corporation. * All rights reserved. */ #include "spdk/likely.h" #include "spdk/stdinc.h" #include "spdk/nvme.h" #include "spdk/thread.h" #include "spdk/bdev_module.h" #include "spdk/string.h" #include "spdk/ftl.h" #include "spdk/crc32.h" #include "ftl_core.h" #include "ftl_io.h" #include "ftl_debug.h" #include "ftl_internal.h" #include "mngt/ftl_mngt.h" #include "utils/ftl_mempool.h" size_t spdk_ftl_io_size(void) { return sizeof(struct ftl_io); } static bool ftl_shutdown_complete(struct spdk_ftl_dev *dev) { if (dev->num_inflight) { return false; } if (!ftl_l2p_is_halted(dev)) { ftl_l2p_halt(dev); return false; } return true; } void ftl_invalidate_addr(struct spdk_ftl_dev *dev, ftl_addr addr) { if (ftl_addr_in_nvc(dev, addr)) { return; } } void spdk_ftl_dev_get_attrs(const struct spdk_ftl_dev *dev, struct spdk_ftl_attrs *attrs) { attrs->num_blocks = dev->num_lbas; attrs->block_size = FTL_BLOCK_SIZE; attrs->num_zones = ftl_get_num_zones(dev); attrs->zone_size = ftl_get_num_blocks_in_zone(dev); attrs->optimum_io_size = dev->xfer_size; } static void start_io(struct ftl_io *io) { struct ftl_io_channel *ioch = ftl_io_channel_get_ctx(io->ioch); io->map = ftl_mempool_get(ioch->map_pool); if (spdk_unlikely(!io->map)) { io->status = -ENOMEM; ftl_io_complete(io); return; } switch (io->type) { case FTL_IO_READ: case FTL_IO_WRITE: case FTL_IO_UNMAP: default: io->status = -EOPNOTSUPP; ftl_io_complete(io); } } #define FTL_IO_QUEUE_BATCH 16 int ftl_io_channel_poll(void *arg) { struct ftl_io_channel *ch = arg; void *ios[FTL_IO_QUEUE_BATCH]; uint64_t i, count; count = spdk_ring_dequeue(ch->cq, ios, FTL_IO_QUEUE_BATCH); if (count == 0) { return SPDK_POLLER_IDLE; } for (i = 0; i < count; i++) { struct ftl_io *io = ios[i]; io->user_fn(io->cb_ctx, io->status); } return SPDK_POLLER_BUSY; } static void ftl_process_io_channel(struct spdk_ftl_dev *dev, struct ftl_io_channel *ioch) { void *ios[FTL_IO_QUEUE_BATCH]; size_t count, i; count = spdk_ring_dequeue(ioch->sq, ios, FTL_IO_QUEUE_BATCH); if (count == 0) { return; } for (i = 0; i < count; i++) { struct ftl_io *io = ios[i]; start_io(io); } } static void ftl_process_io_queue(struct spdk_ftl_dev *dev) { struct ftl_io_channel *ioch; TAILQ_FOREACH(ioch, &dev->ioch_queue, entry) { ftl_process_io_channel(dev, ioch); } } int ftl_core_poller(void *ctx) { struct spdk_ftl_dev *dev = ctx; uint64_t io_activity_total_old = dev->io_activity_total; if (dev->halt && ftl_shutdown_complete(dev)) { spdk_poller_unregister(&dev->core_poller); return SPDK_POLLER_IDLE; } ftl_process_io_queue(dev); ftl_l2p_process(dev); if (io_activity_total_old != dev->io_activity_total) { return SPDK_POLLER_BUSY; } return SPDK_POLLER_IDLE; } void *g_ftl_write_buf; void *g_ftl_read_buf; int spdk_ftl_init(void) { g_ftl_write_buf = spdk_zmalloc(FTL_ZERO_BUFFER_SIZE, FTL_ZERO_BUFFER_SIZE, NULL, SPDK_ENV_LCORE_ID_ANY, SPDK_MALLOC_DMA); if (!g_ftl_write_buf) { return -ENOMEM; } g_ftl_read_buf = spdk_zmalloc(FTL_ZERO_BUFFER_SIZE, FTL_ZERO_BUFFER_SIZE, NULL, SPDK_ENV_LCORE_ID_ANY, SPDK_MALLOC_DMA); if (!g_ftl_read_buf) { spdk_free(g_ftl_write_buf); g_ftl_write_buf = NULL; return -ENOMEM; } return 0; } void spdk_ftl_fini(void) { spdk_free(g_ftl_write_buf); spdk_free(g_ftl_read_buf); } struct spdk_io_channel * spdk_ftl_get_io_channel(struct spdk_ftl_dev *dev) { return spdk_get_io_channel(dev); } SPDK_LOG_REGISTER_COMPONENT(ftl_core)