mpd/src/Partition.cxx

247 lines
4.9 KiB
C++
Raw Normal View History

/*
2021-01-01 19:54:25 +01:00
* Copyright 2003-2021 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.
*/
#include "config.h"
#include "Partition.hxx"
2015-08-06 12:45:22 +02:00
#include "Instance.hxx"
#include "Log.hxx"
#include "config/PartitionConfig.hxx"
2021-06-24 20:22:48 +02:00
#include "lib/fmt/ExceptionFormatter.hxx"
#include "song/DetachedSong.hxx"
#include "mixer/Volume.hxx"
2016-03-10 23:12:03 +01:00
#include "IdleFlags.hxx"
#include "client/Listener.hxx"
#include "client/Client.hxx"
#include "input/cache/Manager.hxx"
#include "util/Domain.hxx"
static constexpr Domain cache_domain("cache");
2016-03-05 19:05:04 +01:00
Partition::Partition(Instance &_instance,
2017-02-17 23:18:51 +01:00
const char *_name,
const PartitionConfig &_config) noexcept
:instance(_instance),
2017-02-17 23:18:51 +01:00
name(_name),
config(_config),
listener(new ClientListener(instance.event_loop, *this)),
idle_monitor(instance.event_loop, BIND_THIS_METHOD(OnIdleMonitor)),
global_events(instance.event_loop, BIND_THIS_METHOD(OnGlobalEvent)),
playlist(config.queue.max_length, *this),
outputs(pc, *this),
pc(*this, outputs,
instance.input_cache.get(),
config.player)
2016-03-05 19:05:04 +01:00
{
UpdateEffectiveReplayGainMode();
2016-03-05 19:05:04 +01:00
}
Partition::~Partition() noexcept = default;
void
Partition::BeginShutdown() noexcept
{
pc.Kill();
listener.reset();
}
static void
PrefetchSong(InputCacheManager &cache, const char *uri) noexcept
{
if (cache.Contains(uri))
return;
2021-06-24 20:22:48 +02:00
FmtDebug(cache_domain, "Prefetch '{}'", uri);
try {
cache.Prefetch(uri);
} catch (...) {
2021-06-24 20:22:48 +02:00
FmtError(cache_domain,
"Prefetch '{}' failed: {}",
uri, std::current_exception());
}
}
static void
PrefetchSong(InputCacheManager &cache, const DetachedSong &song) noexcept
{
PrefetchSong(cache, song.GetURI());
}
inline void
Partition::PrefetchQueue() noexcept
{
if (!instance.input_cache)
return;
auto &cache = *instance.input_cache;
int next = playlist.GetNextPosition();
if (next >= 0)
PrefetchSong(cache, playlist.queue.Get(next));
// TODO: prefetch more songs
}
void
2019-05-31 13:54:52 +02:00
Partition::UpdateEffectiveReplayGainMode() noexcept
{
auto mode = replay_gain_mode;
if (mode == ReplayGainMode::AUTO)
mode = playlist.queue.random
? ReplayGainMode::TRACK
: ReplayGainMode::ALBUM;
pc.LockSetReplayGainMode(mode);
outputs.SetReplayGainMode(mode);
}
#ifdef ENABLE_DATABASE
2015-08-06 12:45:22 +02:00
const Database *
2019-05-31 13:54:52 +02:00
Partition::GetDatabase() const noexcept
2015-08-06 12:45:22 +02:00
{
return instance.GetDatabase();
2015-08-06 12:45:22 +02:00
}
2016-10-26 18:47:19 +02:00
const Database &
Partition::GetDatabaseOrThrow() const
{
return instance.GetDatabaseOrThrow();
}
void
2019-05-31 13:54:52 +02:00
Partition::DatabaseModified(const Database &db) noexcept
{
playlist.DatabaseModified(db);
EmitIdle(IDLE_DATABASE);
}
#endif
void
2019-05-31 13:54:52 +02:00
Partition::TagModified() noexcept
{
auto song = pc.LockReadTaggedSong();
if (song)
playlist.TagModified(std::move(*song));
}
void
Partition::TagModified(const char *uri, const Tag &tag) noexcept
{
playlist.TagModified(uri, tag);
}
void
2019-05-31 13:54:52 +02:00
Partition::SyncWithPlayer() noexcept
{
playlist.SyncWithPlayer(pc);
/* TODO: invoke this function in batches, to let the hard disk
spin down in between */
PrefetchQueue();
}
void
2019-05-31 13:54:52 +02:00
Partition::BorderPause() noexcept
{
playlist.BorderPause(pc);
}
void
2019-05-31 13:57:46 +02:00
Partition::OnQueueModified() noexcept
{
EmitIdle(IDLE_PLAYLIST);
}
void
2019-05-31 13:57:46 +02:00
Partition::OnQueueOptionsChanged() noexcept
{
EmitIdle(IDLE_OPTIONS);
}
void
2019-05-31 13:57:46 +02:00
Partition::OnQueueSongStarted() noexcept
{
EmitIdle(IDLE_PLAYER);
}
void
Partition::OnPlayerError() noexcept
{
EmitIdle(IDLE_PLAYER);
}
void
2017-11-26 12:39:09 +01:00
Partition::OnPlayerSync() noexcept
{
EmitGlobalEvent(SYNC_WITH_PLAYER);
}
void
2017-11-26 12:39:09 +01:00
Partition::OnPlayerTagModified() noexcept
{
EmitGlobalEvent(TAG_MODIFIED);
/* notify all clients that the tag of the current song has
changed */
EmitIdle(IDLE_PLAYER);
}
void
Partition::OnBorderPause() noexcept
{
EmitGlobalEvent(BORDER_PAUSE);
}
void
2019-05-31 14:09:47 +02:00
Partition::OnMixerVolumeChanged(Mixer &, int) noexcept
{
InvalidateHardwareVolume();
/* notify clients */
EmitIdle(IDLE_MIXER);
}
void
Partition::OnIdleMonitor(unsigned mask) noexcept
{
/* send "idle" notifications to all subscribed
clients */
for (auto &client : clients)
client.IdleAdd(mask);
if (mask & (IDLE_PLAYLIST|IDLE_PLAYER|IDLE_MIXER|IDLE_OUTPUT))
instance.OnStateModified();
}
void
Partition::OnGlobalEvent(unsigned mask) noexcept
{
if ((mask & SYNC_WITH_PLAYER) != 0)
SyncWithPlayer();
if ((mask & TAG_MODIFIED) != 0)
TagModified();
if ((mask & BORDER_PAUSE) != 0)
BorderPause();
}