#include "librbd/ImageCtx.h"
#include "librbd/Utils.h"
#include "librbd/io/AioCompletion.h"
+#include "librbd/io/FlushTracker.h"
#include "librbd/io/ImageDispatchSpec.h"
#define dout_subsys ceph_subsys_rbd
template <typename I>
QueueImageDispatch<I>::QueueImageDispatch(I* image_ctx)
- : m_image_ctx(image_ctx) {
+ : m_image_ctx(image_ctx), m_flush_tracker(new FlushTracker<I>(image_ctx)) {
auto cct = m_image_ctx->cct;
ldout(cct, 5) << "ictx=" << image_ctx << dendl;
}
+template <typename I>
+QueueImageDispatch<I>::~QueueImageDispatch() {
+ delete m_flush_tracker;
+}
+
template <typename I>
void QueueImageDispatch<I>::shut_down(Context* on_finish) {
+ m_flush_tracker->shut_down();
on_finish->complete(0);
}
auto cct = m_image_ctx->cct;
ldout(cct, 20) << "tid=" << tid << dendl;
- return enqueue(dispatch_result, on_dispatched);
+ return enqueue(true, tid, dispatch_result, on_dispatched);
}
template <typename I>
auto cct = m_image_ctx->cct;
ldout(cct, 20) << "tid=" << tid << dendl;
- return enqueue(dispatch_result, on_dispatched);
+ return enqueue(false, tid, dispatch_result, on_dispatched);
}
template <typename I>
auto cct = m_image_ctx->cct;
ldout(cct, 20) << "tid=" << tid << dendl;
- return enqueue(dispatch_result, on_dispatched);
+ return enqueue(false, tid, dispatch_result, on_dispatched);
}
template <typename I>
auto cct = m_image_ctx->cct;
ldout(cct, 20) << "tid=" << tid << dendl;
- return enqueue(dispatch_result, on_dispatched);
+ return enqueue(false, tid, dispatch_result, on_dispatched);
}
template <typename I>
auto cct = m_image_ctx->cct;
ldout(cct, 20) << "tid=" << tid << dendl;
- return enqueue(dispatch_result, on_dispatched);
+ return enqueue(false, tid, dispatch_result, on_dispatched);
}
template <typename I>
return false;
}
- return enqueue(dispatch_result, on_dispatched);
+ *dispatch_result = DISPATCH_RESULT_CONTINUE;
+ m_flush_tracker->flush(on_dispatched);
+ return true;
+}
+
+template <typename I>
+void QueueImageDispatch<I>::handle_finished(int r, uint64_t tid) {
+ auto cct = m_image_ctx->cct;
+ ldout(cct, 20) << "tid=" << tid << dendl;
+
+ m_flush_tracker->finish_io(tid);
}
template <typename I>
bool QueueImageDispatch<I>::enqueue(
- DispatchResult* dispatch_result, Context* on_dispatched) {
+ bool read_op, uint64_t tid, DispatchResult* dispatch_result,
+ Context* on_dispatched) {
if (!m_image_ctx->non_blocking_aio) {
return false;
}
+ if (!read_op) {
+ m_flush_tracker->start_io(tid);
+ }
+
*dispatch_result = DISPATCH_RESULT_CONTINUE;
m_image_ctx->asio_engine->post(on_dispatched, 0);
return true;
namespace io {
struct AioCompletion;
+template <typename> class FlushTracker;
template <typename ImageCtxT>
class QueueImageDispatch : public ImageDispatchInterface {
public:
QueueImageDispatch(ImageCtxT* image_ctx);
+ ~QueueImageDispatch();
ImageDispatchLayer get_dispatch_layer() const override {
return IMAGE_DISPATCH_LAYER_QUEUE;
std::atomic<uint32_t>* image_dispatch_flags,
DispatchResult* dispatch_result, Context* on_dispatched) override;
- void handle_finished(int r, uint64_t tid) override {
- }
+ void handle_finished(int r, uint64_t tid) override;
private:
ImageCtxT* m_image_ctx;
- bool enqueue(DispatchResult* dispatch_result, Context* on_dispatched);
+ FlushTracker<ImageCtxT>* m_flush_tracker;
+
+ bool enqueue(bool read_op, uint64_t tid, DispatchResult* dispatch_result,
+ Context* on_dispatched);
};