done bw_{buf,common,math,note_queue,rand,voice_alloc} + related changes

This commit is contained in:
Stefano D'Angelo 2023-08-11 12:49:05 +02:00
parent 41f56ec91c
commit 55a78a5ab4
11 changed files with 300 additions and 213 deletions

3
TODO
View File

@ -33,7 +33,7 @@ code:
* update state ctrl -> process ctrl?
* mem req -> return value of set sample rate?
* peak gain + Q ???
* bwpp -> bwxx
* merge c++ code into c headers
* sr_reduce reset_coeffs? update_coeffs? process_multi?
* fix definitions leading to X ** -> const X ** error https://isocpp.org/wiki/faq/const-correctness#constptrptr-conversion, fix fx_balance, fx_pan, synth_poly
* drywet -> dry_wet, ringmod -> ring_mod, etc?
@ -43,7 +43,6 @@ code:
* c++ get coeffs/state? or public? src nIn/OutSamples case (array vs single value), delay read/write, process1? process single?
* bw_buf invert src dest order?
* check unititialized warnings
* voice alloc mode -> voice alloc priority
* clearly specify that state is tied to a particular set of coeffs (1:N)
* modulation vs process (multi) no update (post 1.0.0)???
* check assumptions w.r.t. usage of math functions

View File

