]> git.proxmox.com Git - ceph.git/blame - ceph/src/librbd/object_map/UpdateRequest.cc
update sources to 12.2.2
[ceph.git] / ceph / src / librbd / object_map / UpdateRequest.cc
CommitLineData
7c673cae
FG
1// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
2// vim: ts=8 sw=2 smarttab
3
4#include "librbd/object_map/UpdateRequest.h"
5#include "include/rbd/object_map_types.h"
6#include "include/stringify.h"
7#include "common/dout.h"
8#include "librbd/ImageCtx.h"
9#include "librbd/ObjectMap.h"
3efd9988 10#include "librbd/Utils.h"
7c673cae
FG
11#include "cls/lock/cls_lock_client.h"
12#include <string>
13
14#define dout_subsys ceph_subsys_rbd
15#undef dout_prefix
3efd9988
FG
16#define dout_prefix *_dout << "librbd::object_map::UpdateRequest: " << this \
17 << " " << __func__ << ": "
7c673cae
FG
18
19namespace librbd {
20namespace object_map {
21
3efd9988
FG
22namespace {
23
24// keep aligned to bit_vector 4K block sizes
25const uint64_t MAX_OBJECTS_PER_UPDATE = 256 * (1 << 10);
26
27}
28
7c673cae
FG
29template <typename I>
30void UpdateRequest<I>::send() {
3efd9988
FG
31 update_object_map();
32}
33
34template <typename I>
35void UpdateRequest<I>::update_object_map() {
7c673cae
FG
36 assert(m_image_ctx.snap_lock.is_locked());
37 assert(m_image_ctx.object_map_lock.is_locked());
38 CephContext *cct = m_image_ctx.cct;
39
3efd9988
FG
40 // break very large requests into manageable batches
41 m_update_end_object_no = MIN(
42 m_end_object_no, m_update_start_object_no + MAX_OBJECTS_PER_UPDATE);
43
7c673cae 44 std::string oid(ObjectMap<>::object_map_name(m_image_ctx.id, m_snap_id));
3efd9988
FG
45 ldout(cct, 20) << "ictx=" << &m_image_ctx << ", oid=" << oid << ", "
46 << "[" << m_update_start_object_no << ","
47 << m_update_end_object_no << ") = "
7c673cae
FG
48 << (m_current_state ?
49 stringify(static_cast<uint32_t>(*m_current_state)) : "")
50 << "->" << static_cast<uint32_t>(m_new_state)
51 << dendl;
52
53 librados::ObjectWriteOperation op;
54 if (m_snap_id == CEPH_NOSNAP) {
55 rados::cls::lock::assert_locked(&op, RBD_LOCK_NAME, LOCK_EXCLUSIVE, "", "");
56 }
3efd9988
FG
57 cls_client::object_map_update(&op, m_update_start_object_no,
58 m_update_end_object_no, m_new_state,
59 m_current_state);
7c673cae 60
3efd9988
FG
61 auto rados_completion = librbd::util::create_rados_callback<
62 UpdateRequest<I>, &UpdateRequest<I>::handle_update_object_map>(this);
31f18b77
FG
63 std::vector<librados::snap_t> snaps;
64 int r = m_image_ctx.md_ctx.aio_operate(
65 oid, rados_completion, &op, 0, snaps,
66 (m_trace.valid() ? m_trace.get_info() : nullptr));
7c673cae
FG
67 assert(r == 0);
68 rados_completion->release();
69}
70
71template <typename I>
3efd9988
FG
72void UpdateRequest<I>::handle_update_object_map(int r) {
73 ldout(m_image_ctx.cct, 20) << "r=" << r << dendl;
74
75 {
76 RWLock::RLocker snap_locker(m_image_ctx.snap_lock);
77 RWLock::WLocker object_map_locker(m_image_ctx.object_map_lock);
78 update_in_memory_object_map();
79
80 if (m_update_end_object_no < m_end_object_no) {
81 m_update_start_object_no = m_update_end_object_no;
82 update_object_map();
83 return;
84 }
85 }
86
87 // no more batch updates to send
88 complete(r);
89}
90
91template <typename I>
92void UpdateRequest<I>::update_in_memory_object_map() {
93 assert(m_image_ctx.snap_lock.is_locked());
94 assert(m_image_ctx.object_map_lock.is_locked());
7c673cae
FG
95
96 // rebuilding the object map might update on-disk only
97 if (m_snap_id == m_image_ctx.snap_id) {
3efd9988
FG
98 ldout(m_image_ctx.cct, 20) << dendl;
99
100 auto it = m_object_map.begin() +
101 MIN(m_update_start_object_no, m_object_map.size());
102 auto end_it = m_object_map.begin() +
103 MIN(m_update_end_object_no, m_object_map.size());
104 for (; it != end_it; ++it) {
105 auto state_ref = *it;
106 uint8_t state = state_ref;
7c673cae
FG
107 if (!m_current_state || state == *m_current_state ||
108 (*m_current_state == OBJECT_EXISTS && state == OBJECT_EXISTS_CLEAN)) {
3efd9988 109 state_ref = m_new_state;
7c673cae
FG
110 }
111 }
112 }
113}
114
3efd9988
FG
115template <typename I>
116void UpdateRequest<I>::finish_request() {
117}
118
7c673cae
FG
119} // namespace object_map
120} // namespace librbd
121
122template class librbd::object_map::UpdateRequest<librbd::ImageCtx>;