14 if (data.empty() || data[0].empty()) {
18 uint64_t num_frames {};
24 std::cerr <<
"Insufficient channel data for planar organization" <<
'\n';
27 num_frames = data[0].size();
29 if (!std::ranges::all_of(data | std::views::drop(1) | std::views::take(
get_num_channels() - 1),
30 [num_frames](
const auto& span) {
return span.size() == num_frames; })) {
31 std::cerr <<
"Mismatched frame counts across channels" <<
'\n';
36 if (num_frames == 0) {
37 std::cerr <<
"Attempting to write to container with insufficient data for complete frame. Returning" <<
'\n';
41 if (uint64_t required_end_frame = start_frame + num_frames;
m_auto_resize) {
48 if (available_frames == 0) {
52 if (num_frames > available_frames) {
53 num_frames = available_frames;
71 std::cerr <<
"Channel index " << channel <<
" exceeds available channels (" <<
get_num_channels() <<
")" <<
'\n';
75 uint64_t num_frames = data.size();
76 uint64_t required_end_frame = start_frame + num_frames;
85 if (available_frames == 0) {
89 if (num_frames > available_frames) {
90 num_frames = available_frames;
104 auto num_frames =
validate(data, start_frame);
111 uint64_t frames_from_start = num_frames - frames_to_end;
113 if (frames_to_end > 0) {
114 std::vector<std::span<const double>> first_part;
115 first_part.reserve(data.size());
116 for (
const auto& span : data) {
117 first_part.emplace_back(span.subspan(0, frames_to_end));
122 if (frames_from_start > 0) {
123 std::vector<std::span<const double>> second_part;
124 second_part.reserve(data.size());
125 for (
const auto& span : data) {
126 second_part.emplace_back(span.subspan(frames_to_end, frames_from_start));
139 std::vector<DataVariant> data_variants;
143 data_variants.emplace_back(
144 std::vector<double>(data[0].begin(), data[0].begin() + samples_to_write));
148 | std::views::transform([num_frames](
const auto& span) ->
DataVariant {
149 return DataVariant(std::vector<double>(span.begin(), span.begin() + num_frames));
151 | std::ranges::to<std::vector>();
172 uint64_t frames_from_start = num_frames - frames_to_end;
174 if (frames_to_end > 0) {
175 write_frames(data.subspan(0, frames_to_end), start_frame, channel);
178 if (frames_from_start > 0) {
179 write_frames(data.subspan(frames_to_end, frames_from_start), 0, channel);
192 auto& interleaved_data = std::get<std::vector<double>>(
m_data[0]);
195 for (uint64_t frame = 0; frame < num_frames; ++frame) {
196 uint64_t interleaved_index = (start_frame + frame) * num_channels + channel;
197 if (interleaved_index < interleaved_data.size()) {
198 interleaved_data[interleaved_index] = data[frame];
204 if (channel >=
m_data.size()) {
208 auto dest_span = convert_variant<double>(
m_data[channel]);
210 if (start_frame + num_frames > dest_span.size()) {
211 std::vector<double> current_data(dest_span.begin(), dest_span.end());
212 current_data.resize(start_frame + num_frames, 0.0);
214 dest_span = convert_variant<double>(
m_data[channel]);
217 std::copy(data.begin(), data.begin() + num_frames,
218 dest_span.begin() + start_frame);
238 std::cerr <<
"Direct span access not supported for interleaved data. Use get_frames() or Kakshya::extract_channel_data() instead." <<
'\n';
244 if (channel >=
m_data.size()) {
254 uint64_t available_frames =
channel_data.size() - start_frame;
255 uint64_t actual_frames = std::min(num_frames, available_frames);
257 return {
channel_data.data() + start_frame, actual_frames };
266 uint64_t num_frames = output.size();
272 std::ranges::fill(output, 0.0);
276 const auto& interleaved_data = std::get<std::vector<double>>(
m_data[0]);
279 for (uint64_t frame = 0; frame < num_frames; ++frame) {
280 uint64_t interleaved_index = (start_frame + frame) * num_channels + channel;
281 if (interleaved_index < interleaved_data.size()) {
282 output[frame] = interleaved_data[interleaved_index];
290 if (channel >=
m_data.size()) {
291 std::ranges::fill(output, 0.0);
297 for (uint64_t frame = 0; frame < num_frames; ++frame) {
298 uint64_t data_index = start_frame + frame;
311 required_frames > current_frames) {
342 m_data.resize(data.size());
344 std::ranges::for_each(std::views::zip(data,
m_data),
346 auto&& [source, dest] = pair;
351 return static_cast<uint64_t
>(vec.size());
371 uint64_t new_capacity = std::max(target_frames, current_frames * 2);
380 std::vector<DataVariant> expanded_data(1);
385 std::vector<double> current_data;
388 std::vector<double> expanded_buffer(new_frame_count *
get_num_channels(), 0.0);
390 std::ranges::copy_n(current_data.begin(),
391 std::min<size_t>(current_data.size(), expanded_buffer.size()),
392 expanded_buffer.begin());
394 expanded_data[0] =
DataVariant(std::move(expanded_buffer));
396 return expanded_data;
400 | std::views::transform([
this, new_frame_count](uint32_t ch) ->
DataVariant {
402 std::vector<double> current_channel_data;
403 extract_from_variant(m_data[ch], current_channel_data);
405 std::vector<double> expanded_channel(new_frame_count, 0.0);
406 std::ranges::copy_n(current_channel_data.begin(),
407 std::min<size_t>(current_channel_data.size(), expanded_channel.size()),
408 expanded_channel.begin());
410 return { std::move(expanded_channel) };
413 return { std::vector<double>(new_frame_count, 0.0) };
415 | std::ranges::to<std::vector>();
uint64_t validate_single_channel(std::span< const double > data, uint64_t start_frame=0, uint32_t channel=0)
bool m_auto_resize
Enable automatic capacity expansion.
DynamicSoundStream(uint32_t sample_rate=48000, uint32_t num_channels=2)
Construct a DynamicSoundStream with specified audio parameters.
std::vector< DataVariant > create_expanded_data(uint64_t new_frame_count)
void expand_to(uint64_t target_frames)
bool m_is_circular
True when operating in circular buffer mode.
void enable_circular_buffer(uint64_t capacity)
Enable circular buffer mode with fixed capacity.
void ensure_capacity(uint64_t required_frames)
Pre-allocate capacity for the specified number of frames.
void set_all_data(const DataVariant &new_data)
void disable_circular_buffer()
Disable circular buffer mode and return to linear operation.
std::span< const double > get_channel_frames(uint32_t channel, uint64_t start_frame, uint64_t num_frames) const
Get the fixed capacity of the circular buffer if enabled.
uint64_t validate(std::vector< std::span< const double > > &data, uint64_t start_frame=0)
uint64_t write_frames(std::span< const double > data, uint64_t start_frame=0, uint32_t channel=0)
Write audio frame data to the container with automatic capacity management.
uint64_t m_circular_capacity
Fixed capacity for circular mode.
void set_looping(bool enable) override
Enable or disable looping behavior for the stream.
uint64_t get_total_elements() const override
Get the total number of elements in the container.
std::atomic< bool > m_double_extraction_dirty
std::shared_mutex m_data_mutex
void update_processing_state(ProcessingState new_state) override
Update the processing state of the container.
std::vector< DataVariant > m_data
void invalidate_span_cache()
Invalidate the span cache when data or layout changes.
virtual uint32_t get_num_channels() const
DataAccess channel_data(size_t channel) override
Get channel data with semantic interpretation.
uint64_t get_num_frames() const override
Get the number of frames in the primary (temporal) dimension.
void lock() override
Acquire a lock for thread-safe access.
void set_region_data(const Region ®ion, const std::vector< DataVariant > &data) override
Set data for a specific region.
void set_loop_region(const Region ®ion) override
Set the loop region using a Region.
ContainerDataStructure m_structure
Concrete base implementation for streaming audio containers.
@ READY
Container has data loaded and is ready for processing.
std::variant< std::vector< double >, std::vector< float >, std::vector< uint8_t >, std::vector< uint16_t >, std::vector< uint32_t >, std::vector< std::complex< float > >, std::vector< std::complex< double > >, std::vector< glm::vec2 >, std::vector< glm::vec3 >, std::vector< glm::vec4 >, std::vector< glm::mat4 > > DataVariant
Multi-type data storage for different precision needs.
@ INTERLEAVED
Single DataVariant with interleaved data (LRLRLR for stereo)
void safe_copy_data_variant(const DataVariant &input, DataVariant &output)
Safely copy data from a DataVariant to another DataVariant, handling type conversion.
std::span< T > extract_from_variant(const DataVariant &variant, std::vector< T > &storage, Utils::ComplexConversionStrategy strategy=Utils::ComplexConversionStrategy::MAGNITUDE)
Get typed span from DataVariant using concepts.
OrganizationStrategy organization
Represents a point or span in N-dimensional space.