2012-09-06 22:11:57 +00:00
|
|
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
|
|
|
|
// vim: ts=8 sw=2 smarttab
|
|
|
|
/*
|
|
|
|
* Ceph - scalable distributed file system
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004-2006 Sage Weil <sage@newdream.net>
|
|
|
|
*
|
|
|
|
* This is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License version 2.1, as published by the Free Software
|
|
|
|
* Foundation. See file COPYING.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef ASYNC_RESERVER_H
|
|
|
|
#define ASYNC_RESERVER_H
|
|
|
|
|
|
|
|
#include "common/Finisher.h"
|
2014-07-02 04:49:53 +00:00
|
|
|
#include "common/Formatter.h"
|
2012-09-06 22:11:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Manages a configurable number of asyncronous reservations.
|
2013-05-07 22:58:48 +00:00
|
|
|
*
|
|
|
|
* Memory usage is linear with the number of items queued and
|
|
|
|
* linear with respect to the total number of priorities used
|
|
|
|
* over all time.
|
2012-09-06 22:11:57 +00:00
|
|
|
*/
|
|
|
|
template <typename T>
|
|
|
|
class AsyncReserver {
|
|
|
|
Finisher *f;
|
2013-01-20 06:06:27 +00:00
|
|
|
unsigned max_allowed;
|
2014-06-17 20:04:18 +00:00
|
|
|
unsigned min_priority;
|
2012-09-06 22:11:57 +00:00
|
|
|
Mutex lock;
|
|
|
|
|
2013-05-07 22:58:48 +00:00
|
|
|
map<unsigned, list<pair<T, Context*> > > queues;
|
|
|
|
map<T, pair<unsigned, typename list<pair<T, Context*> >::iterator > > queue_pointers;
|
2012-09-06 22:11:57 +00:00
|
|
|
set<T> in_progress;
|
|
|
|
|
|
|
|
void do_queues() {
|
2013-05-07 22:58:48 +00:00
|
|
|
typename map<unsigned, list<pair<T, Context*> > >::reverse_iterator it;
|
|
|
|
for (it = queues.rbegin();
|
2014-06-17 20:04:18 +00:00
|
|
|
it != queues.rend() &&
|
|
|
|
in_progress.size() < max_allowed &&
|
|
|
|
it->first >= min_priority;
|
2013-05-07 22:58:48 +00:00
|
|
|
++it) {
|
|
|
|
while (in_progress.size() < max_allowed &&
|
|
|
|
!it->second.empty()) {
|
|
|
|
pair<T, Context*> p = it->second.front();
|
|
|
|
queue_pointers.erase(p.first);
|
|
|
|
it->second.pop_front();
|
|
|
|
f->queue(p.second);
|
|
|
|
in_progress.insert(p.first);
|
|
|
|
}
|
2012-09-06 22:11:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
public:
|
|
|
|
AsyncReserver(
|
|
|
|
Finisher *f,
|
2014-06-17 20:04:18 +00:00
|
|
|
unsigned max_allowed,
|
|
|
|
unsigned min_priority = 0)
|
|
|
|
: f(f),
|
|
|
|
max_allowed(max_allowed),
|
|
|
|
min_priority(min_priority),
|
|
|
|
lock("AsyncReserver::lock") {}
|
2012-09-06 22:11:57 +00:00
|
|
|
|
2013-01-20 06:06:27 +00:00
|
|
|
void set_max(unsigned max) {
|
|
|
|
Mutex::Locker l(lock);
|
|
|
|
max_allowed = max;
|
|
|
|
do_queues();
|
|
|
|
}
|
|
|
|
|
2014-06-17 20:04:18 +00:00
|
|
|
void set_min_priority(unsigned min) {
|
|
|
|
Mutex::Locker l(lock);
|
|
|
|
min_priority = min;
|
|
|
|
do_queues();
|
|
|
|
}
|
|
|
|
|
2014-07-02 04:49:53 +00:00
|
|
|
void dump(Formatter *f) {
|
|
|
|
Mutex::Locker l(lock);
|
|
|
|
f->dump_unsigned("max_allowed", max_allowed);
|
|
|
|
f->dump_unsigned("min_priority", min_priority);
|
|
|
|
f->open_array_section("queues");
|
|
|
|
for (typename map<unsigned, list<pair<T, Context*> > > ::const_iterator p =
|
|
|
|
queues.begin(); p != queues.end(); ++p) {
|
|
|
|
f->open_object_section("queue");
|
|
|
|
f->dump_unsigned("priority", p->first);
|
|
|
|
f->open_array_section("items");
|
|
|
|
for (typename list<pair<T, Context*> >::const_iterator q =
|
|
|
|
p->second.begin(); q != p->second.end(); ++q) {
|
|
|
|
f->dump_stream("item") << q->first;
|
|
|
|
}
|
|
|
|
f->close_section();
|
|
|
|
f->close_section();
|
|
|
|
}
|
|
|
|
f->close_section();
|
|
|
|
f->open_array_section("in_progress");
|
|
|
|
for (typename set<T>::const_iterator p = in_progress.begin();
|
|
|
|
p != in_progress.end();
|
|
|
|
++p) {
|
|
|
|
f->dump_stream("item") << *p;
|
|
|
|
}
|
|
|
|
f->close_section();
|
|
|
|
}
|
|
|
|
|
2012-09-06 22:11:57 +00:00
|
|
|
/**
|
|
|
|
* Requests a reservation
|
|
|
|
*
|
|
|
|
* Note, on_reserved may be called following cancel_reservation. Thus,
|
|
|
|
* the callback must be safe in that case. Callback will be called
|
|
|
|
* with no locks held. cancel_reservation must be called to release the
|
|
|
|
* reservation slot.
|
|
|
|
*/
|
|
|
|
void request_reservation(
|
|
|
|
T item, ///< [in] reservation key
|
2013-05-07 22:58:48 +00:00
|
|
|
Context *on_reserved, ///< [in] callback to be called on reservation
|
|
|
|
unsigned prio
|
2012-09-06 22:11:57 +00:00
|
|
|
) {
|
|
|
|
Mutex::Locker l(lock);
|
|
|
|
assert(!queue_pointers.count(item) &&
|
|
|
|
!in_progress.count(item));
|
2013-05-07 22:58:48 +00:00
|
|
|
queues[prio].push_back(make_pair(item, on_reserved));
|
|
|
|
queue_pointers.insert(make_pair(item, make_pair(prio,--(queues[prio]).end())));
|
2012-09-06 22:11:57 +00:00
|
|
|
do_queues();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cancels reservation
|
|
|
|
*
|
|
|
|
* Frees the reservation under key for use.
|
|
|
|
* Note, after cancel_reservation, the reservation_callback may or
|
|
|
|
* may not still be called.
|
|
|
|
*/
|
|
|
|
void cancel_reservation(
|
|
|
|
T item ///< [in] key for reservation to cancel
|
|
|
|
) {
|
|
|
|
Mutex::Locker l(lock);
|
|
|
|
if (queue_pointers.count(item)) {
|
2013-05-07 22:58:48 +00:00
|
|
|
unsigned prio = queue_pointers[item].first;
|
|
|
|
delete queue_pointers[item].second->second;
|
|
|
|
queues[prio].erase(queue_pointers[item].second);
|
2012-09-06 22:11:57 +00:00
|
|
|
queue_pointers.erase(item);
|
|
|
|
} else {
|
|
|
|
in_progress.erase(item);
|
|
|
|
}
|
|
|
|
do_queues();
|
|
|
|
}
|
2013-05-07 22:58:48 +00:00
|
|
|
static const unsigned MAX_PRIORITY = (unsigned)-1;
|
2012-09-06 22:11:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|