cwWaveTableBank.h/cpp: Implemented gen_notes() and integrated test() with cwTest.

This commit is contained in:
kevin 2024-07-13 15:08:16 -04:00
parent bf9239058d
commit 15487e3eb4
2 changed files with 522 additions and 42 deletions

View File

@ -8,10 +8,13 @@
#include "cwObject.h" #include "cwObject.h"
#include "cwFileSys.h" #include "cwFileSys.h"
#include "cwAudioFile.h" #include "cwAudioFile.h"
#include "cwDspTypes.h" #include "cwMath.h"
#include "cwWaveTableBank.h"
#include "cwVectOps.h" #include "cwVectOps.h"
#include "cwDspTypes.h"
#include "cwDsp.h"
#include "cwWaveTableBank.h"
#include "cwMidi.h" #include "cwMidi.h"
#include "cwTest.h"
namespace cw namespace cw
{ {
@ -122,7 +125,8 @@ namespace cw
af_t* af0 = af->link; af_t* af0 = af->link;
for(unsigned i=0; i<af->segN; ++i) for(unsigned i=0; i<af->segN; ++i)
mem::release(af->segA[i].aV); if(af->segA != nullptr )
mem::release(af->segA[i].aV);
af->cfg->free(); af->cfg->free();
mem::release(af->wtA); mem::release(af->wtA);
@ -210,7 +214,8 @@ namespace cw
const object_t* wtL, const object_t* wtL,
const sample_t* const* audio_ch_buf = nullptr, const sample_t* const* audio_ch_buf = nullptr,
unsigned audio_ch_bufN = 0, unsigned audio_ch_bufN = 0,
unsigned padSmpN = 0 ) unsigned padSmpN = 0,
srate_t srate = 0)
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
unsigned wtN = wtL->child_count(); unsigned wtN = wtL->child_count();
@ -261,10 +266,11 @@ namespace cw
} }
// count or store the wt recd // count or store the wt recd
wr.instr_id = count_fl ? kInvalidId : _find_or_add_instr_id(p,instr_label); wr.instr_id = count_fl ? kInvalidId : _find_or_add_instr_id(p,instr_label);
wr.chN = chL->child_count(); wr.chN = chL->child_count();
wr.chA = count_fl ? nullptr : af->chA + ch_idx; wr.chA = count_fl ? nullptr : af->chA + ch_idx;
ch_idx += wr.chN; wr.srate = srate;
ch_idx += wr.chN;
// for each channel cfg on this wt cfg // for each channel cfg on this wt cfg
for(unsigned j=0; j<wr.chN; ++j) for(unsigned j=0; j<wr.chN; ++j)
@ -316,6 +322,7 @@ namespace cw
// parse the kth seg cfg // parse the kth seg cfg
if((rc = seg->getv("cost",sr.cost, if((rc = seg->getv("cost",sr.cost,
"cyc_per_loop",sr.cyc_per_loop,
"bsi",bsi, "bsi",bsi,
"esi",esi)) != kOkRC ) "esi",esi)) != kOkRC )
{ {
@ -381,7 +388,7 @@ namespace cw
} }
// Parse the cfg and fill the associated af. // Parse the cfg and fill the associated af.
if((rc = _parse_cfg( p, af,wtL, buf, af_info.chCnt, padN)) != kOkRC ) if((rc = _parse_cfg( p, af,wtL, buf, af_info.chCnt, padN, af_info.srate)) != kOkRC )
goto errLabel; goto errLabel;
} }
@ -398,7 +405,6 @@ namespace cw
rc_t _load_cfg_file( wt_bank_t* p, const char* cfg_fname, unsigned padN ) rc_t _load_cfg_file( wt_bank_t* p, const char* cfg_fname, unsigned padN )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
srate_t srate = 0;
const object_t* wtL = nullptr; const object_t* wtL = nullptr;
af_t* af = mem::allocZ<af_t>(); af_t* af = mem::allocZ<af_t>();
@ -414,7 +420,6 @@ namespace cw
} }
if((rc = af->cfg->getv("audio_fname",af->audio_fname, if((rc = af->cfg->getv("audio_fname",af->audio_fname,
"srate",srate,
"wt",wtL)) != kOkRC ) "wt",wtL)) != kOkRC )
{ {
rc = cwLogError(kSyntaxErrorRC,"The wave table header parse failed."); rc = cwLogError(kSyntaxErrorRC,"The wave table header parse failed.");
@ -437,6 +442,9 @@ namespace cw
goto errLabel; goto errLabel;
errLabel: errLabel:
if( rc != kOkRC )
rc = cwLogError(rc,"Wave table load failed on '%s'.",cwStringNullGuard(cfg_fname));
return rc; return rc;
} }
@ -474,6 +482,369 @@ namespace cw
return rc; return rc;
} }
const wt_t* _get_wave_table( wt_bank_t* p, unsigned instr_idx, unsigned pitch, unsigned vel )
{
rc_t rc = kOkRC;
if( instr_idx >= p->wtMapN )
{
rc = cwLogError(kInvalidArgRC,"Invalid instr_idx %i",instr_idx);
goto errLabel;
}
if( pitch >= midi::kMidiNoteCnt )
{
rc = cwLogError(kInvalidArgRC,"Invalid MIDI pitch %i",pitch);
goto errLabel;
}
if( vel >= midi::kMidiVelCnt )
{
rc = cwLogError(kInvalidArgRC,"Invalid MIDI pitch %i",pitch);
goto errLabel;
}
return p->wtMapA[ instr_idx ].map[ vel * midi::kMidiNoteCnt + pitch ];
errLabel:
cwLogError(rc,"Wave table lookup failed.");
return nullptr;
}
typedef struct seg_osc_str
{
seg_tid_t tid;
const seg_t* seg;
unsigned id;
double xphase;
double loop_frac_smpN; // length of the loop including fractional part
unsigned cur_loopN; // count of loops
sample_t* envV;
unsigned envN;
unsigned delay_smp_idx;
unsigned cur_smp_idx;
unsigned ch_idx;
} seg_osc_t;
void _seg_osc_setup( seg_osc_t* osc,
unsigned id,
unsigned ch_idx,
const seg_t* seg,
double smp_per_cyc,
sample_t* envV,
unsigned envN,
unsigned delay_smp_idx,
seg_tid_t tid = kInvalidTId )
{
osc->seg = seg;
osc->id = id;
osc->tid = tid==kInvalidTId ? seg->tid : tid;
osc->loop_frac_smpN = osc->tid==kAttackTId ? seg->aN : smp_per_cyc * seg->cyc_per_loop;
osc->xphase = 0;
osc->envV = envV;
osc->envN = envN;
osc->delay_smp_idx = delay_smp_idx;
osc->cur_smp_idx = 0;
osc->ch_idx = ch_idx;
}
void _seg_osc_update( seg_osc_t* osc, sample_t* yV, unsigned yN, unsigned& actual_yN_ref )
{
unsigned yi = 0;
const float* xV = osc->seg->aV + osc->seg->padN;
actual_yN_ref = 0;
for(yi=0; yi<yN; ++yi,++osc->cur_smp_idx)
if( osc->cur_smp_idx >= osc->delay_smp_idx )
{
double xfi = std::floor(osc->xphase);
double frac = osc->xphase - xfi;
int xi = (int)xfi;
yV[yi] += xV[xi] + (xV[xi+1] - xV[xi]) * frac * osc->envV[xi];
/*
float f = frac;
float f_1 = f - 1.0;
float f_2 = f - 2.0;
float f1 = f + 1.0;
yV[yi] += osc->envV[xi] * ((-f)*f_1*f_2*xV[xi-1]/6.0f + f1*f_1*f_2*xV[xi]/2.0f - f1*f*f_2*xV[xi+1]/2.0f + f1*f*f_1*xV[xi+2]/6.0f);
*/
//if( osc->seg->tid==kLoopTId && osc->ch_idx==0 && osc->cur_loopN < 4 )
// printf("%i,%i,%i,%i,%f,%f,%f,%f\n",osc->id,osc->cur_smp_idx,yi,xi,frac,xV[xi],osc->envV[xi],yV[yi]);
osc->xphase += 1.0f; // osc->seg->tid == kLoopTId ? 0.5f : 1.0f;
// if the end of the wave table is encountered
if( osc->xphase >= osc->loop_frac_smpN )
{
if( osc->tid != kLoopTId)
goto errLabel;
osc->xphase = osc->xphase - osc->loop_frac_smpN;
osc->cur_loopN += 1;
}
}
errLabel:
actual_yN_ref = yi;
}
void _seg_osc_update_0( seg_osc_t* osc, sample_t* yV, unsigned yN, unsigned& actual_yN_ref )
{
unsigned yi = 0;
const float* xV = osc->seg->aV + osc->seg->padN;
double xphs = osc->xphase;
double xfi = std::floor(osc->xphase);
double frac = xphs - xfi;
int xi = (int)xfi;
actual_yN_ref = 0;
for(yi=0; yi<yN; ++yi,++osc->cur_smp_idx)
if( osc->cur_smp_idx >= osc->delay_smp_idx )
{
//yV[yi] = xV[xi] + (xV[xi+1] - xV[xi]) * frac;
float f = frac;
float f_1 = f - 1.0;
float f_2 = f - 2.0;
float f1 = f + 1.0;
yV[yi] += osc->envV[xi] * ((-f)*f_1*f_2*xV[xi-1]/6.0f + f1*f_1*f_2*xV[xi]/2.0f - f1*f*f_2*xV[xi+1]/2.0f + f1*f*f_1*xV[xi+2]/6.0f);
//if( loop_fl && ch_idx==0 && seg_loopN_ref < 4 )
// printf("%i,%i,%f,%f,%f\n",yi,xi,frac,xV[xi],yV[yi]);
xi += 1;
// if the end of the wave table is encountered
if( frac+xi >= osc->loop_frac_smpN )
{
if( osc->tid != kLoopTId)
goto errLabel;
xphs = (frac+xi) - osc->loop_frac_smpN;
xi = (unsigned)std::floor(xphs);
frac = xphs - xi;
osc->cur_loopN += 1;
}
}
errLabel:
actual_yN_ref = yi;
osc->xphase = frac + xi;
}
rc_t _gen_note( const wt_t* wt, srate_t srate, sample_t** outChV, unsigned y_frm_cnt )
{
rc_t rc = kOkRC;
const unsigned frmPerUpdate = 64;
double hz = midi_to_hz( wt->pitch );
double smp_per_cyc = srate / hz;
// for each audio output channel
for(unsigned ch_idx=0; ch_idx<wt->chN; ++ch_idx)
{
seg_osc_t a_osc, b_osc, l0_osc, l1_osc;
seg_osc_t* cur_osc0 = &a_osc;
seg_osc_t* cur_osc1 = nullptr;
seg_osc_t* cur_oscb = nullptr;
unsigned a_envN = wt->chA[ch_idx].segA[0].aN;
sample_t a_envV[ a_envN ];
vop::ones(a_envV,a_envN);
unsigned l_envN = wt->chA[ch_idx].segA[1].aN;
sample_t l_envV[ l_envN ];
dsp::hann(l_envV,l_envN);
unsigned b_envN = wt->chA[ch_idx].segA[1].aN;
sample_t b_envV[ b_envN ];
vop::zero(b_envV,b_envN);
vop::copy(b_envV, l_envV+l_envN/2, l_envN/2);
_seg_osc_setup( &a_osc, 0, ch_idx, wt->chA[ch_idx].segA, smp_per_cyc, a_envV, a_envN, 0 );
_seg_osc_setup( &b_osc, 1, ch_idx, wt->chA[ch_idx].segA + 1, smp_per_cyc, b_envV, b_envN, 0, kAttackTId );
_seg_osc_setup( &l0_osc, 2, ch_idx, wt->chA[ch_idx].segA + 1, smp_per_cyc, l_envV, l_envN, 0 );
_seg_osc_setup( &l1_osc, 3, ch_idx, wt->chA[ch_idx].segA + 1, smp_per_cyc, l_envV, l_envN, l_envN/2 );
sample_t* yV = outChV[ch_idx];
unsigned y_frm_idx = 0;
// while the output channel is not full
while( y_frm_idx < y_frm_cnt )
{
unsigned y_upd_cnt = 0;
// while the current frame update has not generated frmPerUpdate samples
while( y_upd_cnt < frmPerUpdate && y_frm_idx < y_frm_cnt )
{
unsigned y_actual_upd_cnt = 0;
unsigned n = std::min(frmPerUpdate, std::min(frmPerUpdate-y_upd_cnt, y_frm_cnt-y_frm_idx));
if( cur_oscb != nullptr )
{
_seg_osc_update( cur_oscb, yV + y_frm_idx, n, y_actual_upd_cnt);
if( y_actual_upd_cnt != n )
cur_oscb = nullptr;
}
_seg_osc_update( cur_osc0, yV + y_frm_idx, n, y_actual_upd_cnt);
if( cur_osc1 != nullptr )
_seg_osc_update( cur_osc1, yV + y_frm_idx, n, y_actual_upd_cnt);
y_upd_cnt += y_actual_upd_cnt;
y_frm_idx += y_actual_upd_cnt;
// if the segment ran out of samples ...
if( y_actual_upd_cnt < n )
{
// (only attack segments run out of samples - because they do not loop)
assert( cur_osc0->tid == kAttackTId );
cur_osc0 = &l0_osc;
//cur_osc1 = &l1_osc;
//cur_oscb = &b_osc;
}
}
}
}
return rc;
}
/*
void _gen_osc_update(const sample_t* xV, unsigned xN, double loop_frac_smpN, double& xPhs_ref, sample_t* yV, unsigned yN, unsigned& actual_yN_ref, bool loop_fl, unsigned& seg_loopN_ref, unsigned ch_idx )
{
unsigned yi = 0;
double xphs = xPhs_ref;
double xfi = std::floor(xphs);
double frac = xphs - xfi;
int xi = (int)xfi;
actual_yN_ref = 0;
for(yi=0; yi<yN; ++yi)
{
//yV[yi] = xV[xi] + (xV[xi+1] - xV[xi]) * frac;
float f = frac;
float f_1 = f - 1.0;
float f_2 = f - 2.0;
float f1 = f + 1.0;
yV[yi] = (-f)*f_1*f_2*xV[xi-1]/6.0f + f1*f_1*f_2*xV[xi]/2.0f - f1*f*f_2*xV[xi+1]/2.0f + f1*f*f_1*xV[xi+2]/6.0f;
//if( loop_fl && ch_idx==0 && seg_loopN_ref < 4 )
// printf("%i,%i,%f,%f,%f\n",yi,xi,frac,xV[xi],yV[yi]);
xi += 1;
// if the end of the wave table is encountered
if( frac+xi >= loop_frac_smpN )
{
if( !loop_fl)
goto errLabel;
xphs = (frac+xi) - loop_frac_smpN;
xi = (unsigned)std::floor(xphs);
frac = xphs - xi;
seg_loopN_ref += 1;
}
}
errLabel:
actual_yN_ref = yi;
xPhs_ref = frac + xi;
}
rc_t _gen_note_0( const wt_t* wt, srate_t srate, sample_t** outChV, unsigned y_frm_cnt )
{
rc_t rc = kOkRC;
const unsigned frmPerUpdate = 64;
double hz = midi_to_hz( wt->pitch );
double smp_per_cyc = srate / hz;
// for each audio output channel
for(unsigned ch_idx=0; ch_idx<wt->chN; ++ch_idx)
{
sample_t* yV = outChV[ch_idx];
unsigned y_frm_idx = 0;
const seg_t* seg = wt->chA[ch_idx].segA;
unsigned seg_idx = 0;
unsigned seg_smp_cnt = seg->aN;
double seg_phase = 0;
double seg_frac_smpN = seg_smp_cnt;
unsigned seg_loop_cnt = 0;
// while the output channel is not full
while( y_frm_idx < y_frm_cnt )
{
unsigned y_upd_cnt = 0;
// while the current frame update has not generated frmPerUpdate samples
while( y_upd_cnt < frmPerUpdate && y_frm_idx < y_frm_cnt )
{
unsigned y_actual_upd_cnt = 0;
// TODO: handle case where y_frm_cnt is not an even multiple of frmPerUpdate
// attempt to generate frmPerUpdate samples into yV[ y_frm_idx:y_frm_idx + frmPerUpdate ]
_gen_osc_update(seg->aV + seg->padN, seg_smp_cnt, seg_frac_smpN, seg_phase, yV + y_frm_idx, frmPerUpdate, y_actual_upd_cnt, seg->tid==kLoopTId, seg_loop_cnt, ch_idx );
y_upd_cnt += y_actual_upd_cnt;
y_frm_idx += y_actual_upd_cnt;
// if the segment ran out of samples ...
if( y_actual_upd_cnt < frmPerUpdate )
{
// (only attack segments run out of samples - because they do not loop)
assert( seg->tid == kAttackTId );
// ...then advance to the next segment
seg_idx += 1;
if( seg_idx >= wt->chA[ch_idx].segN )
{
// done
goto errLabel;
}
seg = wt->chA[ch_idx].segA + seg_idx;
seg_phase = 0;
seg_smp_cnt = seg->aN;
seg_frac_smpN = smp_per_cyc * seg->cyc_per_loop;
seg_loop_cnt = 0;
}
}
}
}
errLabel:
return rc;
}
*/
} }
} }
@ -559,47 +930,152 @@ unsigned cw::wt_bank::instr_index( handle_t h, const char* instr_label )
const cw::wt_bank::wt_t* cw::wt_bank::get_wave_table( handle_t h, unsigned instr_idx, unsigned pitch, unsigned vel ) const cw::wt_bank::wt_t* cw::wt_bank::get_wave_table( handle_t h, unsigned instr_idx, unsigned pitch, unsigned vel )
{ {
rc_t rc = kOkRC;
wt_bank_t* p = _handleToPtr(h); wt_bank_t* p = _handleToPtr(h);
if( instr_idx >= p->wtMapN ) return _get_wave_table(p,instr_idx, pitch, vel);
{
rc = cwLogError(kInvalidArgRC,"Invalid instr_idx %i",instr_idx);
goto errLabel;
}
if( pitch >= midi::kMidiNoteCnt )
{
rc = cwLogError(kInvalidArgRC,"Invalid MIDI pitch %i",pitch);
goto errLabel;
}
if( vel >= midi::kMidiVelCnt )
{
rc = cwLogError(kInvalidArgRC,"Invalid MIDI pitch %i",pitch);
goto errLabel;
}
return p->wtMapA[ instr_idx ].map[ vel * midi::kMidiNoteCnt + pitch ];
errLabel:
cwLogError(rc,"Wave table lookup failed.");
return nullptr;
} }
cw::rc_t cw::wt_bank::test( const char* cfg_fname ) cw::rc_t cw::wt_bank::gen_notes( handle_t h, unsigned instr_idx, const unsigned* pitchA, const unsigned* velA, unsigned noteN, double note_dur_sec, const char* out_fname, double inter_note_gap_sec )
{
rc_t rc = kOkRC;
wt_bank_t* p = _handleToPtr(h);
const wt_t* wtA[ noteN ];
srate_t srate = 0;
unsigned outFrmN = 0;
unsigned noteSmpN = 0;
unsigned gapSmpN = 0;
unsigned yFrmIdx = 0;
unsigned chN = 0;
sample_t* outV = nullptr;
// Examine the wave table and determine the srate,audio ch. count, and output signal size.
for(unsigned i=0; i<noteN; ++i)
{
if((wtA[i] = _get_wave_table(p,instr_idx,pitchA[i],velA[i])) == nullptr )
{
rc = cwLogError(kInvalidArgRC,"The wave table at instr:%i pitch:%i vel:%i does not exist.",instr_idx,pitchA[i],velA[i]);
goto errLabel;
}
if( i==0 )
{
srate = wtA[i]->srate;
chN = wtA[i]->chN;
noteSmpN = (unsigned)(srate * note_dur_sec);
gapSmpN = (unsigned)(srate * inter_note_gap_sec);
}
else
{
assert( srate == wtA[i]->srate );
assert( chN == wtA[i]->chN );
}
printf("pitch:%i vel:%i s/cyc:%f\n", wtA[i]->pitch, wtA[i]->vel, srate/midi_to_hz(wtA[i]->pitch) );
for(unsigned j=0; j<wtA[i]->chN; ++j)
{
printf(" ch:%i\n",wtA[i]->chA[j].ch_idx);
for(unsigned k=0; k<wtA[i]->chA[j].segN; ++k)
{
printf(" %i aN:%i padN:%i\n",
wtA[i]->chA[j].segA[k].tid,
wtA[i]->chA[j].segA[k].aN,
wtA[i]->chA[j].segA[k].padN );
}
}
outFrmN += noteSmpN + gapSmpN;
}
if( outFrmN==0 || chN == 0 )
{
rc = cwLogError(kInvalidArgRC,"The sample rate:%f, output audio signal length (%i), or channel count (%i) is 0.",srate,outFrmN,chN);
goto errLabel;
}
else
{
sample_t* outChV[ chN ];
outV = mem::allocZ<sample_t>(outFrmN*chN);
// for each note
for(unsigned i=0; i<noteN; ++i)
{
// calc. the output sample frames for this note
unsigned yNoteFrmN = yFrmIdx+noteSmpN > outFrmN ? outFrmN-yFrmIdx : noteSmpN;
// load the output audio channel vector
for(unsigned ch_idx=0; ch_idx<chN; ++ch_idx)
{
outChV[ch_idx] = outV + ch_idx*outFrmN + yFrmIdx;
assert( yFrmIdx+yNoteFrmN < outFrmN );
}
// generate the note audio
if((rc = _gen_note( wtA[i], srate, outChV, yNoteFrmN )) != kOkRC )
{
rc = cwLogError(rc,"Note generation failed on instr:%i pitch:%i vel:%i",instr_idx,pitchA[i],velA[i]);
goto errLabel;
}
yFrmIdx += yNoteFrmN + gapSmpN;
}
for(unsigned i=0; i<chN; ++i)
outChV[i] = outV + i*outFrmN;
// write the output signal to an audio file
if((rc = audiofile::writeFileFloat(out_fname, srate, 32, outFrmN, chN, outChV )) != kOkRC )
{
rc = cwLogError(rc,"Audio file write failed on '%s'.",cwStringNullGuard(out_fname));
goto errLabel;
}
}
errLabel:
mem::release(outV);
return rc;
}
cw::rc_t cw::wt_bank::test( const test::test_args_t& args )
{ {
rc_t rc0 = kOkRC; rc_t rc0 = kOkRC;
rc_t rc1 = kOkRC; rc_t rc1 = kOkRC;
unsigned padN = 8; unsigned padN = 8;
const char* cfg_fname;
handle_t h; handle_t h;
unsigned instr_idx = 0;
unsigned pitchA[] = { 21,21,21,21,21,21,21,21,21,21,21,21,21,21,21,21,21,21,21, 21, 21, 21, 21, 21, 21 };
//unsigned pitchA[] = { 60,60,60,60,60,60,60,60,60,60,60,60,60,60,60,60,60,60,60, 60, 60, 60, 60, 60, 60 };
unsigned velA[] = { 1, 5,10,16,21,26,32,37,42,48,53,58,64,69,74,80,85,90,96,101,106,112,117,122,127 };
//unsigned velA[] = { 117, 122 };
double note_dur_sec = 2.5;
if((rc0 = args.test_args->getv("wtb_cfg_fname",cfg_fname)) != kOkRC )
goto errLabel;
if((rc0 = create(h,cfg_fname,padN)) != kOkRC ) if((rc0 = create(h,cfg_fname,padN)) != kOkRC )
goto errLabel; goto errLabel;
report(h);
assert( sizeof(pitchA)/sizeof(pitchA[0]) == sizeof(velA)/sizeof(velA[0]) );
gen_notes(h,instr_idx,pitchA,velA,sizeof(pitchA)/sizeof(pitchA[0]),note_dur_sec,"~/temp/temp.wav");
//report(h);
errLabel: errLabel:
if((rc1 = destroy(h)) != kOkRC ) if((rc1 = destroy(h)) != kOkRC )

View File

@ -11,13 +11,15 @@ namespace cw
typedef enum { typedef enum {
kAttackTId, kAttackTId,
kLoopTId kLoopTId,
kInvalidTId
} seg_tid_t; } seg_tid_t;
typedef struct seg_str typedef struct seg_str
{ {
seg_tid_t tid; seg_tid_t tid;
double cost; double cost;
unsigned cyc_per_loop; // count of cycles in the loop
sample_t* aV; // aV[ padN + aN + padN ] sample_t* aV; // aV[ padN + aN + padN ]
unsigned aN; // Count of unique samples unsigned aN; // Count of unique samples
unsigned padN; // Count of pre/post repeat samples unsigned padN; // Count of pre/post repeat samples
@ -56,7 +58,9 @@ namespace cw
const wt_t* get_wave_table( handle_t h, unsigned instr_idx, unsigned pitch, unsigned vel ); const wt_t* get_wave_table( handle_t h, unsigned instr_idx, unsigned pitch, unsigned vel );
rc_t test( const char* cfg_fname ); rc_t gen_notes( handle_t h, unsigned instr_idx, const unsigned* pitchA, const unsigned* velA, unsigned noteN, double dur_secs, const char* out_fname, double inter_note_gap_secs=0.1 );
rc_t test( const test::test_args_t& args );
} }