[this](int r) {
Mutex::Locker timer_locker(m_threads->timer_lock);
Mutex::Locker locker(m_lock);
-
- for (auto it : m_instances) {
- cancel_remove_task(it.second);
- }
+ cancel_remove_task();
wait_for_ops();
});
}
template <typename I>
-void Instances<I>::notify(const std::string &instance_id) {
- dout(20) << instance_id << dendl;
+void Instances<I>::acked(const InstanceIds& instance_ids) {
+ dout(20) << "instance_ids=" << instance_ids << dendl;
Mutex::Locker locker(m_lock);
-
if (m_on_finish != nullptr) {
dout(20) << "received on shut down, ignoring" << dendl;
return;
}
- Context *ctx = new C_Notify(this, instance_id);
-
+ Context *ctx = new C_HandleAcked(this, instance_ids);
m_threads->work_queue->queue(ctx, 0);
}
template <typename I>
-void Instances<I>::handle_notify(const std::string &instance_id) {
- dout(20) << instance_id << dendl;
+void Instances<I>::handle_acked(const InstanceIds& instance_ids) {
+ dout(5) << "instance_ids=" << instance_ids << dendl;
Mutex::Locker timer_locker(m_threads->timer_lock);
Mutex::Locker locker(m_lock);
-
if (m_on_finish != nullptr) {
dout(20) << "handled on shut down, ignoring" << dendl;
return;
}
- auto &instance = m_instances.insert(
- std::make_pair(instance_id, Instance(instance_id))).first->second;
+ auto time = ceph_clock_now();
+ for (auto& instance_id : instance_ids) {
+ auto &instance = m_instances.insert(
+ std::make_pair(instance_id, Instance{})).first->second;
+ instance.acked_time = time;
+ }
- schedule_remove_task(instance);
+ schedule_remove_task(time);
}
template <typename I>
Context *on_finish = nullptr;
{
- Mutex::Locker timer_locker(m_threads->timer_lock);
Mutex::Locker locker(m_lock);
-
- if (r < 0) {
- derr << "error retrieving instances: " << cpp_strerror(r) << dendl;
- } else {
- auto my_instance_id = stringify(m_ioctx.get_instance_id());
- for (auto &instance_id : m_instance_ids) {
- if (instance_id == my_instance_id) {
- continue;
- }
- auto &instance = m_instances.insert(
- std::make_pair(instance_id, Instance(instance_id))).first->second;
- schedule_remove_task(instance);
- }
- }
std::swap(on_finish, m_on_finish);
}
+
+ if (r < 0) {
+ derr << "error retrieving instances: " << cpp_strerror(r) << dendl;
+ } else {
+ handle_acked(m_instance_ids);
+ }
on_finish->complete(r);
}
}
template <typename I>
-void Instances<I>::remove_instance(Instance &instance) {
+void Instances<I>::remove_instances(const utime_t& time) {
assert(m_lock.is_locked());
- dout(20) << instance.id << dendl;
+ InstanceIds instance_ids;
+ for (auto& instance_pair : m_instances) {
+ auto& instance = instance_pair.second;
+ if (instance.state != INSTANCE_STATE_REMOVING &&
+ instance.acked_time <= time) {
+ instance.state = INSTANCE_STATE_REMOVING;
+ instance_ids.push_back(instance_pair.first);
+ }
+ }
- Context *ctx = create_async_context_callback(
- m_threads->work_queue, create_context_callback<
- Instances, &Instances<I>::handle_remove_instance>(this));
+ dout(20) << "instance_ids=" << instance_ids << dendl;
+ Context* ctx = new FunctionContext([this, instance_ids](int r) {
+ handle_remove_instances(r, instance_ids);
+ });
+ ctx = create_async_context_callback(m_threads->work_queue, ctx);
+
+ auto gather_ctx = new C_Gather(m_cct, ctx);
+ for (auto& instance_id : instance_ids) {
+ InstanceWatcher<I>::remove_instance(m_ioctx, m_threads->work_queue,
+ instance_id, gather_ctx->new_sub());
+ }
m_async_op_tracker.start_op();
- InstanceWatcher<I>::remove_instance(m_ioctx, m_threads->work_queue,
- instance.id, ctx);
- m_instances.erase(instance.id);
+ gather_ctx->activate();
}
template <typename I>
-void Instances<I>::handle_remove_instance(int r) {
+void Instances<I>::handle_remove_instances(
+ int r, const InstanceIds& instance_ids) {
+ Mutex::Locker timer_locker(m_threads->timer_lock);
Mutex::Locker locker(m_lock);
- dout(20) << " r=" << r << dendl;
-
+ dout(20) << "r=" << r << ", instance_ids=" << instance_ids << dendl;
assert(r == 0);
+ for (auto& instance_id : instance_ids) {
+ m_instances.erase(instance_id);
+ }
+
+ // reschedule the timer for the next batch
+ schedule_remove_task(ceph_clock_now());
m_async_op_tracker.finish_op();
}
template <typename I>
-void Instances<I>::cancel_remove_task(Instance &instance) {
+void Instances<I>::cancel_remove_task() {
assert(m_threads->timer_lock.is_locked());
assert(m_lock.is_locked());
- if (instance.timer_task == nullptr) {
+ if (m_timer_task == nullptr) {
return;
}
- dout(20) << instance.timer_task << dendl;
+ dout(20) << dendl;
- bool canceled = m_threads->timer->cancel_event(instance.timer_task);
+ bool canceled = m_threads->timer->cancel_event(m_timer_task);
assert(canceled);
- instance.timer_task = nullptr;
+ m_timer_task = nullptr;
}
template <typename I>
-void Instances<I>::schedule_remove_task(Instance &instance) {
- dout(20) << dendl;
-
- cancel_remove_task(instance);
+void Instances<I>::schedule_remove_task(const utime_t& time) {
+ cancel_remove_task();
+ if (m_on_finish != nullptr) {
+ dout(20) << "received on shut down, ignoring" << dendl;
+ return;
+ }
+ dout(20) << dendl;
int after = m_cct->_conf->get_val<int64_t>("rbd_mirror_leader_heartbeat_interval") *
(1 + m_cct->_conf->get_val<int64_t>("rbd_mirror_leader_max_missed_heartbeats") +
m_cct->_conf->get_val<int64_t>("rbd_mirror_leader_max_acquire_attempts_before_break"));
- instance.timer_task = new FunctionContext(
- [this, &instance](int r) {
+ bool schedule = false;
+ utime_t oldest_time = time;
+ for (auto& instance : m_instances) {
+ if (instance.second.state == INSTANCE_STATE_REMOVING) {
+ continue;
+ }
+
+ oldest_time = std::min(oldest_time, instance.second.acked_time);
+ schedule = true;
+ }
+
+ if (!schedule) {
+ return;
+ }
+
+ // schedule a time to fire when the oldest instance should be removed
+ m_timer_task = new FunctionContext(
+ [this, oldest_time](int r) {
assert(m_threads->timer_lock.is_locked());
Mutex::Locker locker(m_lock);
- instance.timer_task = nullptr;
- remove_instance(instance);
- });
+ m_timer_task = nullptr;
- dout(20) << "scheduling instance " << instance.id << " remove after " << after
- << " sec (task " << instance.timer_task << ")" << dendl;
+ remove_instances(oldest_time);
+ });
- m_threads->timer->add_event_after(after, instance.timer_task);
+ oldest_time += after;
+ m_threads->timer->add_event_at(oldest_time, m_timer_task);
}
} // namespace mirror
template <typename ImageCtxT = librbd::ImageCtx>
class Instances {
public:
+ typedef std::vector<std::string> InstanceIds;
+
static Instances *create(Threads<ImageCtxT> *threads,
librados::IoCtx &ioctx) {
return new Instances(threads, ioctx);
void init(Context *on_finish);
void shut_down(Context *on_finish);
- void notify(const std::string &instance_id);
+ void acked(const InstanceIds& instance_ids);
+
void list(std::vector<std::string> *instance_ids);
private:
* @endverbatim
*/
- struct Instance {
- std::string id;
- Context *timer_task = nullptr;
+ enum InstanceState {
+ INSTANCE_STATE_IDLE,
+ INSTANCE_STATE_REMOVING
+ };
- Instance(const std::string &instance_id) : id(instance_id) {
- }
+ struct Instance {
+ utime_t acked_time{};
+ InstanceState state = INSTANCE_STATE_IDLE;
};
- struct C_Notify : Context {
+ struct C_NotifyBase : public Context {
Instances *instances;
- std::string instance_id;
+ InstanceIds instance_ids;
- C_Notify(Instances *instances, const std::string &instance_id)
- : instances(instances), instance_id(instance_id) {
+ C_NotifyBase(Instances *instances, const InstanceIds& instance_ids)
+ : instances(instances), instance_ids(instance_ids) {
instances->m_async_op_tracker.start_op();
}
void finish(int r) override {
- instances->handle_notify(instance_id);
+ execute();
instances->m_async_op_tracker.finish_op();
}
+
+ virtual void execute() = 0;
+ };
+
+ struct C_HandleAcked : public C_NotifyBase {
+ C_HandleAcked(Instances *instances, const InstanceIds& instance_ids)
+ : C_NotifyBase(instances, instance_ids) {
+ }
+
+ void execute() override {
+ this->instances->handle_acked(this->instance_ids);
+ }
};
Threads<ImageCtxT> *m_threads;
CephContext *m_cct;
Mutex m_lock;
- std::vector<std::string> m_instance_ids;
+ InstanceIds m_instance_ids;
std::map<std::string, Instance> m_instances;
Context *m_on_finish = nullptr;
AsyncOpTracker m_async_op_tracker;
- void handle_notify(const std::string &instance_id);
+ Context *m_timer_task = nullptr;
+
+ void handle_acked(const InstanceIds& instance_ids);
void get_instances();
void handle_get_instances(int r);
void wait_for_ops();
void handle_wait_for_ops(int r);
- void remove_instance(Instance &instance);
- void handle_remove_instance(int r);
+ void remove_instances(const utime_t& time);
+ void handle_remove_instances(int r, const InstanceIds& instance_ids);
- void cancel_remove_task(Instance &instance);
- void schedule_remove_task(Instance &instance);
+ void cancel_remove_task();
+ void schedule_remove_task(const utime_t& time);
};
} // namespace mirror