Manager.cxx 3.06 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * 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.
 */

#include "Manager.hxx"
21
#include "lib/fmt/ExceptionFormatter.hxx"
22
#include "event/Loop.hxx"
23
#include "util/DeleteDisposer.hxx"
24 25
#include "util/Domain.hxx"
#include "Log.hxx"
26

27 28
#include <string.h>

29 30
static constexpr Domain nfs_domain("nfs");

31
void
32
NfsManager::ManagedConnection::OnNfsConnectionError(std::exception_ptr &&e) noexcept
33
{
34 35
	FmtError(nfs_domain, "NFS error on '{}:{}': {}",
		 GetServer(), GetExportName(), e);
36

37 38 39 40
	/* defer deletion so the caller
	   (i.e. NfsConnection::OnSocketReady()) can still use this
	   object */
	manager.ScheduleDelete(*this);
41 42 43 44
}

inline bool
NfsManager::Compare::operator()(const LookupKey a,
45
				const ManagedConnection &b) const noexcept
46 47 48 49 50 51 52 53 54 55 56
{
	int result = strcmp(a.server, b.GetServer());
	if (result != 0)
		return result < 0;

	result = strcmp(a.export_name, b.GetExportName());
	return result < 0;
}

inline bool
NfsManager::Compare::operator()(const ManagedConnection &a,
57
				const LookupKey b) const noexcept
58 59 60 61 62 63 64 65 66
{
	int result = strcmp(a.GetServer(), b.server);
	if (result != 0)
		return result < 0;

	result = strcmp(a.GetExportName(), b.export_name);
	return result < 0;
}

67 68
inline bool
NfsManager::Compare::operator()(const ManagedConnection &a,
69
				const ManagedConnection &b) const noexcept
70 71 72 73 74 75 76 77 78
{
	int result = strcmp(a.GetServer(), b.GetServer());
	if (result != 0)
		return result < 0;

	result = strcmp(a.GetExportName(), b.GetExportName());
	return result < 0;
}

79
NfsManager::~NfsManager() noexcept
80
{
81
	assert(!GetEventLoop().IsAlive() || GetEventLoop().IsInside());
82 83

	CollectGarbage();
84

85
	connections.clear_and_dispose(DeleteDisposer());
86 87 88
}

NfsConnection &
89
NfsManager::GetConnection(const char *server, const char *export_name) noexcept
90 91 92
{
	assert(server != nullptr);
	assert(export_name != nullptr);
93
	assert(GetEventLoop().IsInside());
94

95 96 97 98
	Map::insert_commit_data hint;
	auto result = connections.insert_check(LookupKey{server, export_name},
					       Compare(), hint);
	if (result.second) {
99
		auto c = new ManagedConnection(*this, GetEventLoop(),
100 101 102 103 104 105
					       server, export_name);
		connections.insert_commit(*c, hint);
		return *c;
	} else {
		return *result.first;
	}
106
}
107 108

void
109
NfsManager::CollectGarbage() noexcept
110
{
111
	assert(!GetEventLoop().IsAlive() || GetEventLoop().IsInside());
112

113
	garbage.clear_and_dispose(DeleteDisposer());
114 115 116
}

void
117
NfsManager::OnIdle() noexcept
118 119 120
{
	CollectGarbage();
}