2014-03-15 15:29:10 +01:00
|
|
|
/*
|
2017-01-03 20:48:59 +01:00
|
|
|
* Copyright 2003-2017 The Music Player Daemon Project
|
2014-03-15 15:29:10 +01:00
|
|
|
* http://www.musicpd.org
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MPD_NFS_CANCELLABLE_HXX
|
|
|
|
#define MPD_NFS_CANCELLABLE_HXX
|
|
|
|
|
|
|
|
#include "Compiler.h"
|
|
|
|
|
2014-09-28 19:18:48 +02:00
|
|
|
#include <boost/intrusive/list.hpp>
|
|
|
|
|
2014-03-15 15:29:10 +01:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
template<typename T>
|
2014-09-28 19:18:48 +02:00
|
|
|
class CancellablePointer
|
|
|
|
: public boost::intrusive::list_base_hook<boost::intrusive::link_mode<boost::intrusive::normal_link>> {
|
2014-03-15 15:29:10 +01:00
|
|
|
public:
|
|
|
|
typedef T *pointer_type;
|
|
|
|
typedef T &reference_type;
|
|
|
|
typedef const T &const_reference_type;
|
|
|
|
|
|
|
|
private:
|
|
|
|
pointer_type p;
|
|
|
|
|
|
|
|
public:
|
2014-09-28 19:18:48 +02:00
|
|
|
explicit CancellablePointer(reference_type _p):p(&_p) {}
|
2014-03-15 15:29:10 +01:00
|
|
|
|
|
|
|
CancellablePointer(const CancellablePointer &) = delete;
|
|
|
|
|
|
|
|
constexpr bool IsCancelled() const {
|
|
|
|
return p == nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Cancel() {
|
|
|
|
assert(!IsCancelled());
|
|
|
|
|
|
|
|
p = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
reference_type Get() {
|
|
|
|
assert(p != nullptr);
|
|
|
|
|
|
|
|
return *p;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool Is(const_reference_type other) const {
|
|
|
|
return p == &other;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T, typename CT=CancellablePointer<T>>
|
|
|
|
class CancellableList {
|
|
|
|
public:
|
|
|
|
typedef typename CT::reference_type reference_type;
|
|
|
|
typedef typename CT::const_reference_type const_reference_type;
|
|
|
|
|
|
|
|
private:
|
2014-09-28 19:18:48 +02:00
|
|
|
typedef boost::intrusive::list<CT,
|
|
|
|
boost::intrusive::constant_time_size<false>> List;
|
2014-03-15 15:29:10 +01:00
|
|
|
typedef typename List::iterator iterator;
|
|
|
|
typedef typename List::const_iterator const_iterator;
|
|
|
|
List list;
|
|
|
|
|
|
|
|
class MatchPointer {
|
|
|
|
const_reference_type p;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit constexpr MatchPointer(const_reference_type _p)
|
|
|
|
:p(_p) {}
|
|
|
|
|
|
|
|
constexpr bool operator()(const CT &a) const {
|
|
|
|
return a.Is(p);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
iterator Find(reference_type p) noexcept {
|
2014-03-15 15:29:10 +01:00
|
|
|
return std::find_if(list.begin(), list.end(), MatchPointer(p));
|
|
|
|
}
|
|
|
|
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
const_iterator Find(const_reference_type p) const noexcept {
|
2014-03-15 15:29:10 +01:00
|
|
|
return std::find_if(list.begin(), list.end(), MatchPointer(p));
|
|
|
|
}
|
|
|
|
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
iterator Find(CT &c) noexcept {
|
2014-09-28 19:18:48 +02:00
|
|
|
return list.iterator_to(c);
|
2014-03-15 15:29:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
const_iterator Find(const CT &c) const noexcept {
|
2014-09-28 19:18:48 +02:00
|
|
|
return list.iterator_to(c);
|
2014-03-15 15:29:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
#ifndef NDEBUG
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
bool IsEmpty() const noexcept {
|
2014-03-15 15:29:10 +01:00
|
|
|
for (const auto &c : list)
|
|
|
|
if (!c.IsCancelled())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
gcc_pure
|
2017-05-08 14:44:49 +02:00
|
|
|
bool Contains(const_reference_type p) const noexcept {
|
2014-03-15 15:29:10 +01:00
|
|
|
return Find(p) != list.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
CT &Add(reference_type p, Args&&... args) {
|
|
|
|
assert(Find(p) == list.end());
|
|
|
|
|
2014-09-28 19:18:48 +02:00
|
|
|
CT *c = new CT(p, std::forward<Args>(args)...);
|
|
|
|
list.push_back(*c);
|
|
|
|
return *c;
|
2014-03-15 15:29:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Remove(CT &ct) {
|
|
|
|
auto i = Find(ct);
|
|
|
|
assert(i != list.end());
|
|
|
|
|
|
|
|
list.erase(i);
|
2014-09-28 19:18:48 +02:00
|
|
|
delete &ct;
|
2014-03-15 15:29:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Cancel(reference_type p) {
|
|
|
|
auto i = Find(p);
|
|
|
|
assert(i != list.end());
|
|
|
|
|
|
|
|
i->Cancel();
|
|
|
|
}
|
2014-09-26 13:29:44 +02:00
|
|
|
|
2017-05-08 14:44:49 +02:00
|
|
|
CT &Get(reference_type p) noexcept {
|
2014-09-26 13:29:44 +02:00
|
|
|
auto i = Find(p);
|
|
|
|
assert(i != list.end());
|
|
|
|
|
|
|
|
return *i;
|
|
|
|
}
|
2014-12-14 15:56:53 +01:00
|
|
|
|
|
|
|
template<typename F>
|
|
|
|
void ForEach(F &&f) {
|
|
|
|
for (CT &i : list)
|
|
|
|
f(i);
|
|
|
|
}
|
2014-03-15 15:29:10 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|