HART  0.2.0
High level Audio Regression and Testing
Loading...
Searching...
No Matches
hart_audio_buffer_signal.hpp
Go to the documentation of this file.
1#pragma once
2
4#include "signals/hart_signal.hpp"
5#include <memory>
6#include <sstream>
7
8// TODO: Better documentation
9
10namespace hart {
11
12/// @brief Plays audio from a pre-rendered AudioBuffer
13/// @ingroup Signals
14template<typename SampleType>
16 public Signal<SampleType, AudioBufferSignal<SampleType>>
17{
18public:
19 enum class Loop
20 {
21 yes,
22 no
23 };
24
25 /// @brief Construct from a shared_ptr (shared ownership)
26 explicit AudioBufferSignal (std::shared_ptr<AudioBuffer<SampleType>> buffer, Loop loop = Loop::no) :
27 m_buffer (std::move (buffer)),
28 m_loop (loop)
29 {
30 if (m_buffer == nullptr)
31 HART_THROW_OR_RETURN_VOID (hart::NullPointerError, "Null pointer buffer shouldn't be used to make an AudioBufferSignal");
32
33 this->setNumChannels (m_buffer->getNumChannels());
34 }
35
36 /// @brief Construct by copying a buffer (takes ownership of a copy)
37 explicit AudioBufferSignal (const AudioBuffer<SampleType>& buffer, Loop loop = Loop::no) :
38 m_buffer (std::make_shared<AudioBuffer<SampleType>> (buffer)),
39 m_loop (loop)
40 {
41 this->setNumChannels (m_buffer->getNumChannels());
42 }
43
44 /// @brief Construct by moving a buffer (takes ownership)
45 explicit AudioBufferSignal (AudioBuffer<SampleType>&& buffer, Loop loop = Loop::no) :
46 m_buffer (std::make_shared<AudioBuffer<SampleType>> (std::move (buffer))),
47 m_loop (loop)
48 {
49 this->setNumChannels (m_buffer->getNumChannels());
50 }
51
52 bool supportsNumChannels (size_t numChannels) const override
53 {
54 return m_buffer != nullptr && numChannels == m_buffer->getNumChannels();
55 }
56
57 bool supportsSampleRate(double sampleRateHz) const override
58 {
59 if (m_buffer == nullptr)
60 return false;
61
62 // If buffer has undefined sample rate, we'll accept any rate
63 return ! m_buffer->hasSampleRate() || m_buffer->getSampleRateHz() == sampleRateHz;
64 }
65
66 void prepare (double sampleRateHz, size_t numOutputChannels, size_t /*maxBlockSizeFrames*/) override
67 {
68 if (m_buffer == nullptr)
69 HART_THROW_OR_RETURN_VOID (hart::NullPointerError, "AudioBufferSignal: buffer is null");
70
71 if (numOutputChannels != m_buffer->getNumChannels())
72 HART_THROW_OR_RETURN_VOID (hart::ChannelLayoutError, "AudioBufferSignal: output channel count does not match buffer");
73
74 if (m_buffer->hasSampleRate() && m_buffer->getSampleRateHz() != sampleRateHz)
75 HART_THROW_OR_RETURN_VOID(hart::SampleRateError, "AudioBufferSignal: sample rate does not match buffer");
76 }
77
78 void renderNextBlock (AudioBuffer<SampleType>& output) override
79 {
80 if (m_buffer == nullptr)
81 {
82 hassertfalse; // No buffer to play audio from!
83 output.clear();
84 return;
85 }
86
87 const size_t numFrames = output.getNumFrames();
88 const size_t bufferTotalFrames = m_buffer->getNumFrames();
89 const size_t bufferNumChannels = m_buffer->getNumChannels();
90 size_t framesWritten = 0;
91
92 while (framesWritten < numFrames)
93 {
94 if (m_readPosition >= bufferTotalFrames)
95 {
96 if (m_loop == Loop::no)
97 break; // fill remaining with zeros below
98
99 m_readPosition = 0; // loop
100 }
101
102 const size_t framesAvailable = bufferTotalFrames - m_readPosition;
103 const size_t framesToCopy = std::min (framesAvailable, numFrames - framesWritten);
104
105 for (size_t channel = 0; channel < bufferNumChannels; ++channel)
106 {
107 const SampleType* src = (*m_buffer)[channel] + m_readPosition;
108 SampleType* dst = output[channel] + framesWritten;
109 std::copy(src, src + framesToCopy, dst);
110 }
111
112 framesWritten += framesToCopy;
113 m_readPosition += framesToCopy;
114 }
115
116 // Zero out any remaining frames (only when non‑looping and we ran out)
117 for (; framesWritten < numFrames; ++framesWritten)
118 for (size_t channel = 0; channel < bufferNumChannels; ++channel)
119 output[channel][framesWritten] = SampleType(0);
120 }
121
122 void reset() override
123 {
124 m_readPosition = 0;
125 }
126
127 void represent (std::ostream& stream) const override
128 {
129 stream << "AudioBufferSignal (";
130
131 if (m_buffer != nullptr)
132 m_buffer->represent (stream);
133 else
134 stream << "<nullptr>";
135
136 stream << ", Loop::" << (m_loop == Loop::yes ? "yes" : "no") << ')';
137 }
138
139 std::unique_ptr<SignalBase<SampleType>> copy() const override
140 {
141 return hart::make_unique<AudioBufferSignal> (m_buffer, m_loop);
142 }
143
144 std::unique_ptr<SignalBase<SampleType>> move() override
145 {
146 return hart::make_unique<AudioBufferSignal> (std::move (m_buffer), m_loop);
147 }
148
149private:
150 std::shared_ptr<AudioBuffer<SampleType>> m_buffer;
151 Loop m_loop = Loop::no;
152 size_t m_readPosition = 0;
153};
154
156
157} // namespace hart
Plays audio from a pre-rendered AudioBuffer.
void renderNextBlock(AudioBuffer< SampleType > &output) override
Renders next block audio for the signal.
bool supportsSampleRate(double sampleRateHz) const override
Tells whether this Signal supports given sample rate.
AudioBufferSignal(std::shared_ptr< AudioBuffer< SampleType > > buffer, Loop loop=Loop::no)
Construct from a shared_ptr (shared ownership)
AudioBufferSignal(const AudioBuffer< SampleType > &buffer, Loop loop=Loop::no)
Construct by copying a buffer (takes ownership of a copy)
void prepare(double sampleRateHz, size_t numOutputChannels, size_t) override
Prepare the signal for rendering.
void represent(std::ostream &stream) const override
Makes a text representation of this Signal for test failure outputs.
std::unique_ptr< SignalBase< SampleType > > move() override
Returns a smart pointer with a moved instance of this object.
bool supportsNumChannels(size_t numChannels) const override
Tells the host whether this Signal is capable of generating audio for a certain amount of channels.
std::unique_ptr< SignalBase< SampleType > > copy() const override
Returns a smart pointer with a copy of this object.
AudioBufferSignal(AudioBuffer< SampleType > &&buffer, Loop loop=Loop::no)
Construct by moving a buffer (takes ownership)
void reset() override
Resets the Signal to initial state.
Container for audio data.
Thrown when a numbers of channels is mismatched.
Thrown when a nullptr could be handled gracefully.
Thrown when sample rate is mismatched.
Polymorphic base for all signals.
Base class for signals.
#define HART_THROW_OR_RETURN_VOID(ExceptionType, message)
Throws an exception if HART_DO_NOT_THROW_EXCEPTIONS is set, prints a message and returns otherwise.
#define hassertfalse
Triggers a HartAssertException
#define HART_SIGNAL_DECLARE_ALIASES_FOR(ClassName)