diff --git a/src/librbd/journal/Replay.cc b/src/librbd/journal/Replay.cc index b65d51c39c1..501fb3d97b1 100644 --- a/src/librbd/journal/Replay.cc +++ b/src/librbd/journal/Replay.cc @@ -54,6 +54,25 @@ struct ExecuteOp : public Context { } }; +template <typename I> +struct C_RefreshIfRequired : public Context { + I &image_ctx; + Context *on_finish; + + C_RefreshIfRequired(I &image_ctx, Context *on_finish) + : image_ctx(image_ctx), on_finish(on_finish) { + } + + virtual void finish(int r) override { + if (image_ctx.state->is_refresh_required()) { + image_ctx.state->refresh(on_finish); + return; + } + + on_finish->complete(0); + } +}; + } // anonymous namespace template <typename I> @@ -323,9 +342,10 @@ void Replay<I>::handle_event(const journal::SnapCreateEvent &event, op_event->ignore_error_codes = {-EEXIST}; // avoid lock cycles - m_image_ctx.op_work_queue->queue( - new ExecuteOp<I, journal::SnapCreateEvent>(m_image_ctx, event, - on_op_complete), 0); + m_image_ctx.op_work_queue->queue(new C_RefreshIfRequired<I>( + m_image_ctx, new ExecuteOp<I, journal::SnapCreateEvent>(m_image_ctx, event, + on_op_complete)), + 0); // do not process more events until the state machine is ready // since it will affect IO @@ -343,11 +363,12 @@ void Replay<I>::handle_event(const journal::SnapRemoveEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->snap_remove(event.snap_name.c_str(), - on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->snap_remove(event.snap_name.c_str(), + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-ENOENT}; @@ -365,12 +386,13 @@ void Replay<I>::handle_event(const journal::SnapRenameEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->snap_rename(event.snap_id, - event.snap_name.c_str(), - on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->snap_rename(event.snap_id, + event.snap_name.c_str(), + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-EEXIST}; @@ -388,11 +410,12 @@ void Replay<I>::handle_event(const journal::SnapProtectEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->snap_protect(event.snap_name.c_str(), - on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->snap_protect(event.snap_name.c_str(), + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-EBUSY}; @@ -411,11 +434,12 @@ void Replay<I>::handle_event(const journal::SnapUnprotectEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->snap_unprotect(event.snap_name.c_str(), - on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->snap_unprotect(event.snap_name.c_str(), + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-EINVAL}; @@ -434,12 +458,13 @@ void Replay<I>::handle_event(const journal::SnapRollbackEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->snap_rollback(event.snap_name.c_str(), - no_op_progress_callback, - on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->snap_rollback(event.snap_name.c_str(), + no_op_progress_callback, + on_op_complete); + })); on_ready->complete(0); } @@ -454,10 +479,12 @@ void Replay<I>::handle_event(const journal::RenameEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->rename(event.image_name.c_str(), on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->rename(event.image_name.c_str(), + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-EEXIST}; @@ -477,9 +504,9 @@ void Replay<I>::handle_event(const journal::ResizeEvent &event, &op_event); // avoid lock cycles - m_image_ctx.op_work_queue->queue( - new ExecuteOp<I, journal::ResizeEvent>(m_image_ctx, event, - on_op_complete), 0); + m_image_ctx.op_work_queue->queue(new C_RefreshIfRequired<I>( + m_image_ctx, new ExecuteOp<I, journal::ResizeEvent>(m_image_ctx, event, + on_op_complete)), 0); // do not process more events until the state machine is ready // since it will affect IO @@ -497,10 +524,12 @@ void Replay<I>::handle_event(const journal::FlattenEvent &event, OpEvent *op_event; Context *on_op_complete = create_op_context_callback(event.op_tid, on_safe, &op_event); - op_event->on_op_finish_event = new FunctionContext( - [this, event, on_op_complete](int r) { - m_image_ctx.operations->flatten(no_op_progress_callback, on_op_complete); - }); + op_event->on_op_finish_event = new C_RefreshIfRequired<I>( + m_image_ctx, new FunctionContext( + [this, event, on_op_complete](int r) { + m_image_ctx.operations->flatten(no_op_progress_callback, + on_op_complete); + })); // ignore errors caused due to replay op_event->ignore_error_codes = {-EINVAL}; diff --git a/src/test/Makefile-client.am b/src/test/Makefile-client.am index 987e773d04e..2b0e804e526 100644 --- a/src/test/Makefile-client.am +++ b/src/test/Makefile-client.am @@ -425,6 +425,7 @@ noinst_HEADERS += \ test/librbd/mock/MockContextWQ.h \ test/librbd/mock/MockExclusiveLock.h \ test/librbd/mock/MockImageCtx.h \ + test/librbd/mock/MockImageState.h \ test/librbd/mock/MockImageWatcher.h \ test/librbd/mock/MockJournal.h \ test/librbd/mock/MockObjectMap.h \ diff --git a/src/test/librbd/journal/test_mock_Replay.cc b/src/test/librbd/journal/test_mock_Replay.cc index 539599538cf..ebb3123490d 100644 --- a/src/test/librbd/journal/test_mock_Replay.cc +++ b/src/test/librbd/journal/test_mock_Replay.cc @@ -188,6 +188,16 @@ public: NotifyInvoke(&m_invoke_lock, &m_invoke_cond))); } + void expect_refresh_image(MockReplayImageCtx &mock_image_ctx, bool required, + int r) { + EXPECT_CALL(*mock_image_ctx.state, is_refresh_required()) + .WillOnce(Return(required)); + if (required) { + EXPECT_CALL(*mock_image_ctx.state, refresh(_)) + .WillOnce(CompleteContext(r, mock_image_ctx.image_ctx->op_work_queue)); + } + } + void when_process(MockJournalReplay &mock_journal_replay, EventEntry &&event_entry, Context *on_ready, Context *on_safe) { @@ -505,6 +515,7 @@ TEST_F(TestMockJournalReplay, BlockedOpFinishError) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_create(mock_image_ctx, &on_finish, "snap", 123); C_SaferCond on_start_ready; @@ -539,9 +550,11 @@ TEST_F(TestMockJournalReplay, MissingOpFinishEvent) { InSequence seq; Context *on_snap_create_finish = nullptr; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_create(mock_image_ctx, &on_snap_create_finish, "snap", 123); Context *on_snap_remove_finish = nullptr; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_remove(mock_image_ctx, &on_snap_remove_finish, "snap"); C_SaferCond on_snap_remove_ready; @@ -582,6 +595,7 @@ TEST_F(TestMockJournalReplay, MissingOpFinishEventCancelOps) { InSequence seq; Context *on_snap_create_finish = nullptr; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_create(mock_image_ctx, &on_snap_create_finish, "snap", 123); C_SaferCond on_snap_remove_ready; @@ -634,6 +648,7 @@ TEST_F(TestMockJournalReplay, OpEventError) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_remove(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -665,6 +680,7 @@ TEST_F(TestMockJournalReplay, SnapCreateEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_create(mock_image_ctx, &on_finish, "snap", 123); C_SaferCond on_start_ready; @@ -701,6 +717,7 @@ TEST_F(TestMockJournalReplay, SnapCreateEventExists) { InSequence seq; Context *on_finish = nullptr; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_create(mock_image_ctx, &on_finish, "snap", 123); C_SaferCond on_start_ready; @@ -733,6 +750,7 @@ TEST_F(TestMockJournalReplay, SnapRemoveEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_remove(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -764,6 +782,7 @@ TEST_F(TestMockJournalReplay, SnapRemoveEventDNE) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_remove(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -795,6 +814,7 @@ TEST_F(TestMockJournalReplay, SnapRenameEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_rename(mock_image_ctx, &on_finish, 234, "snap"); C_SaferCond on_start_ready; @@ -827,6 +847,7 @@ TEST_F(TestMockJournalReplay, SnapRenameEventExists) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_rename(mock_image_ctx, &on_finish, 234, "snap"); C_SaferCond on_start_ready; @@ -859,6 +880,7 @@ TEST_F(TestMockJournalReplay, SnapProtectEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_protect(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -890,6 +912,7 @@ TEST_F(TestMockJournalReplay, SnapProtectEventBusy) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_protect(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -921,6 +944,7 @@ TEST_F(TestMockJournalReplay, SnapUnprotectEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_unprotect(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -952,6 +976,7 @@ TEST_F(TestMockJournalReplay, SnapUnprotectEventInvalid) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_unprotect(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -983,6 +1008,7 @@ TEST_F(TestMockJournalReplay, SnapRollbackEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_snap_rollback(mock_image_ctx, &on_finish, "snap"); C_SaferCond on_start_ready; @@ -1014,6 +1040,7 @@ TEST_F(TestMockJournalReplay, RenameEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_rename(mock_image_ctx, &on_finish, "image"); C_SaferCond on_start_ready; @@ -1045,6 +1072,7 @@ TEST_F(TestMockJournalReplay, RenameEventExists) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_rename(mock_image_ctx, &on_finish, "image"); C_SaferCond on_start_ready; @@ -1076,6 +1104,7 @@ TEST_F(TestMockJournalReplay, ResizeEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_resize(mock_image_ctx, &on_finish, 234, 123); C_SaferCond on_start_ready; @@ -1112,6 +1141,7 @@ TEST_F(TestMockJournalReplay, FlattenEvent) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_flatten(mock_image_ctx, &on_finish); C_SaferCond on_start_ready; @@ -1143,6 +1173,7 @@ TEST_F(TestMockJournalReplay, FlattenEventInvalid) { InSequence seq; Context *on_finish; + expect_refresh_image(mock_image_ctx, false, 0); expect_flatten(mock_image_ctx, &on_finish); C_SaferCond on_start_ready; @@ -1188,5 +1219,42 @@ TEST_F(TestMockJournalReplay, UnknownEvent) { ASSERT_EQ(0, on_ready.wait()); } +TEST_F(TestMockJournalReplay, RefreshImageBeforeOpStart) { + REQUIRE_FEATURE(RBD_FEATURE_JOURNALING); + + librbd::ImageCtx *ictx; + ASSERT_EQ(0, open_image(m_image_name, &ictx)); + + MockReplayImageCtx mock_image_ctx(*ictx); + MockJournalReplay mock_journal_replay(mock_image_ctx); + expect_op_work_queue(mock_image_ctx); + + InSequence seq; + Context *on_finish; + expect_refresh_image(mock_image_ctx, true, 0); + expect_resize(mock_image_ctx, &on_finish, 234, 123); + + C_SaferCond on_start_ready; + C_SaferCond on_start_safe; + when_process(mock_journal_replay, EventEntry{ResizeEvent(123, 234)}, + &on_start_ready, &on_start_safe); + + C_SaferCond on_resume; + when_replay_op_ready(mock_journal_replay, 123, &on_resume); + ASSERT_EQ(0, on_start_ready.wait()); + + C_SaferCond on_finish_ready; + C_SaferCond on_finish_safe; + when_process(mock_journal_replay, EventEntry{OpFinishEvent(123, 0)}, + &on_finish_ready, &on_finish_safe); + + ASSERT_EQ(0, on_resume.wait()); + on_finish->complete(0); + + ASSERT_EQ(0, on_start_safe.wait()); + ASSERT_EQ(0, on_finish_ready.wait()); + ASSERT_EQ(0, on_finish_safe.wait()); +} + } // namespace journal } // namespace librbd diff --git a/src/test/librbd/mock/MockImageCtx.h b/src/test/librbd/mock/MockImageCtx.h index c7f2abfad32..b6e8e9e7362 100644 --- a/src/test/librbd/mock/MockImageCtx.h +++ b/src/test/librbd/mock/MockImageCtx.h @@ -7,6 +7,7 @@ #include "test/librbd/mock/MockAioImageRequestWQ.h" #include "test/librbd/mock/MockContextWQ.h" #include "test/librbd/mock/MockExclusiveLock.h" +#include "test/librbd/mock/MockImageState.h" #include "test/librbd/mock/MockImageWatcher.h" #include "test/librbd/mock/MockJournal.h" #include "test/librbd/mock/MockObjectMap.h" @@ -59,6 +60,7 @@ struct MockImageCtx { aio_work_queue(new MockAioImageRequestWQ()), op_work_queue(new MockContextWQ()), parent(NULL), operations(new MockOperations()), + state(new MockImageState()), image_watcher(NULL), object_map(NULL), exclusive_lock(NULL), journal(NULL), concurrent_management_ops(image_ctx.concurrent_management_ops), @@ -85,6 +87,7 @@ struct MockImageCtx { image_ctx->md_ctx.aio_flush(); image_ctx->data_ctx.aio_flush(); image_ctx->op_work_queue->drain(); + delete state; delete operations; delete image_watcher; delete op_work_queue; @@ -200,6 +203,7 @@ struct MockImageCtx { MockImageCtx *parent; MockOperations *operations; + MockImageState *state; MockImageWatcher *image_watcher; MockObjectMap *object_map; diff --git a/src/test/librbd/mock/MockImageState.h b/src/test/librbd/mock/MockImageState.h new file mode 100644 index 00000000000..6f75ecd100a --- /dev/null +++ b/src/test/librbd/mock/MockImageState.h @@ -0,0 +1,20 @@ +// -*- mode:C; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- +// vim: ts=8 sw=2 smarttab + +#ifndef CEPH_TEST_LIBRBD_MOCK_IMAGE_STATE_H +#define CEPH_TEST_LIBRBD_MOCK_IMAGE_STATE_H + +#include <gmock/gmock.h> + +class Context; + +namespace librbd { + +struct MockImageState { + MOCK_CONST_METHOD0(is_refresh_required, bool()); + MOCK_METHOD1(refresh, void(Context*)); +}; + +} // namespace librbd + +#endif // CEPH_TEST_LIBRBD_MOCK_IMAGE_STATE_H