aboutsummaryrefslogblamecommitdiffstats
path: root/src/lib/nfs/Cancellable.hxx
blob: 50762b582d90802497894f846aae3732ee215969 (plain) (tree)
















































































































































































                                                                               
/*
 * 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