// SPDX-License-Identifier: GPL-2.0-or-later // Copyright The Music Player Daemon Project #ifndef MPD_INSTANCE_HXX #define MPD_INSTANCE_HXX #include "config.h" #include "event/Loop.hxx" #include "event/Thread.hxx" #include "event/MaskMonitor.hxx" #ifdef ENABLE_SYSTEMD_DAEMON #include "lib/systemd/Watchdog.hxx" #endif #ifdef ENABLE_CURL #include "RemoteTagCacheHandler.hxx" #endif #ifdef ENABLE_NEIGHBOR_PLUGINS #include "neighbor/Listener.hxx" class NeighborGlue; #endif #ifdef ENABLE_DATABASE #include "db/DatabaseListener.hxx" #include "db/Ptr.hxx" class Storage; class UpdateService; #ifdef ENABLE_INOTIFY class InotifyUpdate; #endif #endif #include #include class ClientList; struct Partition; class StateFile; class RemoteTagCache; class StickerDatabase; class StickerCleanupService; class InputCacheManager; /** * A utility class which, when used as the first base class, ensures * that the #EventLoop gets initialized before the other base classes. */ struct EventLoopHolder { EventLoop event_loop; }; struct Instance final : EventLoopHolder #if defined(ENABLE_DATABASE) || defined(ENABLE_NEIGHBOR_PLUGINS) , #endif #ifdef ENABLE_DATABASE public DatabaseListener #ifdef ENABLE_NEIGHBOR_PLUGINS , #endif #endif #ifdef ENABLE_NEIGHBOR_PLUGINS public NeighborListener #endif #ifdef ENABLE_CURL , public RemoteTagCacheHandler #endif { /** * A thread running an #EventLoop for non-blocking (bulk) I/O. */ EventThread io_thread{true}; /** * Another thread running an #EventLoop for non-blocking * (real-time) I/O. This is used instead of #io_thread for * events which require low latency, e.g. for filling hardware * ring buffers. */ EventThread rtio_thread; #ifdef ENABLE_SYSTEMD_DAEMON Systemd::Watchdog systemd_watchdog{event_loop}; #endif std::unique_ptr input_cache; /** * Monitor for global idle events to be broadcasted to all * partitions. */ MaskMonitor idle_monitor{event_loop, BIND_THIS_METHOD(OnIdle)}; #ifdef ENABLE_NEIGHBOR_PLUGINS std::unique_ptr neighbors; #endif #ifdef ENABLE_DATABASE DatabasePtr database; /** * This is really a #CompositeStorage. To avoid heavy include * dependencies, we declare it as just #Storage. */ Storage *storage = nullptr; UpdateService *update = nullptr; #ifdef ENABLE_INOTIFY std::unique_ptr inotify_update; #endif #endif #ifdef ENABLE_CURL std::unique_ptr remote_tag_cache; #endif std::unique_ptr client_list; std::list partitions; std::unique_ptr state_file; #ifdef ENABLE_SQLITE std::unique_ptr sticker_database; std::unique_ptr sticker_cleanup; bool need_sticker_cleanup = false; #endif Instance(); ~Instance() noexcept; /** * Wrapper for EventLoop::Break(). Call to initiate shutdown. */ void Break() noexcept { event_loop.Break(); } /** * Emit an "idle" event to all clients of all partitions. * * This method can be called from any thread. */ void EmitIdle(unsigned mask) noexcept { idle_monitor.OrMask(mask); } /** * Notify the #Instance that the state has been modified, and * the #StateFile may need to be saved. * * This method must be called from the main thread. */ void OnStateModified() noexcept; /** * Find a #Partition with the given name. Returns nullptr if * no such partition was found. */ [[gnu::pure]] Partition *FindPartition(const char *name) noexcept; void DeletePartition(Partition &partition) noexcept; void BeginShutdownPartitions() noexcept; #ifdef ENABLE_DATABASE /** * Returns the global #Database instance. May return nullptr * if this MPD configuration has no database (no * music_directory was configured). */ Database *GetDatabase() noexcept { return database.get(); } /** * Returns the global #Database instance. Throws * DatabaseError if this MPD configuration has no database (no * music_directory was configured). */ const Database &GetDatabaseOrThrow() const; #endif #ifdef ENABLE_SQLITE bool HasStickerDatabase() const noexcept { return sticker_database != nullptr; } void OnStickerCleanupDone(bool changed) noexcept; void StartStickerCleanup(); #endif void BeginShutdownUpdate() noexcept; #ifdef ENABLE_CURL void LookupRemoteTag(const char *uri) noexcept; #else void LookupRemoteTag(const char *) noexcept { /* no-op */ } #endif void FlushCaches() noexcept; void OnPlaylistDeleted(const char *name) const noexcept; private: #ifdef ENABLE_DATABASE /* virtual methods from class DatabaseListener */ void OnDatabaseModified() noexcept override; void OnDatabaseSongRemoved(const char *uri) noexcept override; #endif #ifdef ENABLE_NEIGHBOR_PLUGINS /* virtual methods from class NeighborListener */ void FoundNeighbor(const NeighborInfo &info) noexcept override; void LostNeighbor(const NeighborInfo &info) noexcept override; #endif #ifdef ENABLE_CURL /* virtual methods from class RemoteTagCacheHandler */ void OnRemoteTag(const char *uri, const Tag &tag) noexcept override; #endif /* callback for #idle_monitor */ void OnIdle(unsigned mask) noexcept; }; #endif