// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // vim: ts=8 sw=2 smarttab #include "tools/rbd_mirror/pool_watcher/RefreshImagesRequest.h" #include "common/debug.h" #include "common/errno.h" #include "cls/rbd/cls_rbd_client.h" #include "librbd/Utils.h" #include #define dout_context g_ceph_context #define dout_subsys ceph_subsys_rbd_mirror #undef dout_prefix #define dout_prefix *_dout << "rbd::mirror::pool_watcher::RefreshImagesRequest " \ << this << " " << __func__ << ": " namespace rbd { namespace mirror { namespace pool_watcher { static const uint32_t MAX_RETURN = 1024; using librbd::util::create_rados_callback; template void RefreshImagesRequest::send() { m_image_ids->clear(); mirror_image_list(); } template void RefreshImagesRequest::mirror_image_list() { dout(10) << dendl; librados::ObjectReadOperation op; librbd::cls_client::mirror_image_list_start(&op, m_start_after, MAX_RETURN); librados::AioCompletion *aio_comp = create_rados_callback< RefreshImagesRequest, &RefreshImagesRequest::handle_mirror_image_list>(this); int r = m_remote_io_ctx.aio_operate(RBD_MIRRORING, aio_comp, &op, &m_out_bl); assert(r == 0); aio_comp->release(); } template void RefreshImagesRequest::handle_mirror_image_list(int r) { dout(10) << "r=" << r << dendl; std::map ids; if (r == 0) { bufferlist::iterator it = m_out_bl.begin(); r = librbd::cls_client::mirror_image_list_finish(&it, &ids); } if (r < 0 && r != -ENOENT) { derr << "failed to list mirrored images: " << cpp_strerror(r) << dendl; finish(r); return; } // store as global -> local image ids for (auto &id : ids) { m_image_ids->emplace(id.second, id.first); } if (ids.size() == MAX_RETURN) { m_start_after = ids.rbegin()->first; mirror_image_list(); return; } finish(0); } template void RefreshImagesRequest::finish(int r) { dout(10) << "r=" << r << dendl; m_on_finish->complete(r); delete this; } } // namespace pool_watcher } // namespace mirror } // namespace rbd template class rbd::mirror::pool_watcher::RefreshImagesRequest;