2017-12-19 21:08:10 -05:00
|
|
|
//
|
|
|
|
// CompoundSource.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 19/12/2017.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2017 Thomas Harte. All rights reserved.
|
2017-12-19 21:08:10 -05:00
|
|
|
//
|
|
|
|
|
2024-01-16 23:34:46 -05:00
|
|
|
#pragma once
|
2017-12-19 21:08:10 -05:00
|
|
|
|
2024-02-09 14:25:40 -05:00
|
|
|
#include "BufferSource.hpp"
|
2017-12-19 21:08:10 -05:00
|
|
|
|
2024-02-08 10:47:05 -05:00
|
|
|
#include <algorithm>
|
2018-04-07 14:30:02 -04:00
|
|
|
#include <cassert>
|
2017-12-19 21:08:10 -05:00
|
|
|
#include <cstring>
|
2020-05-09 17:57:21 -04:00
|
|
|
#include <atomic>
|
2017-12-19 21:08:10 -05:00
|
|
|
|
2023-05-10 16:02:18 -05:00
|
|
|
namespace Outputs::Speaker {
|
2017-12-19 21:08:10 -05:00
|
|
|
|
2024-02-08 11:10:08 -05:00
|
|
|
/// @returns @c true if any of the templated sources is stereo; @c false otherwise.
|
|
|
|
template <typename... S> constexpr bool is_stereo() {
|
|
|
|
bool is_stereo = false;
|
|
|
|
([&] {
|
|
|
|
is_stereo |= S::is_stereo;
|
|
|
|
}(), ...);
|
|
|
|
return is_stereo;
|
|
|
|
}
|
|
|
|
|
2024-02-09 14:34:59 -05:00
|
|
|
/// @returns @c true if the variadic template arguments are ordered as all stereo sources followed by
|
|
|
|
/// all mono; @c false otherwise.
|
|
|
|
template <typename... S> constexpr bool are_properly_ordered() {
|
|
|
|
bool is_ordered = true;
|
|
|
|
bool is_stereo = true;
|
|
|
|
([&] {
|
|
|
|
if(S::is_stereo && !is_stereo) {
|
|
|
|
is_ordered = false;
|
|
|
|
}
|
|
|
|
is_stereo &= S::is_stereo;
|
|
|
|
}(), ...);
|
|
|
|
return is_ordered;
|
|
|
|
}
|
|
|
|
|
2017-12-19 21:08:10 -05:00
|
|
|
/*!
|
|
|
|
A CompoundSource adds together the sound generated by multiple individual SampleSources.
|
2018-04-07 14:30:02 -04:00
|
|
|
An owner may optionally assign relative volumes.
|
2017-12-19 21:08:10 -05:00
|
|
|
*/
|
2018-04-07 14:30:02 -04:00
|
|
|
template <typename... T> class CompoundSource:
|
2024-02-09 14:25:40 -05:00
|
|
|
public Outputs::Speaker::BufferSource<CompoundSource<T...>, ::Outputs::Speaker::is_stereo<T...>()> {
|
2018-04-07 14:30:02 -04:00
|
|
|
private:
|
2024-02-08 12:07:12 -05:00
|
|
|
template <typename... S> class CompoundSourceHolder {
|
2018-04-07 14:30:02 -04:00
|
|
|
public:
|
2024-02-01 21:29:00 -05:00
|
|
|
static constexpr bool is_stereo = false;
|
2024-02-12 14:33:55 -05:00
|
|
|
void set_scaled_volume_range(int16_t, double *, double) {}
|
|
|
|
static constexpr std::size_t size() { return 0; }
|
|
|
|
double total_scale(double *) const { return 0.0; }
|
2018-04-07 14:30:02 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename S, typename... R> class CompoundSourceHolder<S, R...> {
|
|
|
|
public:
|
|
|
|
CompoundSourceHolder(S &source, R &...next) : source_(source), next_source_(next...) {}
|
|
|
|
|
2024-02-08 15:21:47 -05:00
|
|
|
static constexpr bool is_stereo = S::is_stereo || CompoundSourceHolder<R...>::is_stereo;
|
|
|
|
|
2024-02-12 10:55:52 -05:00
|
|
|
template <Outputs::Speaker::Action action, bool output_stereo>
|
|
|
|
void apply_samples(std::size_t number_of_samples, typename ::Outputs::Speaker::SampleT<output_stereo>::type *target) {
|
2024-02-10 21:53:12 -05:00
|
|
|
|
|
|
|
// If this is the step at which a mono-to-stereo adaptation happens, apply it.
|
|
|
|
if constexpr (output_stereo && !S::is_stereo) {
|
|
|
|
// There'll be only one place in the chain that this conversion happens, but it'll
|
|
|
|
// happen there often. So avoid continuously reallocating.
|
|
|
|
if(conversion_source_.size() < number_of_samples) {
|
|
|
|
conversion_source_.resize(number_of_samples);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Populate the conversion buffer with this source and all below.
|
2024-02-12 10:55:52 -05:00
|
|
|
apply_samples<Action::Store, false>(number_of_samples, conversion_source_.data());
|
2024-02-10 21:53:12 -05:00
|
|
|
|
|
|
|
// Map up and return.
|
|
|
|
for(std::size_t c = 0; c < number_of_samples; c++) {
|
2024-02-12 10:55:52 -05:00
|
|
|
Outputs::Speaker::apply<action>(target[c].left, StereoSample(conversion_source_[c]));
|
2024-02-10 21:53:12 -05:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-02-12 14:33:55 -05:00
|
|
|
constexpr bool is_final_source = sizeof...(R) == 0;
|
|
|
|
|
|
|
|
// Get the rest of the output, if any.
|
|
|
|
if constexpr (!is_final_source) {
|
|
|
|
next_source_.template apply_samples<action, output_stereo>(number_of_samples, target);
|
|
|
|
}
|
2020-02-16 18:45:36 -05:00
|
|
|
|
2018-04-07 14:30:02 -04:00
|
|
|
if(source_.is_zero_level()) {
|
2020-02-16 18:45:36 -05:00
|
|
|
// This component is currently outputting silence; therefore don't add anything to the output
|
2024-02-12 14:33:55 -05:00
|
|
|
// audio. Zero fill only if this is the final source (as everything above will be additive).
|
|
|
|
if constexpr (is_final_source) {
|
|
|
|
Outputs::Speaker::fill<action>(target, target + number_of_samples, typename SampleT<output_stereo>::type());
|
|
|
|
}
|
2020-02-16 18:45:36 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-02-12 10:55:52 -05:00
|
|
|
// Add in this component's output.
|
2024-02-12 14:33:55 -05:00
|
|
|
source_.template apply_samples<is_final_source ? Action::Store : Action::Mix>(number_of_samples, target);
|
2018-04-07 14:30:02 -04:00
|
|
|
}
|
|
|
|
|
2020-05-09 17:57:21 -04:00
|
|
|
void set_scaled_volume_range(int16_t range, double *volumes, double scale) {
|
2024-02-01 21:29:00 -05:00
|
|
|
const auto scaled_range = volumes[0] / double(source_.average_output_peak()) * double(range) / scale;
|
2020-05-09 17:57:21 -04:00
|
|
|
source_.set_sample_volume_range(int16_t(scaled_range));
|
|
|
|
next_source_.set_scaled_volume_range(range, &volumes[1], scale);
|
2018-04-07 14:30:02 -04:00
|
|
|
}
|
|
|
|
|
2020-05-09 17:57:21 -04:00
|
|
|
static constexpr std::size_t size() {
|
|
|
|
return 1 + CompoundSourceHolder<R...>::size();
|
2018-04-07 14:30:02 -04:00
|
|
|
}
|
|
|
|
|
2020-05-09 17:57:21 -04:00
|
|
|
double total_scale(double *volumes) const {
|
2024-02-01 21:29:00 -05:00
|
|
|
return (volumes[0] / source_.average_output_peak()) + next_source_.total_scale(&volumes[1]);
|
2020-05-09 17:57:21 -04:00
|
|
|
}
|
|
|
|
|
2018-04-07 14:30:02 -04:00
|
|
|
private:
|
|
|
|
S &source_;
|
|
|
|
CompoundSourceHolder<R...> next_source_;
|
2024-02-10 21:53:12 -05:00
|
|
|
std::vector<MonoSample> conversion_source_;
|
2018-04-07 14:30:02 -04:00
|
|
|
};
|
|
|
|
|
2024-02-01 21:29:00 -05:00
|
|
|
public:
|
2024-02-10 21:53:12 -05:00
|
|
|
using Sample = typename SampleT<::Outputs::Speaker::is_stereo<T...>()>::type;
|
|
|
|
|
2024-02-09 14:34:59 -05:00
|
|
|
// To ensure at most one mono to stereo conversion, require appropriate source ordering.
|
|
|
|
static_assert(are_properly_ordered<T...>(), "Sources should be listed with all stereo sources before all mono sources");
|
|
|
|
|
2024-02-01 21:29:00 -05:00
|
|
|
CompoundSource(T &... sources) : source_holder_(sources...) {
|
|
|
|
// Default: give all sources equal volume.
|
|
|
|
const auto volume = 1.0 / double(source_holder_.size());
|
|
|
|
for(std::size_t c = 0; c < source_holder_.size(); ++c) {
|
|
|
|
volumes_.push_back(volume);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-12 10:55:52 -05:00
|
|
|
template <Outputs::Speaker::Action action>
|
|
|
|
void apply_samples(std::size_t number_of_samples, Sample *target) {
|
|
|
|
source_holder_.template apply_samples<action, ::Outputs::Speaker::is_stereo<T...>()>(number_of_samples, target);
|
2024-02-01 21:29:00 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Sets the total output volume of this CompoundSource.
|
|
|
|
*/
|
|
|
|
void set_sample_volume_range(int16_t range) {
|
|
|
|
volume_range_ = range;
|
|
|
|
push_volumes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Sets the relative volumes of the various sources underlying this
|
|
|
|
compound. The caller should ensure that the number of items supplied
|
|
|
|
matches the number of sources and that the values in it sum to 1.0.
|
|
|
|
*/
|
|
|
|
void set_relative_volumes(const std::vector<double> &volumes) {
|
|
|
|
assert(volumes.size() == source_holder_.size());
|
|
|
|
volumes_ = volumes;
|
|
|
|
push_volumes();
|
|
|
|
average_output_peak_ = 1.0 / source_holder_.total_scale(volumes_.data());
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
@returns the average output peak given the sources owned by this CompoundSource and the
|
|
|
|
current relative volumes.
|
|
|
|
*/
|
|
|
|
double average_output_peak() const {
|
|
|
|
return average_output_peak_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void push_volumes() {
|
|
|
|
const double scale = source_holder_.total_scale(volumes_.data());
|
|
|
|
source_holder_.set_scaled_volume_range(volume_range_, volumes_.data(), scale);
|
|
|
|
}
|
|
|
|
|
2018-04-07 14:30:02 -04:00
|
|
|
CompoundSourceHolder<T...> source_holder_;
|
2020-05-09 17:57:21 -04:00
|
|
|
std::vector<double> volumes_;
|
2018-04-07 14:30:02 -04:00
|
|
|
int16_t volume_range_ = 0;
|
2020-11-21 22:52:57 -05:00
|
|
|
std::atomic<double> average_output_peak_{1.0};
|
2017-12-19 21:08:10 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|