178 lines
3.5 KiB
C++
178 lines
3.5 KiB
C++
|
/*
|
||
|
* Copyright (C) 2003-2014 The Music Player Daemon Project
|
||
|
* 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"
|
||
|
|
||
|
#include <list>
|
||
|
#include <algorithm>
|
||
|
|
||
|
#include <assert.h>
|
||
|
|
||
|
template<typename T>
|
||
|
class CancellablePointer {
|
||
|
public:
|
||
|
typedef T *pointer_type;
|
||
|
typedef T &reference_type;
|
||
|
typedef const T &const_reference_type;
|
||
|
|
||
|
private:
|
||
|
pointer_type p;
|
||
|
|
||
|
public:
|
||
|
explicit constexpr CancellablePointer(reference_type _p):p(&_p) {}
|
||
|
|
||
|
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:
|
||
|
typedef std::list<CT> List;
|
||
|
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
|
||
|
iterator Find(reference_type p) {
|
||
|
return std::find_if(list.begin(), list.end(), MatchPointer(p));
|
||
|
}
|
||
|
|
||
|
gcc_pure
|
||
|
const_iterator Find(const_reference_type p) const {
|
||
|
return std::find_if(list.begin(), list.end(), MatchPointer(p));
|
||
|
}
|
||
|
|
||
|
class MatchReference {
|
||
|
const CT &c;
|
||
|
|
||
|
public:
|
||
|
constexpr explicit MatchReference(const CT &_c):c(_c) {}
|
||
|
|
||
|
gcc_pure
|
||
|
bool operator()(const CT &a) const {
|
||
|
return &a == &c;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
gcc_pure
|
||
|
iterator Find(CT &c) {
|
||
|
return std::find_if(list.begin(), list.end(),
|
||
|
MatchReference(c));
|
||
|
}
|
||
|
|
||
|
gcc_pure
|
||
|
const_iterator Find(const CT &c) const {
|
||
|
return std::find_if(list.begin(), list.end(),
|
||
|
MatchReference(c));
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
#ifndef NDEBUG
|
||
|
gcc_pure
|
||
|
bool IsEmpty() const {
|
||
|
for (const auto &c : list)
|
||
|
if (!c.IsCancelled())
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
gcc_pure
|
||
|
bool Contains(const_reference_type p) const {
|
||
|
return Find(p) != list.end();
|
||
|
}
|
||
|
|
||
|
template<typename... Args>
|
||
|
CT &Add(reference_type p, Args&&... args) {
|
||
|
assert(Find(p) == list.end());
|
||
|
|
||
|
list.emplace_back(p, std::forward<Args>(args)...);
|
||
|
return list.back();
|
||
|
}
|
||
|
|
||
|
void RemoveLast() {
|
||
|
list.pop_back();
|
||
|
}
|
||
|
|
||
|
bool RemoveOptional(CT &ct) {
|
||
|
auto i = Find(ct);
|
||
|
if (i == list.end())
|
||
|
return false;
|
||
|
|
||
|
list.erase(i);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void Remove(CT &ct) {
|
||
|
auto i = Find(ct);
|
||
|
assert(i != list.end());
|
||
|
|
||
|
list.erase(i);
|
||
|
}
|
||
|
|
||
|
void Cancel(reference_type p) {
|
||
|
auto i = Find(p);
|
||
|
assert(i != list.end());
|
||
|
|
||
|
i->Cancel();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#endif
|