/* * Copyright 2003-2018 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. */ /** \file * * Internal declarations for the "httpd" audio output plugin. */ #ifndef MPD_OUTPUT_HTTPD_INTERNAL_H #define MPD_OUTPUT_HTTPD_INTERNAL_H #include "HttpdClient.hxx" #include "output/Interface.hxx" #include "output/Timer.hxx" #include "thread/Mutex.hxx" #include "thread/Cond.hxx" #include "event/ServerSocket.hxx" #include "event/DeferEvent.hxx" #include "util/Cast.hxx" #include "util/Compiler.h" #include #include #include #include struct ConfigBlock; class EventLoop; class ServerSocket; class HttpdClient; class PreparedEncoder; class Encoder; struct Tag; class HttpdOutput final : AudioOutput, ServerSocket { /** * True if the audio output is open and accepts client * connections. */ bool open; bool pause; /** * The configured encoder plugin. */ std::unique_ptr prepared_encoder; Encoder *encoder = nullptr; /** * Number of bytes which were fed into the encoder, without * ever receiving new output. This is used to estimate * whether MPD should manually flush the encoder, to avoid * buffer underruns in the client. */ size_t unflushed_input = 0; public: /** * The MIME type produced by the #encoder. */ const char *content_type; /** * This mutex protects the listener socket and the client * list. */ mutable Mutex mutex; /** * This condition gets signalled when an item is removed from * #pages. */ Cond cond; private: /** * A #Timer object to synchronize this output with the * wallclock. */ Timer *timer; /** * The header page, which is sent to every client on connect. */ PagePtr header; /** * The metadata, which is sent to every client. */ PagePtr metadata; /** * The page queue, i.e. pages from the encoder to be * broadcasted to all clients. This container is necessary to * pass pages from the OutputThread to the IOThread. It is * protected by #mutex, and removing signals #cond. */ std::queue> pages; DeferEvent defer_broadcast; public: /** * The configured name. */ char const *name; /** * The configured genre. */ char const *genre; /** * The configured website address. */ char const *website; private: /** * A linked list containing all clients which are currently * connected. */ boost::intrusive::list> clients; /** * A temporary buffer for the httpd_output_read_page() * function. */ char buffer[32768]; /** * The maximum and current number of clients connected * at the same time. */ unsigned clients_max; public: HttpdOutput(EventLoop &_loop, const ConfigBlock &block); static AudioOutput *Create(EventLoop &event_loop, const ConfigBlock &block) { return new HttpdOutput(event_loop, block); } using ServerSocket::GetEventLoop; void Bind(); void Unbind() noexcept; void Enable() override { Bind(); } void Disable() noexcept override { Unbind(); } /** * Caller must lock the mutex. * * Throws #std::runtime_error on error. */ void OpenEncoder(AudioFormat &audio_format); /** * Caller must lock the mutex. */ void Open(AudioFormat &audio_format) override; /** * Caller must lock the mutex. */ void Close() noexcept override; /** * Check whether there is at least one client. * * Caller must lock the mutex. */ gcc_pure bool HasClients() const noexcept { return !clients.empty(); } /** * Check whether there is at least one client. */ gcc_pure bool LockHasClients() const noexcept { const std::lock_guard protect(mutex); return HasClients(); } void AddClient(UniqueSocketDescriptor fd) noexcept; /** * Removes a client from the httpd_output.clients linked list. */ void RemoveClient(HttpdClient &client) noexcept; /** * Sends the encoder header to the client. This is called * right after the response headers have been sent. */ void SendHeader(HttpdClient &client) const noexcept; gcc_pure std::chrono::steady_clock::duration Delay() const noexcept override; /** * Reads data from the encoder (as much as available) and * returns it as a new #page object. */ PagePtr ReadPage(); /** * Broadcasts a page struct to all clients. * * Mutext must not be locked. */ void BroadcastPage(PagePtr page) noexcept; /** * Broadcasts data from the encoder to all clients. */ void BroadcastFromEncoder(); /** * Throws #std::runtime_error on error. */ void EncodeAndPlay(const void *chunk, size_t size); void SendTag(const Tag &tag) override; size_t Play(const void *chunk, size_t size) override; void CancelAllClients() noexcept; void Cancel() noexcept override; bool Pause() override; private: /* DeferEvent callback */ void OnDeferredBroadcast() noexcept; void OnAccept(UniqueSocketDescriptor fd, SocketAddress address, int uid) noexcept override; }; extern const class Domain httpd_output_domain; #endif