|
- // Copyright (c) Vitaliy Filippov, 2019+
- // License: VNPL-1.0 (see README.md for details)
-
- // FIO engine to test Blockstore
- //
- // Initialize storage for tests:
- //
- // dd if=/dev/zero of=test_data.bin bs=1024 count=1048576
- // dd if=/dev/zero of=test_meta.bin bs=1024 count=256
- // dd if=/dev/zero of=test_journal.bin bs=1024 count=4096
- //
- // Random write:
- //
- // fio -thread -ioengine=./libfio_blockstore.so -name=test -bs=4k -direct=1 -fsync=16 -iodepth=16 -rw=randwrite \
- // -bs_config='{"data_device":"./test_data.bin"}' -size=1000M
- //
- // Linear write:
- //
- // fio -thread -ioengine=./libfio_blockstore.so -name=test -bs=128k -direct=1 -fsync=32 -iodepth=32 -rw=write \
- // -bs_config='{"data_device":"./test_data.bin"}' -size=1000M
- //
- // Random read (run with -iodepth=32 or -iodepth=1):
- //
- // fio -thread -ioengine=./libfio_blockstore.so -name=test -bs=4k -direct=1 -iodepth=32 -rw=randread \
- // -bs_config='{"data_device":"./test_data.bin"}' -size=1000M
-
- #include "blockstore.h"
- #include "fio_headers.h"
-
- #include "json11/json11.hpp"
-
- struct bs_data
- {
- blockstore_t *bs;
- ring_loop_t *ringloop;
- /* The list of completed io_u structs. */
- std::vector<io_u*> completed;
- int op_n = 0, inflight = 0;
- bool last_sync = false;
- };
-
- struct bs_options
- {
- int __pad;
- char *json_config = NULL;
- };
-
- static struct fio_option options[] = {
- {
- .name = "bs_config",
- .lname = "JSON config for Blockstore",
- .type = FIO_OPT_STR_STORE,
- .off1 = offsetof(struct bs_options, json_config),
- .help = "JSON config for Blockstore",
- .category = FIO_OPT_C_ENGINE,
- .group = FIO_OPT_G_FILENAME,
- },
- {
- .name = NULL,
- },
- };
-
- static int bs_setup(struct thread_data *td)
- {
- bs_data *bsd;
- //fio_file *f;
- //int r;
- //int64_t size;
-
- bsd = new bs_data;
- if (!bsd)
- {
- td_verror(td, errno, "calloc");
- return 1;
- }
- td->io_ops_data = bsd;
-
- if (!td->files_index)
- {
- add_file(td, "blockstore", 0, 0);
- td->o.nr_files = td->o.nr_files ? : 1;
- td->o.open_files++;
- }
-
- //f = td->files[0];
- //f->real_file_size = size;
- return 0;
- }
-
- static void bs_cleanup(struct thread_data *td)
- {
- bs_data *bsd = (bs_data*)td->io_ops_data;
- if (bsd)
- {
- while (1)
- {
- do
- {
- bsd->ringloop->loop();
- if (bsd->bs->is_safe_to_stop())
- goto safe;
- } while (bsd->ringloop->has_work());
- bsd->ringloop->wait();
- }
- safe:
- delete bsd->bs;
- delete bsd->ringloop;
- delete bsd;
- }
- }
-
- /* Connect to the server from each thread. */
- static int bs_init(struct thread_data *td)
- {
- bs_options *o = (bs_options*)td->eo;
- bs_data *bsd = (bs_data*)td->io_ops_data;
-
- blockstore_config_t config;
- if (o->json_config)
- {
- std::string json_err;
- auto json_cfg = json11::Json::parse(o->json_config, json_err);
- for (auto p: json_cfg.object_items())
- {
- if (p.second.is_string())
- config[p.first] = p.second.string_value();
- else
- config[p.first] = p.second.dump();
- }
- }
- bsd->ringloop = new ring_loop_t(512);
- bsd->bs = new blockstore_t(config, bsd->ringloop);
- while (1)
- {
- bsd->ringloop->loop();
- if (bsd->bs->is_started())
- break;
- bsd->ringloop->wait();
- }
-
- log_info("fio: blockstore initialized\n");
- return 0;
- }
-
- /* Begin read or write request. */
- static enum fio_q_status bs_queue(struct thread_data *td, struct io_u *io)
- {
- bs_data *bsd = (bs_data*)td->io_ops_data;
- int n = bsd->op_n;
- if (io->ddir == DDIR_SYNC && bsd->last_sync)
- {
- return FIO_Q_COMPLETED;
- }
-
- fio_ro_check(td, io);
-
- io->engine_data = bsd;
-
- if (io->ddir == DDIR_WRITE || io->ddir == DDIR_READ)
- assert(io->xfer_buflen <= bsd->bs->get_block_size());
-
- blockstore_op_t *op = new blockstore_op_t;
- op->callback = NULL;
-
- switch (io->ddir)
- {
- case DDIR_READ:
- op->opcode = BS_OP_READ;
- op->buf = io->xfer_buf;
- op->oid = {
- .inode = 1,
- .stripe = io->offset / bsd->bs->get_block_size(),
- };
- op->version = UINT64_MAX; // last unstable
- op->offset = io->offset % bsd->bs->get_block_size();
- op->len = io->xfer_buflen;
- op->callback = [io, n](blockstore_op_t *op)
- {
- io->error = op->retval < 0 ? -op->retval : 0;
- bs_data *bsd = (bs_data*)io->engine_data;
- bsd->inflight--;
- bsd->completed.push_back(io);
- #ifdef BLOCKSTORE_DEBUG
- printf("--- OP_READ %llx n=%d retval=%d\n", io, n, op->retval);
- #endif
- delete op;
- };
- break;
- case DDIR_WRITE:
- op->opcode = BS_OP_WRITE;
- op->buf = io->xfer_buf;
- op->oid = {
- .inode = 1,
- .stripe = io->offset / bsd->bs->get_block_size(),
- };
- op->version = 0; // assign automatically
- op->offset = io->offset % bsd->bs->get_block_size();
- op->len = io->xfer_buflen;
- op->callback = [io, n](blockstore_op_t *op)
- {
- io->error = op->retval < 0 ? -op->retval : 0;
- bs_data *bsd = (bs_data*)io->engine_data;
- bsd->inflight--;
- bsd->completed.push_back(io);
- #ifdef BLOCKSTORE_DEBUG
- printf("--- OP_WRITE %llx n=%d retval=%d\n", io, n, op->retval);
- #endif
- delete op;
- };
- bsd->last_sync = false;
- break;
- case DDIR_SYNC:
- op->opcode = BS_OP_SYNC_STAB_ALL;
- op->callback = [io, n](blockstore_op_t *op)
- {
- bs_data *bsd = (bs_data*)io->engine_data;
- io->error = op->retval < 0 ? -op->retval : 0;
- bsd->completed.push_back(io);
- bsd->inflight--;
- #ifdef BLOCKSTORE_DEBUG
- printf("--- OP_SYNC %llx n=%d retval=%d\n", io, n, op->retval);
- #endif
- delete op;
- };
- bsd->last_sync = true;
- break;
- default:
- io->error = EINVAL;
- return FIO_Q_COMPLETED;
- }
-
- #ifdef BLOCKSTORE_DEBUG
- printf("+++ %s %llx n=%d\n", op->opcode == OP_READ ? "OP_READ" : (op->opcode == OP_WRITE ? "OP_WRITE" : "OP_SYNC"), io, n);
- #endif
- io->error = 0;
- bsd->inflight++;
- bsd->bs->enqueue_op(op);
- bsd->op_n++;
-
- if (io->error != 0)
- return FIO_Q_COMPLETED;
- return FIO_Q_QUEUED;
- }
-
- static int bs_getevents(struct thread_data *td, unsigned int min, unsigned int max, const struct timespec *t)
- {
- bs_data *bsd = (bs_data*)td->io_ops_data;
- // FIXME timeout
- while (true)
- {
- bsd->ringloop->loop();
- if (bsd->completed.size() >= min)
- break;
- bsd->ringloop->wait();
- }
- return bsd->completed.size();
- }
-
- static struct io_u *bs_event(struct thread_data *td, int event)
- {
- bs_data *bsd = (bs_data*)td->io_ops_data;
- if (bsd->completed.size() == 0)
- return NULL;
- /* FIXME We ignore the event number and assume fio calls us exactly once for [0..nr_events-1] */
- struct io_u *ev = bsd->completed.back();
- bsd->completed.pop_back();
- return ev;
- }
-
- static int bs_io_u_init(struct thread_data *td, struct io_u *io)
- {
- io->engine_data = NULL;
- return 0;
- }
-
- static void bs_io_u_free(struct thread_data *td, struct io_u *io)
- {
- }
-
- static int bs_open_file(struct thread_data *td, struct fio_file *f)
- {
- return 0;
- }
-
- static int bs_invalidate(struct thread_data *td, struct fio_file *f)
- {
- return 0;
- }
-
- struct ioengine_ops ioengine = {
- .name = "vitastor_blockstore",
- .version = FIO_IOOPS_VERSION,
- .flags = FIO_MEMALIGN | FIO_DISKLESSIO | FIO_NOEXTEND,
- .setup = bs_setup,
- .init = bs_init,
- .queue = bs_queue,
- .getevents = bs_getevents,
- .event = bs_event,
- .cleanup = bs_cleanup,
- .open_file = bs_open_file,
- .invalidate = bs_invalidate,
- .io_u_init = bs_io_u_init,
- .io_u_free = bs_io_u_free,
- .option_struct_size = sizeof(struct bs_options),
- .options = options,
- };
-
- static void fio_init fio_bs_register(void)
- {
- register_ioengine(&ioengine);
- }
-
- static void fio_exit fio_bs_unregister(void)
- {
- unregister_ioengine(&ioengine);
- }
|