From 45071607aa277dbbf3814e9a06c8371be75f27c5 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Tue, 12 Jul 2022 12:31:35 +0200 Subject: [PATCH] output/Interface: pass std::span to Play() --- src/output/Filtered.cxx | 6 +++--- src/output/Filtered.hxx | 9 +++++---- src/output/Interface.hxx | 5 +++-- src/output/Thread.cxx | 4 ++-- src/output/plugins/AlsaOutputPlugin.cxx | 20 +++++++++---------- src/output/plugins/AoOutputPlugin.cxx | 19 +++++++++--------- src/output/plugins/FifoOutputPlugin.cxx | 12 +++++------ src/output/plugins/HaikuOutputPlugin.cxx | 12 +++++------ src/output/plugins/JackOutputPlugin.cxx | 16 +++++++++------ src/output/plugins/NullOutputPlugin.cxx | 6 +++--- src/output/plugins/OSXOutputPlugin.cxx | 8 +++----- src/output/plugins/OpenALOutputPlugin.cxx | 10 +++++----- src/output/plugins/OssOutputPlugin.cxx | 19 +++++++----------- src/output/plugins/PipeOutputPlugin.cxx | 8 ++++---- src/output/plugins/PipeWireOutputPlugin.cxx | 8 ++++---- src/output/plugins/PulseOutputPlugin.cxx | 16 +++++++-------- src/output/plugins/RecorderOutputPlugin.cxx | 12 +++++------ src/output/plugins/ShoutOutputPlugin.cxx | 10 +++++----- src/output/plugins/SndioOutputPlugin.cxx | 6 ++---- src/output/plugins/SndioOutputPlugin.hxx | 2 +- src/output/plugins/SolarisOutputPlugin.cxx | 8 ++++---- src/output/plugins/WinmmOutputPlugin.cxx | 10 +++++----- src/output/plugins/httpd/HttpdInternal.hxx | 2 +- .../plugins/httpd/HttpdOutputPlugin.cxx | 14 ++++++------- src/output/plugins/sles/SlesOutputPlugin.cxx | 10 +++++----- src/output/plugins/snapcast/Internal.hxx | 2 +- .../plugins/snapcast/SnapcastOutputPlugin.cxx | 14 ++++++------- .../plugins/wasapi/WasapiOutputPlugin.cxx | 9 ++++----- test/run_output.cxx | 2 +- 29 files changed, 138 insertions(+), 141 deletions(-) diff --git a/src/output/Filtered.cxx b/src/output/Filtered.cxx index 7ac2ead16..3f687d9bf 100644 --- a/src/output/Filtered.cxx +++ b/src/output/Filtered.cxx @@ -173,10 +173,10 @@ FilteredAudioOutput::SendTag(const Tag &tag) output->SendTag(tag); } -size_t -FilteredAudioOutput::Play(const void *data, size_t size) +std::size_t +FilteredAudioOutput::Play(std::span src) { - return output->Play(data, size); + return output->Play(src); } void diff --git a/src/output/Filtered.hxx b/src/output/Filtered.hxx index 8219896a6..094d9bdde 100644 --- a/src/output/Filtered.hxx +++ b/src/output/Filtered.hxx @@ -23,10 +23,11 @@ #include "pcm/AudioFormat.hxx" #include "filter/Observer.hxx" -#include -#include -#include #include +#include +#include +#include +#include class FilterFactory; class PreparedFilter; @@ -224,7 +225,7 @@ public: void SendTag(const Tag &tag); - size_t Play(const void *data, size_t size); + std::size_t Play(std::span src); void Drain(); void Cancel() noexcept; diff --git a/src/output/Interface.hxx b/src/output/Interface.hxx index 8405e1f95..c66d8fb96 100644 --- a/src/output/Interface.hxx +++ b/src/output/Interface.hxx @@ -21,8 +21,9 @@ #define MPD_AUDIO_OUTPUT_INTERFACE_HXX #include -#include #include +#include +#include struct AudioFormat; struct Tag; @@ -180,7 +181,7 @@ public: * @return the number of bytes played (must be a multiple of * the frame size) */ - virtual size_t Play(const void *chunk, size_t size) = 0; + virtual std::size_t Play(std::span src) = 0; /** * Wait until the device has finished playing. diff --git a/src/output/Thread.cxx b/src/output/Thread.cxx index 48724e56e..e19b66130 100644 --- a/src/output/Thread.cxx +++ b/src/output/Thread.cxx @@ -265,7 +265,7 @@ AudioOutputControl::PlayChunk(std::unique_lock &lock) noexcept try { const ScopeUnlock unlock(mutex); - nbytes = output->Play(data.data(), data.size()); + nbytes = output->Play(data); assert(nbytes > 0); assert(nbytes <= data.size()); } catch (AudioOutputInterrupted) { @@ -380,7 +380,7 @@ static void PlayFull(FilteredAudioOutput &output, std::span buffer) { while (!buffer.empty()) { - size_t nbytes = output.Play(buffer.data(), buffer.size()); + size_t nbytes = output.Play(buffer); assert(nbytes > 0); buffer = buffer.subspan(nbytes); diff --git a/src/output/plugins/AlsaOutputPlugin.cxx b/src/output/plugins/AlsaOutputPlugin.cxx index db209416e..21a7bff92 100644 --- a/src/output/plugins/AlsaOutputPlugin.cxx +++ b/src/output/plugins/AlsaOutputPlugin.cxx @@ -283,7 +283,7 @@ private: void Interrupt() noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; bool Pause() noexcept override; @@ -1219,26 +1219,26 @@ AlsaOutput::LockWaitWriteAvailable() } } -size_t -AlsaOutput::Play(const void *chunk, size_t size) +std::size_t +AlsaOutput::Play(std::span src) { - assert(size > 0); - assert(size % in_frame_size == 0); + assert(!src.empty()); + assert(src.size() % in_frame_size == 0); const size_t max_frames = LockWaitWriteAvailable(); const size_t max_size = max_frames * in_frame_size; - if (size > max_size) - size = max_size; + if (src.size() > max_size) + src = src.first(max_size); - const auto e = pcm_export->Export({(const std::byte *)chunk, size}); + const auto e = pcm_export->Export(src); if (e.empty()) - return size; + return src.size(); size_t bytes_written = ring_buffer->push(e.data(), e.size()); assert(bytes_written == e.size()); (void)bytes_written; - return size; + return src.size(); } Event::Duration diff --git a/src/output/plugins/AoOutputPlugin.cxx b/src/output/plugins/AoOutputPlugin.cxx index e4c938c2d..4bfb3b649 100644 --- a/src/output/plugins/AoOutputPlugin.cxx +++ b/src/output/plugins/AoOutputPlugin.cxx @@ -74,7 +74,7 @@ public: void Open(AudioFormat &audio_format) override; void Close() noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; }; static constexpr Domain ao_output_domain("ao_output"); @@ -195,23 +195,24 @@ AoOutput::Close() noexcept ao_close(device); } -size_t -AoOutput::Play(const void *chunk, size_t size) +std::size_t +AoOutput::Play(std::span src) { - assert(size % frame_size == 0); + assert(src.size() % frame_size == 0); - if (size > max_size) - size = max_size; + if (src.size() > max_size) + /* round down to a multiple of the frame size */ + src = src.first(max_size); /* For whatever reason, libao wants a non-const pointer. Let's hope it does not write to the buffer, and use the union deconst hack to * work around this API misdesign. */ - char *data = const_cast((const char *)chunk); + char *data = const_cast((const char *)src.data()); - if (ao_play(device, data, size) == 0) + if (ao_play(device, data, src.size()) == 0) throw MakeAoError(); - return size; + return src.size(); } const struct AudioOutputPlugin ao_output_plugin = { diff --git a/src/output/plugins/FifoOutputPlugin.cxx b/src/output/plugins/FifoOutputPlugin.cxx index 12869cca9..ca8ce5959 100644 --- a/src/output/plugins/FifoOutputPlugin.cxx +++ b/src/output/plugins/FifoOutputPlugin.cxx @@ -69,7 +69,7 @@ private: void Close() noexcept override; [[nodiscard]] std::chrono::steady_clock::duration Delay() const noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; }; @@ -208,17 +208,17 @@ FifoOutput::Delay() const noexcept : std::chrono::steady_clock::duration::zero(); } -size_t -FifoOutput::Play(const void *chunk, size_t size) +std::size_t +FifoOutput::Play(std::span src) { if (!timer->IsStarted()) timer->Start(); - timer->Add(size); + timer->Add(src.size()); while (true) { - ssize_t bytes = write(output, chunk, size); + ssize_t bytes = write(output, src.data(), src.size()); if (bytes > 0) - return (size_t)bytes; + return (std::size_t)bytes; if (bytes < 0) { switch (errno) { diff --git a/src/output/plugins/HaikuOutputPlugin.cxx b/src/output/plugins/HaikuOutputPlugin.cxx index d2d6fc807..bb689b8c4 100644 --- a/src/output/plugins/HaikuOutputPlugin.cxx +++ b/src/output/plugins/HaikuOutputPlugin.cxx @@ -75,7 +75,7 @@ private: void Open(AudioFormat &audio_format) override; void Close() noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; std::chrono::steady_clock::duration Delay() const noexcept override; @@ -256,17 +256,17 @@ HaikuOutput::Open(AudioFormat &audio_format) sound_player->SetHasData(false); } -size_t -HaikuOutput::Play(const void *chunk, size_t size) +std::size_t +HaikuOutput::Play(std::span src) { BSoundPlayer* const soundPlayer = sound_player; - const uint8 *data = (const uint8 *)chunk; + const uint8 *data = (const uint8 *)src.data(); if (!soundPlayer->HasData()) soundPlayer->SetHasData(true); acquire_sem(new_buffer); - size_t bytesLeft = size; + size_t bytesLeft = src.size(); while (bytesLeft > 0) { if (buffer_filled == buffer_size) { // Request another buffer from BSoundPlayer @@ -293,7 +293,7 @@ HaikuOutput::Play(const void *chunk, size_t size) //soundPlayer->SetHasData(false); } - return size; + return src.size(); } inline std::chrono::steady_clock::duration diff --git a/src/output/plugins/JackOutputPlugin.cxx b/src/output/plugins/JackOutputPlugin.cxx index f8069a30d..721154ce0 100644 --- a/src/output/plugins/JackOutputPlugin.cxx +++ b/src/output/plugins/JackOutputPlugin.cxx @@ -26,6 +26,7 @@ #include "util/ScopeExit.hxx" #include "util/IterableSplitString.hxx" #include "util/RuntimeError.hxx" +#include "util/SpanCast.hxx" #include "util/Domain.hxx" #include "Log.hxx" @@ -178,7 +179,7 @@ public: : std::chrono::steady_clock::duration::zero(); } - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; bool Pause() override; @@ -689,14 +690,17 @@ JackOutput::WriteSamples(const float *src, size_t n_frames) return result; } -inline size_t -JackOutput::Play(const void *chunk, size_t size) +std::size_t +JackOutput::Play(std::span _src) { + const auto src = FromBytesStrict(_src); + pause = false; const size_t frame_size = audio_format.GetFrameSize(); - assert(size % frame_size == 0); - size /= frame_size; + assert(src.size() % frame_size == 0); + + const std::size_t n_frames = src.size() / audio_format.channels; while (true) { { @@ -709,7 +713,7 @@ JackOutput::Play(const void *chunk, size_t size) } size_t frames_written = - WriteSamples((const float *)chunk, size); + WriteSamples(src.data(), n_frames); if (frames_written > 0) return frames_written * frame_size; diff --git a/src/output/plugins/NullOutputPlugin.cxx b/src/output/plugins/NullOutputPlugin.cxx index ace57539a..6bf3739ba 100644 --- a/src/output/plugins/NullOutputPlugin.cxx +++ b/src/output/plugins/NullOutputPlugin.cxx @@ -53,14 +53,14 @@ private: : std::chrono::steady_clock::duration::zero(); } - size_t Play([[maybe_unused]] const void *chunk, size_t size) override { + size_t Play(std::span src) override { if (sync) { if (!timer->IsStarted()) timer->Start(); - timer->Add(size); + timer->Add(src.size()); } - return size; + return src.size(); } void Cancel() noexcept override { diff --git a/src/output/plugins/OSXOutputPlugin.cxx b/src/output/plugins/OSXOutputPlugin.cxx index ab809ae43..444d9e804 100644 --- a/src/output/plugins/OSXOutputPlugin.cxx +++ b/src/output/plugins/OSXOutputPlugin.cxx @@ -112,7 +112,7 @@ private: void Close() noexcept override; std::chrono::steady_clock::duration Delay() const noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; bool Pause() override; void Cancel() noexcept override; }; @@ -770,15 +770,13 @@ OSXOutput::Open(AudioFormat &audio_format) started = false; } -size_t -OSXOutput::Play(const void *chunk, size_t size) +std::size_t +OSXOutput::Play(std::span input) { assert(size > 0); pause = false; - std::span input((const std::byte *)chunk, size); - #ifdef ENABLE_DSD if (dop_enabled) { input = pcm_export->Export(input); diff --git a/src/output/plugins/OpenALOutputPlugin.cxx b/src/output/plugins/OpenALOutputPlugin.cxx index 95864273b..dcceb19a4 100644 --- a/src/output/plugins/OpenALOutputPlugin.cxx +++ b/src/output/plugins/OpenALOutputPlugin.cxx @@ -69,7 +69,7 @@ private: : std::chrono::milliseconds(50); } - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; @@ -180,8 +180,8 @@ OpenALOutput::Close() noexcept alcCloseDevice(device); } -size_t -OpenALOutput::Play(const void *chunk, size_t size) +std::size_t +OpenALOutput::Play(std::span src) { if (alcGetCurrentContext() != context) alcMakeContextCurrent(context); @@ -199,13 +199,13 @@ OpenALOutput::Play(const void *chunk, size_t size) alSourceUnqueueBuffers(source, 1, &buffer); } - alBufferData(buffer, format, chunk, size, frequency); + alBufferData(buffer, format, src.data(), src.size(), frequency); alSourceQueueBuffers(source, 1, &buffer); if (!IsPlaying()) alSourcePlay(source); - return size; + return src.size(); } void diff --git a/src/output/plugins/OssOutputPlugin.cxx b/src/output/plugins/OssOutputPlugin.cxx index 25a7723d7..3ff68a3d3 100644 --- a/src/output/plugins/OssOutputPlugin.cxx +++ b/src/output/plugins/OssOutputPlugin.cxx @@ -114,7 +114,7 @@ public: DoClose(); } - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; private: @@ -679,26 +679,21 @@ OssOutput::Cancel() noexcept pcm_export->Reset(); } -size_t -OssOutput::Play(const void *chunk, size_t size) +std::size_t +OssOutput::Play(std::span src) { - ssize_t ret; - - assert(size > 0); + assert(!src.empty()); /* reopen the device since it was closed by dropBufferedAudio */ if (!fd.IsDefined()) Reopen(); - const auto e = pcm_export->Export({(const std::byte *)chunk, size}); + const auto e = pcm_export->Export(src); if (e.empty()) - return size; - - chunk = e.data(); - size = e.size(); + return src.size(); while (true) { - ret = fd.Write(chunk, size); + const ssize_t ret = fd.Write(e.data(), e.size()); if (ret > 0) return pcm_export->CalcInputSize(ret); diff --git a/src/output/plugins/PipeOutputPlugin.cxx b/src/output/plugins/PipeOutputPlugin.cxx index 3d2e53ce5..d18b5155d 100644 --- a/src/output/plugins/PipeOutputPlugin.cxx +++ b/src/output/plugins/PipeOutputPlugin.cxx @@ -45,7 +45,7 @@ private: pclose(fh); } - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; }; PipeOutput::PipeOutput(const ConfigBlock &block) @@ -64,10 +64,10 @@ PipeOutput::Open([[maybe_unused]] AudioFormat &audio_format) throw FormatErrno("Error opening pipe \"%s\"", cmd.c_str()); } -inline size_t -PipeOutput::Play(const void *chunk, size_t size) +std::size_t +PipeOutput::Play(std::span src) { - size_t nbytes = fwrite(chunk, 1, size, fh); + size_t nbytes = fwrite(src.data(), 1, src.size(), fh); if (nbytes == 0) throw MakeErrno("Write error on pipe"); diff --git a/src/output/plugins/PipeWireOutputPlugin.cxx b/src/output/plugins/PipeWireOutputPlugin.cxx index ff0177f89..0c36dd98c 100644 --- a/src/output/plugins/PipeWireOutputPlugin.cxx +++ b/src/output/plugins/PipeWireOutputPlugin.cxx @@ -290,7 +290,7 @@ private: } [[nodiscard]] std::chrono::steady_clock::duration Delay() const noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; @@ -828,8 +828,8 @@ PipeWireOutput::Delay() const noexcept return result; } -size_t -PipeWireOutput::Play(const void *chunk, size_t size) +std::size_t +PipeWireOutput::Play(std::span src) { const PipeWire::ThreadLoopLock lock(thread_loop); @@ -839,7 +839,7 @@ PipeWireOutput::Play(const void *chunk, size_t size) CheckThrowError(); std::size_t bytes_written = - ring_buffer->push((const std::byte *)chunk, size); + ring_buffer->push(src.data(), src.size()); if (bytes_written > 0) { drained = false; return bytes_written; diff --git a/src/output/plugins/PulseOutputPlugin.cxx b/src/output/plugins/PulseOutputPlugin.cxx index ab1e8c551..9454a9a98 100644 --- a/src/output/plugins/PulseOutputPlugin.cxx +++ b/src/output/plugins/PulseOutputPlugin.cxx @@ -110,7 +110,7 @@ public: void Interrupt() noexcept override; [[nodiscard]] std::chrono::steady_clock::duration Delay() const noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; bool Pause() override; @@ -775,8 +775,8 @@ PulseOutput::Delay() const noexcept return result; } -size_t -PulseOutput::Play(const void *chunk, size_t size) +std::size_t +PulseOutput::Play(std::span src) { assert(mainloop != nullptr); assert(stream != nullptr); @@ -811,18 +811,18 @@ PulseOutput::Play(const void *chunk, size_t size) /* now write */ - if (size > writable) + if (src.size() > writable) /* don't send more than possible */ - size = writable; + src = src.first(writable); - writable -= size; + writable -= src.size(); - int result = pa_stream_write(stream, chunk, size, nullptr, + int result = pa_stream_write(stream, src.data(), src.size(), nullptr, 0, PA_SEEK_RELATIVE); if (result < 0) throw MakePulseError(context, "pa_stream_write() failed"); - return size; + return src.size(); } void diff --git a/src/output/plugins/RecorderOutputPlugin.cxx b/src/output/plugins/RecorderOutputPlugin.cxx index 9b2112339..306212c47 100644 --- a/src/output/plugins/RecorderOutputPlugin.cxx +++ b/src/output/plugins/RecorderOutputPlugin.cxx @@ -86,7 +86,7 @@ private: void SendTag(const Tag &tag) override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; [[nodiscard]] gcc_pure bool HasDynamicPath() const noexcept { @@ -322,22 +322,22 @@ RecorderOutput::SendTag(const Tag &tag) encoder->SendTag(tag); } -size_t -RecorderOutput::Play(const void *chunk, size_t size) +std::size_t +RecorderOutput::Play(std::span src) { if (file == nullptr) { /* not currently encoding to a file; discard incoming data */ assert(HasDynamicPath()); assert(path.IsNull()); - return size; + return src.size(); } - encoder->Write({(const std::byte *)chunk, size}); + encoder->Write(src); EncoderToFile(); - return size; + return src.size(); } const struct AudioOutputPlugin recorder_output_plugin = { diff --git a/src/output/plugins/ShoutOutputPlugin.cxx b/src/output/plugins/ShoutOutputPlugin.cxx index a338d66d2..74b6ee7cb 100644 --- a/src/output/plugins/ShoutOutputPlugin.cxx +++ b/src/output/plugins/ShoutOutputPlugin.cxx @@ -88,7 +88,7 @@ struct ShoutOutput final : AudioOutput { [[nodiscard]] std::chrono::steady_clock::duration Delay() const noexcept override; void SendTag(const Tag &tag) override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; bool Pause() override; @@ -413,12 +413,12 @@ ShoutOutput::Delay() const noexcept return std::chrono::milliseconds(delay); } -size_t -ShoutOutput::Play(const void *chunk, size_t size) +std::size_t +ShoutOutput::Play(std::span src) { - encoder->Write({(const std::byte *)chunk, size}); + encoder->Write(src); WritePage(); - return size; + return src.size(); } bool diff --git a/src/output/plugins/SndioOutputPlugin.cxx b/src/output/plugins/SndioOutputPlugin.cxx index 5ae18347c..c69f47747 100644 --- a/src/output/plugins/SndioOutputPlugin.cxx +++ b/src/output/plugins/SndioOutputPlugin.cxx @@ -144,11 +144,9 @@ SndioOutput::Close() noexcept } size_t -SndioOutput::Play(const void *chunk, size_t size) +SndioOutput::Play(std::span src) { - size_t n; - - n = sio_write(hdl, chunk, size); + const std::size_t n = sio_write(hdl, src.data(), src.size()); if (n == 0 && sio_eof(hdl) != 0) throw std::runtime_error("sndio write failed"); return n; diff --git a/src/output/plugins/SndioOutputPlugin.hxx b/src/output/plugins/SndioOutputPlugin.hxx index 415af3d53..15d76da18 100644 --- a/src/output/plugins/SndioOutputPlugin.hxx +++ b/src/output/plugins/SndioOutputPlugin.hxx @@ -49,7 +49,7 @@ public: private: void Open(AudioFormat &audio_format) override; void Close() noexcept override; - size_t Play(const void *chunk, size_t size) override; + size_t Play(std::span src) override; }; #endif diff --git a/src/output/plugins/SolarisOutputPlugin.cxx b/src/output/plugins/SolarisOutputPlugin.cxx index 1b3cd5d5e..178b13bd1 100644 --- a/src/output/plugins/SolarisOutputPlugin.cxx +++ b/src/output/plugins/SolarisOutputPlugin.cxx @@ -76,7 +76,7 @@ private: void Open(AudioFormat &audio_format) override; void Close() noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Cancel() noexcept override; }; @@ -139,10 +139,10 @@ SolarisOutput::Close() noexcept fd.Close(); } -size_t -SolarisOutput::Play(const void *chunk, size_t size) +std::size_t +SolarisOutput::Play(std::span src) { - ssize_t nbytes = fd.Write(chunk, size); + ssize_t nbytes = fd.Write(src.data(), src.size()); if (nbytes <= 0) throw MakeErrno("Write failed"); diff --git a/src/output/plugins/WinmmOutputPlugin.cxx b/src/output/plugins/WinmmOutputPlugin.cxx index 66b3dc3b5..5426974f8 100644 --- a/src/output/plugins/WinmmOutputPlugin.cxx +++ b/src/output/plugins/WinmmOutputPlugin.cxx @@ -69,7 +69,7 @@ private: void Open(AudioFormat &audio_format) override; void Close() noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; @@ -256,13 +256,13 @@ WinmmOutput::DrainBuffer(WinmmBuffer &buffer) } } -size_t -WinmmOutput::Play(const void *chunk, size_t size) +std::size_t +WinmmOutput::Play(std::span src) { /* get the next buffer from the ring and prepare it */ WinmmBuffer *buffer = &buffers[next_buffer]; DrainBuffer(*buffer); - winmm_set_buffer(handle, buffer, chunk, size); + winmm_set_buffer(handle, buffer, src.data(), src.size()); /* enqueue the buffer */ MMRESULT result = waveOutWrite(handle, &buffer->hdr, @@ -276,7 +276,7 @@ WinmmOutput::Play(const void *chunk, size_t size) /* mark our buffer as "used" */ next_buffer = (next_buffer + 1) % buffers.size(); - return size; + return src.size(); } void diff --git a/src/output/plugins/httpd/HttpdInternal.hxx b/src/output/plugins/httpd/HttpdInternal.hxx index b56f2702a..b2e5264e0 100644 --- a/src/output/plugins/httpd/HttpdInternal.hxx +++ b/src/output/plugins/httpd/HttpdInternal.hxx @@ -254,7 +254,7 @@ public: void SendTag(const Tag &tag) override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; /** * Mutext must not be locked. diff --git a/src/output/plugins/httpd/HttpdOutputPlugin.cxx b/src/output/plugins/httpd/HttpdOutputPlugin.cxx index acb5889e9..a897263c8 100644 --- a/src/output/plugins/httpd/HttpdOutputPlugin.cxx +++ b/src/output/plugins/httpd/HttpdOutputPlugin.cxx @@ -309,19 +309,19 @@ HttpdOutput::EncodeAndPlay(std::span src) BroadcastFromEncoder(); } -size_t -HttpdOutput::Play(const void *chunk, size_t size) +std::size_t +HttpdOutput::Play(std::span src) { pause = false; if (LockHasClients()) - EncodeAndPlay({(const std::byte *)chunk, size}); + EncodeAndPlay(src); if (!timer->IsStarted()) timer->Start(); - timer->Add(size); + timer->Add(src.size()); - return size; + return src.size(); } bool @@ -330,8 +330,8 @@ HttpdOutput::Pause() pause = true; if (LockHasClients()) { - static const char silence[1020] = { 0 }; - Play(silence, sizeof(silence)); + static constexpr std::byte silence[1020]{}; + Play(std::span{silence}); } return true; diff --git a/src/output/plugins/sles/SlesOutputPlugin.cxx b/src/output/plugins/sles/SlesOutputPlugin.cxx index cdb4d4b88..b0fbc31d6 100644 --- a/src/output/plugins/sles/SlesOutputPlugin.cxx +++ b/src/output/plugins/sles/SlesOutputPlugin.cxx @@ -99,7 +99,7 @@ private: : std::chrono::steady_clock::duration::zero(); } - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; @@ -333,8 +333,8 @@ SlesOutput::Close() noexcept engine_object.Destroy(); } -size_t -SlesOutput::Play(const void *chunk, size_t size) +std::size_t +SlesOutput::Play(std::span src) { cancel = false; @@ -356,8 +356,8 @@ SlesOutput::Play(const void *chunk, size_t size) return ret; }); - size_t nbytes = std::min(BUFFER_SIZE - filled, size); - memcpy(buffers[next] + filled, chunk, nbytes); + size_t nbytes = std::min(BUFFER_SIZE - filled, src.size()); + memcpy(buffers[next] + filled, src.data(), nbytes); filled += nbytes; if (filled < BUFFER_SIZE) return nbytes; diff --git a/src/output/plugins/snapcast/Internal.hxx b/src/output/plugins/snapcast/Internal.hxx index 54fca7430..5f464ae60 100644 --- a/src/output/plugins/snapcast/Internal.hxx +++ b/src/output/plugins/snapcast/Internal.hxx @@ -183,7 +183,7 @@ public: void SendTag(const Tag &tag) override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; diff --git a/src/output/plugins/snapcast/SnapcastOutputPlugin.cxx b/src/output/plugins/snapcast/SnapcastOutputPlugin.cxx index f7d128b2f..64610c4e0 100644 --- a/src/output/plugins/snapcast/SnapcastOutputPlugin.cxx +++ b/src/output/plugins/snapcast/SnapcastOutputPlugin.cxx @@ -297,8 +297,8 @@ SnapcastOutput::SendTag(const Tag &tag) #endif } -size_t -SnapcastOutput::Play(const void *chunk, size_t size) +std::size_t +SnapcastOutput::Play(std::span src) { pause = false; @@ -306,13 +306,13 @@ SnapcastOutput::Play(const void *chunk, size_t size) if (!timer->IsStarted()) timer->Start(); - timer->Add(size); + timer->Add(src.size()); if (!LockHasClients()) - return size; + return src.size(); - encoder->Write({(const std::byte *)chunk, size}); - unflushed_input += size; + encoder->Write(src); + unflushed_input += src.size(); if (unflushed_input >= 65536) { /* we have fed a lot of input into the encoder, but it @@ -343,7 +343,7 @@ SnapcastOutput::Play(const void *chunk, size_t size) chunks.push(std::make_shared(now, AllocatedArray{payload})); } - return size; + return src.size(); } bool diff --git a/src/output/plugins/wasapi/WasapiOutputPlugin.cxx b/src/output/plugins/wasapi/WasapiOutputPlugin.cxx index 3fc4de1b5..dde4e6d5c 100644 --- a/src/output/plugins/wasapi/WasapiOutputPlugin.cxx +++ b/src/output/plugins/wasapi/WasapiOutputPlugin.cxx @@ -357,7 +357,7 @@ public: } void Close() noexcept override; std::chrono::steady_clock::duration Delay() const noexcept override; - size_t Play(const void *chunk, size_t size) override; + std::size_t Play(std::span src) override; void Drain() override; void Cancel() noexcept override; bool Pause() override; @@ -707,8 +707,8 @@ WasapiOutput::Delay() const noexcept return std::chrono::steady_clock::duration::zero(); } -size_t -WasapiOutput::Play(const void *chunk, size_t size) +std::size_t +WasapiOutput::Play(std::span input) { assert(thread); @@ -716,12 +716,11 @@ WasapiOutput::Play(const void *chunk, size_t size) not_interrupted.test_and_set(); - std::span input{(const std::byte*)chunk, size}; if (pcm_export) { input = pcm_export->Export(input); } if (input.empty()) - return size; + return input.size(); do { const size_t consumed_size = thread->Push(input); diff --git a/test/run_output.cxx b/test/run_output.cxx index 53010bc2f..ba6d50836 100644 --- a/test/run_output.cxx +++ b/test/run_output.cxx @@ -155,7 +155,7 @@ RunOutput(AudioOutput &ao, AudioFormat audio_format, if (src.empty()) continue; - size_t consumed = ao.Play(src.data(), src.size()); + size_t consumed = ao.Play(src); assert(consumed <= src.size()); assert(consumed % in_frame_size == 0);