mirror of
https://github.com/ceph/ceph
synced 2025-01-19 09:32:00 +00:00
test: use randomized write sizes for rbd-mirror stress test
Signed-off-by: Jason Dillaman <dillaman@redhat.com>
This commit is contained in:
parent
d7b511b854
commit
b9edff2174
@ -324,9 +324,10 @@ write_image()
|
||||
local image=$2
|
||||
local duration=$(($RANDOM % 35 + 15))
|
||||
|
||||
timeout ${duration}s rbd --cluster ${cluster} -p ${POOL} bench-write \
|
||||
${image} --io-size 4096 --io-threads 8 --io-total 10G --io-pattern rand \
|
||||
--debug-rbd=20 --debug-journaler=20 2> ${TEMPDIR}/rbd-bench-write.log || true
|
||||
timeout ${duration}s ceph_test_rbd_mirror_random_write \
|
||||
--cluster ${cluster} ${POOL} ${image} \
|
||||
--debug-rbd=20 --debug-journaler=20 \
|
||||
2> ${TEMPDIR}/rbd-mirror-random-write.log || true
|
||||
}
|
||||
|
||||
create_snap()
|
||||
@ -346,7 +347,7 @@ wait_for_snap()
|
||||
local snap_name=$3
|
||||
local s
|
||||
|
||||
for s in 1 2 4 8 8 8 8 8 8 8 8 16 16 16 16 32 32; do
|
||||
for s in 1 2 4 8 8 8 8 8 8 8 8 16 16 16 16 32 32 32 32; do
|
||||
sleep ${s}
|
||||
rbd --cluster ${cluster} -p ${POOL} info ${image}@${snap_name} || continue
|
||||
return 0
|
||||
|
@ -524,6 +524,13 @@ ceph_test_rbd_mirror_LDADD = \
|
||||
$(CEPH_GLOBAL) $(RADOS_TEST_LDADD)
|
||||
bin_DEBUGPROGRAMS += ceph_test_rbd_mirror
|
||||
|
||||
ceph_test_rbd_mirror_random_write_SOURCES = \
|
||||
test/rbd_mirror/random_write.cc
|
||||
ceph_test_rbd_mirror_random_write_CXXFLAGS = $(UNITTEST_CXXFLAGS)
|
||||
ceph_test_rbd_mirror_random_write_LDADD = \
|
||||
$(LIBRBD) $(LIBRADOS) $(CEPH_GLOBAL)
|
||||
bin_DEBUGPROGRAMS += ceph_test_rbd_mirror_random_write
|
||||
|
||||
ceph_test_rbd_mirror_image_replay_SOURCES = \
|
||||
test/rbd_mirror/image_replay.cc
|
||||
ceph_test_rbd_mirror_image_replay_LDADD = \
|
||||
|
@ -67,3 +67,12 @@ target_link_libraries(ceph_test_rbd_mirror
|
||||
${UNITTEST_LIBS}
|
||||
)
|
||||
|
||||
add_executable(ceph_test_rbd_mirror_random_write
|
||||
random_write.cc
|
||||
$<TARGET_OBJECTS:parse_secret_objs>)
|
||||
set_target_properties(ceph_test_rbd_mirror_random_write PROPERTIES COMPILE_FLAGS
|
||||
${UNITTEST_CXX_FLAGS})
|
||||
target_link_libraries(ceph_test_rbd_mirror_random_write
|
||||
librbd librados global keyutils
|
||||
)
|
||||
|
||||
|
214
src/test/rbd_mirror/random_write.cc
Normal file
214
src/test/rbd_mirror/random_write.cc
Normal file
@ -0,0 +1,214 @@
|
||||
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
|
||||
// vim: ts=8 sw=2 smarttab
|
||||
|
||||
#include "common/ceph_argparse.h"
|
||||
#include "common/config.h"
|
||||
#include "common/debug.h"
|
||||
#include "common/errno.h"
|
||||
#include "common/Cond.h"
|
||||
#include "include/rados/librados.hpp"
|
||||
#include "include/rbd/librbd.hpp"
|
||||
#include "global/global_init.h"
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
#define dout_subsys ceph_subsys_rbd_mirror
|
||||
#undef dout_prefix
|
||||
#define dout_prefix *_dout << "random-write: "
|
||||
|
||||
namespace {
|
||||
|
||||
const uint32_t NUM_THREADS = 8;
|
||||
const uint32_t MAX_IO_SIZE = 24576;
|
||||
const uint32_t MIN_IO_SIZE = 4;
|
||||
|
||||
void usage() {
|
||||
std::cout << "usage: ceph_test_rbd_mirror_random_write [options...] \\" << std::endl;
|
||||
std::cout << " <pool> <image>" << std::endl;
|
||||
std::cout << std::endl;
|
||||
std::cout << " pool image pool" << std::endl;
|
||||
std::cout << " image image to write" << std::endl;
|
||||
std::cout << std::endl;
|
||||
std::cout << "options:\n";
|
||||
std::cout << " -m monaddress[:port] connect to specified monitor\n";
|
||||
std::cout << " --keyring=<path> path to keyring for local cluster\n";
|
||||
std::cout << " --log-file=<logfile> file to log debug output\n";
|
||||
std::cout << " --debug-rbd-mirror=<log-level>/<memory-level> set rbd-mirror debug level\n";
|
||||
generic_server_usage();
|
||||
}
|
||||
|
||||
void rbd_bencher_completion(void *c, void *pc);
|
||||
|
||||
struct rbd_bencher {
|
||||
librbd::Image *image;
|
||||
Mutex lock;
|
||||
Cond cond;
|
||||
int in_flight;
|
||||
|
||||
explicit rbd_bencher(librbd::Image *i)
|
||||
: image(i),
|
||||
lock("rbd_bencher::lock"),
|
||||
in_flight(0) {
|
||||
}
|
||||
|
||||
bool start_write(int max, uint64_t off, uint64_t len, bufferlist& bl,
|
||||
int op_flags) {
|
||||
{
|
||||
Mutex::Locker l(lock);
|
||||
if (in_flight >= max)
|
||||
return false;
|
||||
in_flight++;
|
||||
}
|
||||
librbd::RBD::AioCompletion *c =
|
||||
new librbd::RBD::AioCompletion((void *)this, rbd_bencher_completion);
|
||||
image->aio_write2(off, len, bl, c, op_flags);
|
||||
//cout << "start " << c << " at " << off << "~" << len << std::endl;
|
||||
return true;
|
||||
}
|
||||
|
||||
void wait_for(int max) {
|
||||
Mutex::Locker l(lock);
|
||||
while (in_flight > max) {
|
||||
utime_t dur;
|
||||
dur.set_from_double(.2);
|
||||
cond.WaitInterval(g_ceph_context, lock, dur);
|
||||
}
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
void rbd_bencher_completion(void *vc, void *pc) {
|
||||
librbd::RBD::AioCompletion *c = (librbd::RBD::AioCompletion *)vc;
|
||||
rbd_bencher *b = static_cast<rbd_bencher *>(pc);
|
||||
//cout << "complete " << c << std::endl;
|
||||
int ret = c->get_return_value();
|
||||
if (ret != 0) {
|
||||
cout << "write error: " << cpp_strerror(ret) << std::endl;
|
||||
exit(ret < 0 ? -ret : ret);
|
||||
}
|
||||
b->lock.Lock();
|
||||
b->in_flight--;
|
||||
b->cond.Signal();
|
||||
b->lock.Unlock();
|
||||
c->release();
|
||||
}
|
||||
|
||||
void write_image(librbd::Image &image) {
|
||||
srand(time(NULL) % (unsigned long) -1);
|
||||
|
||||
uint64_t max_io_bytes = MAX_IO_SIZE * 1024;
|
||||
bufferptr bp(max_io_bytes);
|
||||
memset(bp.c_str(), rand() & 0xff, bp.length());
|
||||
bufferlist bl;
|
||||
bl.push_back(bp);
|
||||
|
||||
uint64_t size = 0;
|
||||
image.size(&size);
|
||||
assert(size != 0);
|
||||
|
||||
vector<uint64_t> thread_offset;
|
||||
uint64_t i;
|
||||
uint64_t start_pos;
|
||||
|
||||
// disturb all thread's offset, used by seq write
|
||||
for (i = 0; i < NUM_THREADS; i++) {
|
||||
start_pos = (rand() % (size / max_io_bytes)) * max_io_bytes;
|
||||
thread_offset.push_back(start_pos);
|
||||
}
|
||||
|
||||
uint64_t total_ios = 0;
|
||||
uint64_t total_bytes = 0;
|
||||
rbd_bencher b(&image);
|
||||
while (true) {
|
||||
b.wait_for(NUM_THREADS - 1);
|
||||
for (uint32_t i = 0; i < NUM_THREADS; ++i) {
|
||||
// mostly small writes with a small chance of large writes
|
||||
uint32_t io_modulo = MIN_IO_SIZE + 1;
|
||||
if (rand() % 30 == 0) {
|
||||
io_modulo += MAX_IO_SIZE;
|
||||
}
|
||||
|
||||
uint32_t io_size = (((rand() % io_modulo) + MIN_IO_SIZE) * 1024);
|
||||
thread_offset[i] = (rand() % (size / io_size)) * io_size;
|
||||
if (!b.start_write(NUM_THREADS, thread_offset[i], io_size, bl,
|
||||
LIBRADOS_OP_FLAG_FADVISE_RANDOM)) {
|
||||
break;
|
||||
}
|
||||
++i;
|
||||
|
||||
++total_ios;
|
||||
total_bytes += io_size;
|
||||
if (total_ios % 100 == 0) {
|
||||
std::cout << total_ios << " IOs, " << total_bytes << " bytes"
|
||||
<< std::endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
b.wait_for(0);
|
||||
}
|
||||
|
||||
} // anonymous namespace
|
||||
|
||||
int main(int argc, const char **argv)
|
||||
{
|
||||
std::vector<const char*> args;
|
||||
argv_to_vec(argc, argv, args);
|
||||
env_to_vec(args);
|
||||
|
||||
global_init(NULL, args, CEPH_ENTITY_TYPE_CLIENT, CODE_ENVIRONMENT_UTILITY, 0);
|
||||
|
||||
for (auto i = args.begin(); i != args.end(); ++i) {
|
||||
if (ceph_argparse_flag(args, i, "-h", "--help", (char*)NULL)) {
|
||||
usage();
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
}
|
||||
|
||||
if (args.size() < 2) {
|
||||
usage();
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
std::string pool_name = args[0];
|
||||
std::string image_name = args[1];
|
||||
|
||||
common_init_finish(g_ceph_context);
|
||||
|
||||
dout(5) << "connecting to cluster" << dendl;
|
||||
librados::Rados rados;
|
||||
librados::IoCtx io_ctx;
|
||||
librbd::RBD rbd;
|
||||
librbd::Image image;
|
||||
int r = rados.init_with_context(g_ceph_context);
|
||||
if (r < 0) {
|
||||
derr << "could not initialize RADOS handle" << dendl;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
r = rados.connect();
|
||||
if (r < 0) {
|
||||
derr << "error connecting to local cluster" << dendl;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
r = rados.ioctx_create(pool_name.c_str(), io_ctx);
|
||||
if (r < 0) {
|
||||
derr << "error finding local pool " << pool_name << ": "
|
||||
<< cpp_strerror(r) << dendl;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
r = rbd.open(io_ctx, image, image_name.c_str());
|
||||
if (r < 0) {
|
||||
derr << "error opening image " << image_name << ": "
|
||||
<< cpp_strerror(r) << dendl;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
write_image(image);
|
||||
|
||||
cleanup:
|
||||
g_ceph_context->put();
|
||||
|
||||
return r < 0 ? EXIT_SUCCESS : EXIT_FAILURE;
|
||||
}
|
Loading…
Reference in New Issue
Block a user