2013-10-21 23:40:52 +02:00
|
|
|
/*
|
2022-07-14 17:58:12 +02:00
|
|
|
* Copyright 2003-2022 The Music Player Daemon Project
|
2013-10-21 23:40:52 +02:00
|
|
|
* 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"
|
2019-05-08 18:39:00 +02:00
|
|
|
#include "Log.hxx"
|
2021-12-03 20:22:52 +01:00
|
|
|
#include "config/PartitionConfig.hxx"
|
2021-06-24 20:22:48 +02:00
|
|
|
#include "lib/fmt/ExceptionFormatter.hxx"
|
2018-08-02 13:45:43 +02:00
|
|
|
#include "song/DetachedSong.hxx"
|
2016-03-10 23:12:03 +01:00
|
|
|
#include "IdleFlags.hxx"
|
2018-01-29 23:53:52 +01:00
|
|
|
#include "client/Listener.hxx"
|
2020-01-20 13:16:13 +01:00
|
|
|
#include "client/Client.hxx"
|
2019-05-08 18:39:00 +02:00
|
|
|
#include "input/cache/Manager.hxx"
|
|
|
|
#include "util/Domain.hxx"
|
|
|
|
|
|
|
|
static constexpr Domain cache_domain("cache");
|
2013-10-21 23:40:52 +02:00
|
|
|
|
2016-03-05 19:05:04 +01:00
|
|
|
Partition::Partition(Instance &_instance,
|
2017-02-17 23:18:51 +01:00
|
|
|
const char *_name,
|
2021-12-03 20:22:52 +01:00
|
|
|
const PartitionConfig &_config) noexcept
|
2016-03-10 22:47:47 +01:00
|
|
|
:instance(_instance),
|
2017-02-17 23:18:51 +01:00
|
|
|
name(_name),
|
2021-12-03 20:22:52 +01:00
|
|
|
config(_config),
|
2018-01-29 23:53:52 +01:00
|
|
|
listener(new ClientListener(instance.event_loop, *this)),
|
2020-01-20 13:28:58 +01:00
|
|
|
idle_monitor(instance.event_loop, BIND_THIS_METHOD(OnIdleMonitor)),
|
2016-06-17 17:00:05 +02:00
|
|
|
global_events(instance.event_loop, BIND_THIS_METHOD(OnGlobalEvent)),
|
2021-12-03 20:22:52 +01:00
|
|
|
playlist(config.queue.max_length, *this),
|
2020-01-20 14:51:06 +01:00
|
|
|
outputs(pc, *this),
|
2019-05-08 18:39:00 +02:00
|
|
|
pc(*this, outputs,
|
|
|
|
instance.input_cache.get(),
|
2021-12-03 20:22:52 +01:00
|
|
|
config.player)
|
2016-03-05 19:05:04 +01:00
|
|
|
{
|
2016-12-03 13:56:25 +01:00
|
|
|
UpdateEffectiveReplayGainMode();
|
2016-03-05 19:05:04 +01:00
|
|
|
}
|
|
|
|
|
2018-01-29 23:53:52 +01:00
|
|
|
Partition::~Partition() noexcept = default;
|
|
|
|
|
2020-01-20 09:21:56 +01:00
|
|
|
void
|
|
|
|
Partition::BeginShutdown() noexcept
|
|
|
|
{
|
|
|
|
pc.Kill();
|
|
|
|
listener.reset();
|
|
|
|
}
|
|
|
|
|
2019-05-08 18:39:00 +02:00
|
|
|
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);
|
2019-05-08 18:39:00 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
cache.Prefetch(uri);
|
|
|
|
} catch (...) {
|
2021-06-24 20:22:48 +02:00
|
|
|
FmtError(cache_domain,
|
|
|
|
"Prefetch '{}' failed: {}",
|
|
|
|
uri, std::current_exception());
|
2019-05-08 18:39:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2016-11-24 15:18:57 +01:00
|
|
|
void
|
2019-05-31 13:54:52 +02:00
|
|
|
Partition::UpdateEffectiveReplayGainMode() noexcept
|
2016-11-24 15:18:57 +01:00
|
|
|
{
|
2016-12-03 13:56:25 +01:00
|
|
|
auto mode = replay_gain_mode;
|
2016-11-24 17:21:23 +01:00
|
|
|
if (mode == ReplayGainMode::AUTO)
|
2016-11-24 15:18:57 +01:00
|
|
|
mode = playlist.queue.random
|
2016-11-24 17:21:23 +01:00
|
|
|
? ReplayGainMode::TRACK
|
|
|
|
: ReplayGainMode::ALBUM;
|
2016-11-24 15:18:57 +01:00
|
|
|
|
2016-12-03 13:07:33 +01:00
|
|
|
pc.LockSetReplayGainMode(mode);
|
2016-11-25 12:51:55 +01:00
|
|
|
|
2016-11-24 15:18:57 +01:00
|
|
|
outputs.SetReplayGainMode(mode);
|
|
|
|
}
|
|
|
|
|
2014-02-01 00:44:41 +01:00
|
|
|
#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
|
|
|
{
|
2016-10-26 18:52:00 +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();
|
|
|
|
}
|
|
|
|
|
2013-10-22 00:26:20 +02:00
|
|
|
void
|
2019-05-31 13:54:52 +02:00
|
|
|
Partition::DatabaseModified(const Database &db) noexcept
|
2013-10-22 00:26:20 +02:00
|
|
|
{
|
2014-02-01 00:45:58 +01:00
|
|
|
playlist.DatabaseModified(db);
|
2016-03-05 19:16:39 +01:00
|
|
|
EmitIdle(IDLE_DATABASE);
|
2013-10-22 00:26:20 +02:00
|
|
|
}
|
|
|
|
|
2014-02-01 00:44:41 +01:00
|
|
|
#endif
|
|
|
|
|
2013-10-21 23:40:52 +02:00
|
|
|
void
|
2019-05-31 13:54:52 +02:00
|
|
|
Partition::TagModified() noexcept
|
2013-10-21 23:40:52 +02:00
|
|
|
{
|
2017-11-26 11:46:14 +01:00
|
|
|
auto song = pc.LockReadTaggedSong();
|
|
|
|
if (song)
|
2013-10-21 23:22:16 +02:00
|
|
|
playlist.TagModified(std::move(*song));
|
2013-10-21 23:40:52 +02:00
|
|
|
}
|
|
|
|
|
2018-01-29 12:02:14 +01:00
|
|
|
void
|
|
|
|
Partition::TagModified(const char *uri, const Tag &tag) noexcept
|
|
|
|
{
|
|
|
|
playlist.TagModified(uri, tag);
|
|
|
|
}
|
|
|
|
|
2013-10-21 23:40:52 +02:00
|
|
|
void
|
2019-05-31 13:54:52 +02:00
|
|
|
Partition::SyncWithPlayer() noexcept
|
2013-10-21 23:40:52 +02:00
|
|
|
{
|
|
|
|
playlist.SyncWithPlayer(pc);
|
2019-05-08 18:39:00 +02:00
|
|
|
|
|
|
|
/* TODO: invoke this function in batches, to let the hard disk
|
|
|
|
spin down in between */
|
|
|
|
PrefetchQueue();
|
2013-10-21 23:40:52 +02:00
|
|
|
}
|
2014-02-05 23:20:33 +01:00
|
|
|
|
2018-02-05 17:13:00 +01:00
|
|
|
void
|
2019-05-31 13:54:52 +02:00
|
|
|
Partition::BorderPause() noexcept
|
2018-02-05 17:13:00 +01:00
|
|
|
{
|
|
|
|
playlist.BorderPause(pc);
|
|
|
|
}
|
|
|
|
|
2016-03-10 20:10:14 +01:00
|
|
|
void
|
2019-05-31 13:57:46 +02:00
|
|
|
Partition::OnQueueModified() noexcept
|
2016-03-10 20:10:14 +01:00
|
|
|
{
|
|
|
|
EmitIdle(IDLE_PLAYLIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-05-31 13:57:46 +02:00
|
|
|
Partition::OnQueueOptionsChanged() noexcept
|
2016-03-10 20:10:14 +01:00
|
|
|
{
|
|
|
|
EmitIdle(IDLE_OPTIONS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-05-31 13:57:46 +02:00
|
|
|
Partition::OnQueueSongStarted() noexcept
|
2016-03-10 20:10:14 +01:00
|
|
|
{
|
|
|
|
EmitIdle(IDLE_PLAYER);
|
|
|
|
}
|
|
|
|
|
2022-07-13 13:55:27 +02:00
|
|
|
void
|
|
|
|
Partition::OnPlayerError() noexcept
|
|
|
|
{
|
|
|
|
EmitIdle(IDLE_PLAYER);
|
|
|
|
}
|
|
|
|
|
2022-07-13 13:34:45 +02:00
|
|
|
void
|
|
|
|
Partition::OnPlayerStateChanged() noexcept
|
|
|
|
{
|
|
|
|
EmitIdle(IDLE_PLAYER);
|
|
|
|
}
|
|
|
|
|
2014-02-21 08:55:52 +01:00
|
|
|
void
|
2017-11-26 12:39:09 +01:00
|
|
|
Partition::OnPlayerSync() noexcept
|
2014-02-21 08:55:52 +01:00
|
|
|
{
|
2016-03-10 23:10:14 +01:00
|
|
|
EmitGlobalEvent(SYNC_WITH_PLAYER);
|
2014-02-21 08:55:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2017-11-26 12:39:09 +01:00
|
|
|
Partition::OnPlayerTagModified() noexcept
|
2014-02-21 08:55:52 +01:00
|
|
|
{
|
2016-03-10 23:10:14 +01:00
|
|
|
EmitGlobalEvent(TAG_MODIFIED);
|
2022-07-13 13:51:28 +02:00
|
|
|
|
|
|
|
/* notify all clients that the tag of the current song has
|
|
|
|
changed */
|
|
|
|
EmitIdle(IDLE_PLAYER);
|
2014-02-21 08:55:52 +01:00
|
|
|
}
|
|
|
|
|
2018-02-05 17:13:00 +01:00
|
|
|
void
|
|
|
|
Partition::OnBorderPause() noexcept
|
|
|
|
{
|
|
|
|
EmitGlobalEvent(BORDER_PAUSE);
|
|
|
|
}
|
|
|
|
|
2014-02-05 23:20:33 +01:00
|
|
|
void
|
2019-05-31 14:09:47 +02:00
|
|
|
Partition::OnMixerVolumeChanged(Mixer &, int) noexcept
|
2014-02-05 23:20:33 +01:00
|
|
|
{
|
2022-08-08 23:15:09 +02:00
|
|
|
mixer_memento.InvalidateHardwareVolume();
|
2014-02-05 23:20:33 +01:00
|
|
|
|
|
|
|
/* notify clients */
|
2016-03-05 19:16:39 +01:00
|
|
|
EmitIdle(IDLE_MIXER);
|
2014-02-05 23:20:33 +01:00
|
|
|
}
|
2016-03-10 22:44:34 +01:00
|
|
|
|
2020-01-20 13:28:58 +01:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2016-03-10 22:44:34 +01:00
|
|
|
void
|
2019-08-02 14:44:00 +02:00
|
|
|
Partition::OnGlobalEvent(unsigned mask) noexcept
|
2016-03-10 22:44:34 +01:00
|
|
|
{
|
2016-03-10 23:10:14 +01:00
|
|
|
if ((mask & SYNC_WITH_PLAYER) != 0)
|
2016-03-10 22:44:34 +01:00
|
|
|
SyncWithPlayer();
|
2017-03-10 16:11:34 +01:00
|
|
|
|
|
|
|
if ((mask & TAG_MODIFIED) != 0)
|
|
|
|
TagModified();
|
2018-02-05 17:13:00 +01:00
|
|
|
|
|
|
|
if ((mask & BORDER_PAUSE) != 0)
|
|
|
|
BorderPause();
|
2016-03-10 22:44:34 +01:00
|
|
|
}
|