/* * 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 #include #include template 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> class CancellableList { public: typedef typename CT::reference_type reference_type; typedef typename CT::const_reference_type const_reference_type; private: typedef std::list 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 CT &Add(reference_type p, Args&&... args) { assert(Find(p) == list.end()); list.emplace_back(p, std::forward(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