@ -149,12 +149,12 @@ static void note_off(void *BW_RESTRICT voice, float velocity) {
v->gate = 0;
}
static unsigned char get_note(void *BW_RESTRICT voice) {
static unsigned char get_note(const void *BW_RESTRICT voice) {
bw_example_synth_poly_voice *v = (bw_example_synth_poly_voice *)voice;
return v->note;
}
static char is_free(void *BW_RESTRICT voice) {
static char is_free(const void *BW_RESTRICT voice) {
bw_example_synth_poly_voice *v = (bw_example_synth_poly_voice *)voice;
bw_env_gen_phase phase = bw_env_gen_get_phase(&v->vca_env_gen_state);
return !v->gate && phase == bw_env_gen_phase_off;

View File

@ -21,7 +21,7 @@
#include "bw_example_synthpp_poly.h"
#include <bw_math.h>
#include <bwpp_buf.h>
#include <bw_buf.h>
#include <bw_voice_alloc.h>
void bw_example_synthpp_poly_init(bw_example_synthpp_poly *instance) {
@ -121,12 +121,12 @@ static void note_off(void *BW_RESTRICT voice, float velocity) {
v->gate = 0;
}
static unsigned char get_note(void *BW_RESTRICT voice) {
static unsigned char get_note(const void *BW_RESTRICT voice) {
bw_example_synthpp_poly_voice *v = (bw_example_synthpp_poly_voice *)voice;
return v->note;
}
static char is_free(void *BW_RESTRICT voice) {
static char is_free(const void *BW_RESTRICT voice) {
bw_example_synthpp_poly_voice *v = (bw_example_synthpp_poly_voice *)voice;
bw_env_gen_phase phase = v->instance->vcaEnvGen.getPhase(v->index);
return !v->gate && phase == bw_env_gen_phase_off;

View File

@ -23,7 +23,7 @@
#include "platform.h"
#include <bwpp_note_queue.h>
#include <bw_note_queue.h>
#include <bwpp_phase_gen.h>
#include <bwpp_osc_saw.h>
#include <bwpp_osc_pulse.h>

View File

@ -31,6 +31,8 @@
* <ul>
* <li>Now using <code>size_t</code> instead of
* <code>BW_SIZE_T</code>.</li>
* <li>Moved C++ code to C header.</li>
* <li>Removed usage of reserved identifiers.</li>
* </ul>
* </li>
* <li>Version <strong>0.6.0</strong>:
@ -61,8 +63,8 @@
* }}}
*/
#ifndef _BW_BUF_H
#define _BW_BUF_H
#ifndef BW_BUF_H
#define BW_BUF_H
#include <bw_common.h>
@ -160,11 +162,85 @@ static inline void bw_buf_mul_multi(float **dest, const float **src1, const floa
* `n_channels` buffers `dest`.
* }}} */
#ifdef __cplusplus
}
#include <array>
namespace Brickworks {
/*! api_cpp {{{
* ##### Brickworks::bufFill
* ```>>> */
template<size_t N_CHANNELS>
void bufFill(
std::array<float *, N_CHANNELS> dest,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufNeg
* ```>>> */
template<size_t N_CHANNELS>
void bufNeg(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufAdd
* ```>>> */
template<size_t N_CHANNELS>
void bufAdd(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufScale
* ```>>> */
template<size_t N_CHANNELS>
void bufScale(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufMix
* ```>>> */
template<size_t N_CHANNELS>
void bufMix(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufMul
* ```>>> */
template<size_t N_CHANNELS>
void bufMul(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples);
/*! <<<```
* }}} */
}
#endif
/*** Implementation ***/
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
#ifdef __cplusplus
extern "C" {
#endif
static inline void bw_buf_fill(float *dest, float k, size_t n_elems) {
BW_ASSERT(!(dest == NULL && n_elems != 0));
BW_ASSERT(!bw_is_nan(k));
@ -272,6 +348,62 @@ static inline void bw_buf_mul_multi(float **dest, const float **src1, const floa
}
#ifdef __cplusplus
}
namespace Brickworks {
template<size_t N_CHANNELS>
inline void bufFill(
std::array<float *, N_CHANNELS> dest,
float k,
int nSamples) {
bw_buf_fill_multi(dest.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufNeg(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
int nSamples) {
bw_buf_neg_multi(dest.data(), src.data(), N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufAdd(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples) {
bw_buf_add_multi(dest.data(), src.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufScale(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples) {
bw_buf_scale_multi(dest.data(), src.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufMix(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples) {
bw_buf_mix_multi(dest.data(), src1.data(), src2.data(), N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufMul(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples) {
bw_buf_mul_multi(dest.data(), src1.data(), src2.data(), N_CHANNELS, nSamples);
}
}
#endif

View File

@ -38,7 +38,8 @@
* <li>Introduced <code>BW_NO_DEBUG</code> to replace
* <code>NDEBUG</code> and prevented useless inclusion of
* <code>assert.h</code>.</li>
* <li>Avoided using reserved identifiers.</li>
* <li>Removed usage of reserved identifiers.</li>
* <li>Added <code>extern "C"</code> to functions.</li>
* </ul>
* </li>
* <li>Version <strong>0.6.0</strong>:
@ -194,6 +195,11 @@
# define BW_ASSERT_DEEP(ignore) ((void)0)
# endif
#endif
#ifdef __cplusplus
extern "C" {
#endif
/*! ...
*
* #### bw_is_inf()
@ -244,11 +250,19 @@ static inline uint32_t bw_hash_sdbm(const char *string);
* Returns the sdbm hash of the given `string`.
* }}} */
#ifdef __cplusplus
}
#endif
/*** Implementation ***/
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
#ifdef __cplusplus
extern "C" {
#endif
static inline char bw_is_inf(float x) {
union { uint32_t u; float f; } v;
v.f = x;
@ -295,4 +309,8 @@ static inline uint32_t bw_hash_sdbm(const char *string) {
return hash;
}
#ifdef __cplusplus
}
#endif
#endif

View File

@ -62,6 +62,7 @@
* <code>bw_acoshf()</code>.</li>
* <li>Removed usage of reserved identifiers and designated
* initializers.</li>
* <li>Added <code>extern "C"</code> to functions.</li>
* <li>Improved documentation w.r.t. validity of input values and
* approximation errors.</li>
* </ul>
@ -110,12 +111,12 @@
#ifndef BW_MATH_H
#define BW_MATH_H
#include <bw_common.h>
#ifdef __cplusplus
extern "C" {
#endif
#include <bw_common.h>
/*! api {{{
* #### bw_signfilli32()
* ```>>> */
@ -482,11 +483,19 @@ static inline float bw_acoshf(float x);
* Absolute error < 0.004 or relative error < 0.8%, whatever is worse.
* }}} */
#ifdef __cplusplus
}
#endif
/*** Implementation ***/
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
#ifdef __cplusplus
extern "C" {
#endif
// I hope the target architecture and compiler will use conditional ops here
static inline int32_t bw_signfilli32(int32_t x) {

View File

@ -20,7 +20,7 @@
/*!
* module_type {{{ utility }}}
* version {{{ 0.6.0 }}}
* version {{{ 1.0.0 }}}
* requires {{{ bw_common }}}
* description {{{
* Simple data structure that helps keeping track of note on/off events and
@ -30,6 +30,19 @@
* }}}
* changelog {{{
* <ul>
* <li>Version <strong>1.0.0</strong>:
* <ul>
* <li>Removed redundant <code>status</code> member from
* <code>bw_note_queue_event</code>.</li>
* <li>Possibly improved memeory layout of
* <code>bw_note_queue</code>.</li>
* <li>Moved C++ code to C header.</li>
* <li>Removed usage of reserved identifiers and designated
* initializers.</li>
* <li>Clarified ambiguity in the documentation of
* <code>bw_note_queue_status</code>.</li>
* </ul>
* </li>
* <li>Version <strong>0.6.0</strong>:
* <ul>
* <li>Added debugging code.</li>
@ -45,38 +58,37 @@
* }}}
*/
#ifndef _BW_NOTE_QUEUE_H
#define _BW_NOTE_QUEUE_H
#ifndef BW_NOTE_QUEUE_H
#define BW_NOTE_QUEUE_H
#include <bw_common.h>
#ifdef __cplusplus
extern "C" {
#endif
#include <bw_common.h>
/*! api {{{
* #### bw_note_queue_status
* ```>>> */
typedef struct {
char pressed;
float velocity; // negative = unknown / not available
float velocity;
} bw_note_queue_status;
/*! <<<```
* Note status:
* * `pressed`: whether the note is pressed (non-`0`) or not (`0`);
* * `velocity`: velocity in [`0.f`, `1.f`].
* * `velocity`: velocity in [`0.f`, `1.f`], otherwise negative to indicate
* unknown / not available.
*
* #### bw_note_queue_event
* ```>>> */
typedef struct {
unsigned char note;
bw_note_queue_status status;
char went_off;
} bw_note_queue_event;
/*! <<<```
* Note on/off event:
* * `note`: note number in [`0`, `127`];
* * `status`: note status;
* * `went_off`: whether a note off event fired on the same note (non-`0`)
* or not (`0`) -- see `bw_note_queue`.
*
@ -84,8 +96,8 @@ typedef struct {
* ```>>> */
typedef struct {
bw_note_queue_event events[128];
unsigned char n_events;
bw_note_queue_status status[128];
unsigned char n_events;
unsigned char n_pressed;
} bw_note_queue;
/*! <<<```
@ -95,8 +107,8 @@ typedef struct {
* event added for a given note overwrites the previous if it exists;
* `went_off` is set to non-`0` in case of a note off event or when
* overwriting an event whose `went_off` was already non-`0`;
* * `n_events`: number of elements in `events`;
* * `status`: current status of all notes;
* * `n_events`: number of elements in `events`;
* * `n_pressed`: number of currently pressed keys.
*
* #### bw_note_queue_reset()
@ -135,17 +147,46 @@ static inline char bw_note_queue_is_valid(const bw_note_queue *BW_RESTRICT queue
* than or equal to that of `bw_note_queue`.
* }}} */
#ifdef __cplusplus
}
namespace Brickworks {
/*! api_cpp {{{
* ##### Brickworks::NoteQueue
* ```>>> */
class NoteQueue {
public:
NoteQueue();
void clear();
void add(unsigned char note, bool pressed, float velocity, bool force_went_off);
bw_note_queue queue;
};
/*! <<<```
* }}} */
}
#endif
/*** Implementation ***/
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
#ifdef __cplusplus
extern "C" {
#endif
static inline void bw_note_queue_reset(bw_note_queue *BW_RESTRICT queue) {
BW_ASSERT(queue != NULL);
for (int i = 0; i < 128; i++)
queue->status[i] = (bw_note_queue_status){ .pressed = 0, .velocity = 0.f };
queue->n_pressed = 0;
for (int i = 0; i < 128; i++) {
queue->status[i].pressed = 0;
queue->status[i].velocity = 0.f;
}
queue->n_events = 0;
queue->n_pressed = 0;
BW_ASSERT_DEEP(bw_note_queue_is_valid(queue));
}
@ -172,14 +213,16 @@ static inline void bw_note_queue_add(bw_note_queue *BW_RESTRICT queue, unsigned
if (i == queue->n_events)
queue->n_events++;
else
went_off = queue->events[i].went_off || !queue->events[i].status.pressed;
went_off = queue->events[i].went_off || !queue->status[note].pressed;
queue->events[i] = (bw_note_queue_event){ .note = note, .status = { pressed, velocity }, .went_off = went_off || force_went_off };
queue->events[i].note = note;
queue->events[i].went_off = went_off || force_went_off;
if (pressed && !queue->status[note].pressed)
queue->n_pressed++;
else if (!pressed && queue->status[note].pressed)
queue->n_pressed--;
queue->status[note] = (bw_note_queue_status){ .pressed = pressed, .velocity = velocity };
queue->status[note].pressed = pressed;
queue->status[note].velocity = velocity;
BW_ASSERT_DEEP(bw_note_queue_is_valid(queue));
}
@ -190,19 +233,19 @@ static inline char bw_note_queue_is_valid(const bw_note_queue *BW_RESTRICT queue
if (queue->n_events >= 128 || queue->n_pressed >= 128)
return 0;
for (int i = 0; i < (int)queue->n_events; i++) {
for (unsigned char i = 0; i < queue->n_events; i++) {
const bw_note_queue_event *ev = queue->events + i;
if (ev->note >= 128 || !bw_is_finite(ev->status.velocity) || ev->status.velocity > 1.f)
if (ev->note >= 128)
return 0;
for (int j = 0; j < i; j++) {
for (unsigned char j = 0; j < i; j++) {
const bw_note_queue_event *ev2 = queue->events + j;
if (ev2->note == ev->note)
return 0;
}
}
int cnt = 0;
for (int i = 0; i < 128; i++) {
unsigned char cnt = 0;
for (unsigned char i = 0; i < 128; i++) {
const bw_note_queue_status *st = queue->status + i;
if (st->pressed)
cnt++;
@ -214,6 +257,22 @@ static inline char bw_note_queue_is_valid(const bw_note_queue *BW_RESTRICT queue
}
#ifdef __cplusplus
}
namespace Brickworks {
inline NoteQueue::NoteQueue() {
bw_note_queue_reset(&queue);
}
inline void NoteQueue::clear() {
bw_note_queue_clear(&queue);
}
inline void NoteQueue::add(unsigned char note, bool pressed, float velocity, bool force_went_off) {
bw_note_queue_add(&queue, note, pressed, velocity, force_went_off);
}
}
#endif

View File

@ -20,7 +20,7 @@
/*!
* module_type {{{ utility }}}
* version {{{ 0.6.0 }}}
* version {{{ 1.0.0 }}}
* requires {{{ bw_common }}}
* description {{{
* Pseudo-random number generators.
@ -34,6 +34,12 @@
* }}}
* changelog {{{
* <ul>
* <li>Version <strong>1.0.0</strong>:
* <ul>
* <li>Added one more assertion in <code>bw_randf()</code>.</li>
* <li>Removed usage of reserved identifiers.</li>
* </ul>
* </li>
* <li>Version <strong>0.6.0</strong>:
* <ul>
* <li>Added debugging code.</li>
@ -55,8 +61,8 @@
* }}}
*/
#ifndef _BW_RAND_H
#define _BW_RAND_H
#ifndef BW_RAND_H
#define BW_RAND_H
#include <bw_common.h>
@ -86,8 +92,16 @@ static inline float bw_randf(uint64_t *BW_RESTRICT state);
* between calls and which gets updated by this function.
* }}} */
#ifdef __cplusplus
}
#endif
/*** Implementation ***/
#ifdef __cplusplus
extern "C" {
#endif
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
@ -101,7 +115,9 @@ static inline uint32_t bw_randu32(uint64_t *BW_RESTRICT state) {
static inline float bw_randf(uint64_t *BW_RESTRICT state) {
BW_ASSERT(state != NULL);
return (2.f / (float)UINT32_MAX) * (float)bw_randu32(state) - 1.f;
const float y = (2.f / (float)UINT32_MAX) * (float)bw_randu32(state) - 1.f;
BW_ASSERT(bw_is_finite(y));
return y;
}
#ifdef __cplusplus

View File

@ -31,6 +31,10 @@
* <ul>
* <li>Now using <code>size_t</code> instead of
* <code>BW_SIZE_T</code>.</li>
* <li>Added <code>const</code> where needed to callback types in
* <code>bw_voice_alloc_opts</code> and to
* <code>bw_voice_alloc()</code>.</li>
* <li>Removed usage of reserved identifiers.</li>
* </ul>
* </li>
* <li>Version <strong>0.6.0</strong>:
@ -50,8 +54,8 @@
* }}}
*/
#ifndef _BW_VOICE_ALLOC_H
#define _BW_VOICE_ALLOC_H
#ifndef BW_VOICE_ALLOC_H
#define BW_VOICE_ALLOC_H
#ifdef __cplusplus
extern "C" {
@ -79,8 +83,8 @@ typedef struct {
void (*note_on)(void *BW_RESTRICT voice, unsigned char note, float velocity);
void (*note_off)(void *BW_RESTRICT voice, float velocity);
unsigned char (*get_note)(void *BW_RESTRICT voice);
char (*is_free)(void *BW_RESTRICT voice);
unsigned char (*get_note)(const void *BW_RESTRICT voice);
char (*is_free)(const void *BW_RESTRICT voice);
} bw_voice_alloc_opts;
/*! <<<```
* Voice allocation options:
@ -97,7 +101,7 @@ typedef struct {
*
* #### bw_voice_alloc()
* ```>>> */
void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *BW_RESTRICT queue, void **BW_RESTRICT voices, size_t n_voices);
void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *BW_RESTRICT queue, void * const *BW_RESTRICT voices, size_t n_voices);
/*! <<<```
* It performs voice allocation according to `opts` and using the events in
* `queue`.
@ -111,7 +115,7 @@ void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *BW_RESTRICT queue, void **BW_RESTRICT voices, size_t n_voices) {
void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *BW_RESTRICT queue, void * const *BW_RESTRICT voices, size_t n_voices) {
BW_ASSERT(opts != NULL);
BW_ASSERT(opts->priority == bw_voice_alloc_priority_low || opts->priority == bw_voice_alloc_priority_high);
BW_ASSERT(n_voices == 0 || opts->note_on != NULL);
@ -124,45 +128,43 @@ void bw_voice_alloc(const bw_voice_alloc_opts *BW_RESTRICT opts, bw_note_queue *
for (unsigned char i = 0; i < queue->n_events; i++) {
bw_note_queue_event *ev = queue->events + i;
bw_note_queue_status *st = queue->status + ev->note;
for (size_t j = 0; j < n_voices; j++)
if (!opts->is_free(voices[j]) && opts->get_note(voices[j]) == ev->note) {
if (!ev->status.pressed || ev->went_off)
opts->note_off(voices[j], ev->status.velocity);
if (ev->status.pressed)
opts->note_on(voices[j], ev->note, ev->status.velocity);
if (!st->pressed || ev->went_off)
opts->note_off(voices[j], st->velocity);
if (st->pressed)
opts->note_on(voices[j], ev->note, st->velocity);
goto next_event;
}
if (ev->status.pressed) {
if (st->pressed) {
for (size_t j = 0; j < n_voices; j++)
if (opts->is_free(voices[j])) {
opts->note_on(voices[j], ev->note, ev->status.velocity);
opts->note_on(voices[j], ev->note, st->velocity);
goto next_event;
}
size_t k = n_voices;
int v = ev->note;
unsigned char v = ev->note;
for (size_t j = 0; j < n_voices; j++) {
int n = opts->get_note(voices[j]);
unsigned char n = opts->get_note(voices[j]);
if (!queue->status[n].pressed && (k == n_voices || (opts->priority == bw_voice_alloc_priority_low ? n > v : n < v))) {
v = n;
k = j;
}
}
if (k != n_voices) {
opts->note_on(voices[k], ev->note, ev->status.velocity);
continue;
}
for (size_t j = 0; j < n_voices; j++) {
int n = opts->get_note(voices[j]);
if (opts->priority == bw_voice_alloc_priority_low ? n > v : n < v) {
v = n;
k = j;
if (k == n_voices)
for (size_t j = 0; j < n_voices; j++) {
unsigned char n = opts->get_note(voices[j]);
if (opts->priority == bw_voice_alloc_priority_low ? n > v : n < v) {
v = n;
k = j;
}
}
}
if (k != n_voices)
opts->note_on(voices[k], ev->note, ev->status.velocity);
opts->note_on(voices[k], ev->note, st->velocity);
}
next_event:;

View File

@ -1,148 +0,0 @@
/*
* Brickworks
*
* Copyright (C) 2023 Orastron Srl unipersonale
*
* Brickworks is free software: you can reosc_sinribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3 of the License.
*
* Brickworks is osc_sinributed 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 Brickworks. If not, see <http://www.gnu.org/licenses/>.
*
* File author: Stefano D'Angelo
*/
#ifndef BWPP_BUF_H
#define BWPP_BUF_H
#include <bw_buf.h>
#include <array>
namespace Brickworks {
/*! api {{{
* ##### Brickworks::bufFill
* ```>>> */
template<size_t N_CHANNELS>
void bufFill(
std::array<float *, N_CHANNELS> dest,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufNeg
* ```>>> */
template<size_t N_CHANNELS>
void bufNeg(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufAdd
* ```>>> */
template<size_t N_CHANNELS>
void bufAdd(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufScale
* ```>>> */
template<size_t N_CHANNELS>
void bufScale(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufMix
* ```>>> */
template<size_t N_CHANNELS>
void bufMix(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples);
/*! <<<```
*
* ##### Brickworks::bufMul
* ```>>> */
template<size_t N_CHANNELS>
void bufMul(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples);
/*! <<<```
* }}} */
/*** Implementation ***/
/* WARNING: This part of the file is not part of the public API. Its content may
* change at any time in future versions. Please, do not use it directly. */
template<size_t N_CHANNELS>
inline void bufFill(
std::array<float *, N_CHANNELS> dest,
float k,
int nSamples) {
bw_buf_fill_multi(dest.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufNeg(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
int nSamples) {
bw_buf_neg_multi(dest.data(), src.data(), N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufAdd(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples) {
bw_buf_add_multi(dest.data(), src.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufScale(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src,
float k,
int nSamples) {
bw_buf_scale_multi(dest.data(), src.data(), k, N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufMix(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples) {
bw_buf_mix_multi(dest.data(), src1.data(), src2.data(), N_CHANNELS, nSamples);
}
template<size_t N_CHANNELS>
inline void bufMul(
std::array<float *, N_CHANNELS> dest,
std::array<const float *, N_CHANNELS> src1,
std::array<const float *, N_CHANNELS> src2,
int nSamples) {
bw_buf_mul_multi(dest.data(), src1.data(), src2.data(), N_CHANNELS, nSamples);
}
}
#endif