Observer.cxx 2.99 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 "Observer.hxx"
21 22
#include "Filter.hxx"
#include "Prepared.hxx"
23 24
#include "util/ConstBuffer.hxx"

25
#include <cassert>
26 27 28 29

class FilterObserver::PreparedProxy final : public PreparedFilter {
	FilterObserver &observer;

30
	std::unique_ptr<PreparedFilter> prepared_filter;
31 32 33 34
	Proxy *child = nullptr;

public:
	PreparedProxy(FilterObserver &_observer,
35
		      std::unique_ptr<PreparedFilter> _prepared_filter) noexcept
36
		:observer(_observer),
37
		 prepared_filter(std::move(_prepared_filter)) {}
38

39
	~PreparedProxy() noexcept override {
40 41 42 43 44 45
		assert(child == nullptr);
		assert(observer.proxy == this);

		observer.proxy = nullptr;
	}

46 47 48
	PreparedProxy(const PreparedProxy &) = delete;
	PreparedProxy &operator=(const PreparedProxy &) = delete;

Rosen Penev's avatar
Rosen Penev committed
49
	void Clear([[maybe_unused]] Proxy *_child) noexcept {
50 51 52 53
		assert(child == _child);
		child = nullptr;
	}

54
	Filter *Get() noexcept;
55

56
	std::unique_ptr<Filter> Open(AudioFormat &af) override;
57 58 59 60 61
};

class FilterObserver::Proxy final : public Filter {
	PreparedProxy &parent;

62
	std::unique_ptr<Filter> filter;
63 64

public:
65
	Proxy(PreparedProxy &_parent, std::unique_ptr<Filter> _filter) noexcept
66
		:Filter(_filter->GetOutAudioFormat()),
67
		 parent(_parent), filter(std::move(_filter)) {}
68

69
	~Proxy() noexcept override {
70 71 72
		parent.Clear(this);
	}

73 74 75
	Proxy(const Proxy &) = delete;
	Proxy &operator=(const Proxy &) = delete;

76
	Filter *Get() noexcept {
77
		return filter.get();
78 79
	}

80
	void Reset() noexcept override {
81 82 83
		filter->Reset();
	}

84 85
	ConstBuffer<void> FilterPCM(ConstBuffer<void> src) override {
		return filter->FilterPCM(src);
86
	}
87

88
	ConstBuffer<void> Flush() override {
89 90
		return filter->Flush();
	}
91 92 93
};

Filter *
94
FilterObserver::PreparedProxy::Get() noexcept
95 96 97 98 99 100
{
	return child != nullptr
		? child->Get()
		: nullptr;
}

101
std::unique_ptr<Filter>
102
FilterObserver::PreparedProxy::Open(AudioFormat &af)
103 104 105
{
	assert(child == nullptr);

106 107 108
	auto c = std::make_unique<Proxy>(*this, prepared_filter->Open(af));
	child = c.get();
	return c;
109 110
}

111 112
std::unique_ptr<PreparedFilter>
FilterObserver::Set(std::unique_ptr<PreparedFilter> pf)
113 114 115
{
	assert(proxy == nullptr);

116 117 118
	auto p = std::make_unique<PreparedProxy>(*this, std::move(pf));
	proxy = p.get();
	return p;
119 120 121
}

Filter *
122
FilterObserver::Get() noexcept
123 124 125 126 127
{
	return proxy != nullptr
		? proxy->Get()
		: nullptr;
}