/*** This file is part of snapcast Copyright (C) 2014-2024 Johannes Pohl Copyright (C) 2024 Marcus Weseloh 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 3 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, see . ***/ #pragma once // local headers #include "pcm_stream.hpp" // 3rd party headers #include #include #include #include namespace streamreader { /// Reads and decodes PCM data from a Jack server /** * Reads PCM from a Jack server and passes the data to an encoder. * Implements EncoderListener to get the encoded data. * Data is passed to the PcmStream::Listener */ class JackStream : public PcmStream { public: /// ctor. Encoded PCM data is passed to the PipeListener JackStream(PcmStream::Listener* pcmListener, boost::asio::io_context& ioc, const ServerSettings& server_settings, const StreamUri& uri); void start() override; void stop() override; protected: bool openJackConnection(); void closeJackConnection(); bool createJackPorts(); void tryConnect(); int readJackBuffers(jack_nframes_t nframes); void onJackPortRegistration(jack_port_id_t port_id, int registered); void onJackShutdown(); void autoConnectPorts(); std::string serverName_; jack_client_t* client_; std::vector ports_; SampleFormat jackSampleFormat_; std::string autoConnectRegex_; bool doAutoConnect_ = false; int autoConnectSkip_; std::function interleave_func_; boost::asio::steady_timer read_timer_; std::chrono::microseconds silence_; /// send silent chunks to clients bool send_silence_; /// silence duration before switching the stream to idle std::chrono::milliseconds idle_threshold_; /// first read after idle => reset chunk timestamp bool first_; }; } // namespace streamreader